Coverage Report

Created: 2025-11-24 06:58

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/PROJ/src/projections/hammer.cpp
Line
Count
Source
1
2
3
#include <errno.h>
4
#include <math.h>
5
6
#include "proj.h"
7
#include "proj_internal.h"
8
9
PROJ_HEAD(hammer, "Hammer & Eckert-Greifendorff")
10
"\n\tMisc Sph, \n\tW= M=";
11
12
0
#define EPS 1.0e-10
13
14
namespace { // anonymous namespace
15
struct pq_hammer {
16
    double w;
17
    double m, rm;
18
};
19
} // anonymous namespace
20
21
0
static PJ_XY hammer_s_forward(PJ_LP lp, PJ *P) { /* Spheroidal, forward */
22
0
    PJ_XY xy = {0.0, 0.0};
23
0
    struct pq_hammer *Q = static_cast<struct pq_hammer *>(P->opaque);
24
0
    double cosphi, d;
25
26
0
    cosphi = cos(lp.phi);
27
0
    lp.lam *= Q->w;
28
0
    double denom = 1. + cosphi * cos(lp.lam);
29
0
    if (denom == 0.0) {
30
0
        proj_errno_set(P, PROJ_ERR_COORD_TRANSFM_OUTSIDE_PROJECTION_DOMAIN);
31
0
        return proj_coord_error().xy;
32
0
    }
33
0
    d = sqrt(2. / denom);
34
0
    xy.x = Q->m * d * cosphi * sin(lp.lam);
35
0
    xy.y = Q->rm * d * sin(lp.phi);
36
0
    return xy;
37
0
}
38
39
0
static PJ_LP hammer_s_inverse(PJ_XY xy, PJ *P) { /* Spheroidal, inverse */
40
0
    PJ_LP lp = {0.0, 0.0};
41
0
    struct pq_hammer *Q = static_cast<struct pq_hammer *>(P->opaque);
42
0
    double z;
43
44
0
    z = sqrt(1. - 0.25 * Q->w * Q->w * xy.x * xy.x - 0.25 * xy.y * xy.y);
45
0
    if (fabs(2. * z * z - 1.) < EPS) {
46
0
        lp.lam = HUGE_VAL;
47
0
        lp.phi = HUGE_VAL;
48
0
        proj_errno_set(P, PROJ_ERR_COORD_TRANSFM_OUTSIDE_PROJECTION_DOMAIN);
49
0
    } else {
50
0
        lp.lam = aatan2(Q->w * xy.x * z, 2. * z * z - 1) / Q->w;
51
0
        lp.phi = aasin(P->ctx, z * xy.y);
52
0
    }
53
0
    return lp;
54
0
}
55
56
27
PJ *PJ_PROJECTION(hammer) {
57
27
    struct pq_hammer *Q =
58
27
        static_cast<struct pq_hammer *>(calloc(1, sizeof(struct pq_hammer)));
59
27
    if (nullptr == Q)
60
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
61
27
    P->opaque = Q;
62
63
27
    if (pj_param(P->ctx, P->params, "tW").i) {
64
9
        Q->w = fabs(pj_param(P->ctx, P->params, "dW").f);
65
9
        if (Q->w <= 0.) {
66
2
            proj_log_error(P, _("Invalid value for W: it should be > 0"));
67
2
            return pj_default_destructor(P,
68
2
                                         PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
69
2
        }
70
9
    } else
71
18
        Q->w = .5;
72
25
    if (pj_param(P->ctx, P->params, "tM").i) {
73
11
        Q->m = fabs(pj_param(P->ctx, P->params, "dM").f);
74
11
        if (Q->m <= 0.) {
75
0
            proj_log_error(P, _("Invalid value for M: it should be > 0"));
76
0
            return pj_default_destructor(P,
77
0
                                         PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
78
0
        }
79
11
    } else
80
14
        Q->m = 1.;
81
82
25
    Q->rm = 1. / Q->m;
83
25
    Q->m /= Q->w;
84
85
25
    P->es = 0.;
86
25
    P->fwd = hammer_s_forward;
87
25
    P->inv = hammer_s_inverse;
88
89
25
    return P;
90
25
}
91
92
#undef EPS