Coverage Report

Created: 2025-06-13 06:18

/src/proj/src/projections/lcc.cpp
Line
Count
Source (jump to first uncovered line)
1
2
#include "proj.h"
3
#include "proj_internal.h"
4
#include <errno.h>
5
#include <math.h>
6
7
PROJ_HEAD(lcc, "Lambert Conformal Conic")
8
"\n\tConic, Sph&Ell\n\tlat_1= and lat_2= or lat_0, k_0=";
9
10
0
#define EPS10 1.e-10
11
12
namespace { // anonymous namespace
13
struct pj_lcc_data {
14
    double phi1;
15
    double phi2;
16
    double n;
17
    double rho0;
18
    double c;
19
};
20
} // anonymous namespace
21
22
0
static PJ_XY lcc_e_forward(PJ_LP lp, PJ *P) { /* Ellipsoidal, forward */
23
0
    PJ_XY xy = {0., 0.};
24
0
    struct pj_lcc_data *Q = static_cast<struct pj_lcc_data *>(P->opaque);
25
0
    double rho;
26
27
0
    if (fabs(fabs(lp.phi) - M_HALFPI) < EPS10) {
28
0
        if ((lp.phi * Q->n) <= 0.) {
29
0
            proj_errno_set(P, PROJ_ERR_COORD_TRANSFM_OUTSIDE_PROJECTION_DOMAIN);
30
0
            return xy;
31
0
        }
32
0
        rho = 0.;
33
0
    } else {
34
0
        rho =
35
0
            Q->c * (P->es != 0. ? pow(pj_tsfn(lp.phi, sin(lp.phi), P->e), Q->n)
36
0
                                : pow(tan(M_FORTPI + .5 * lp.phi), -Q->n));
37
0
    }
38
0
    lp.lam *= Q->n;
39
0
    xy.x = P->k0 * (rho * sin(lp.lam));
40
0
    xy.y = P->k0 * (Q->rho0 - rho * cos(lp.lam));
41
0
    return xy;
42
0
}
43
44
0
static PJ_LP lcc_e_inverse(PJ_XY xy, PJ *P) { /* Ellipsoidal, inverse */
45
0
    PJ_LP lp = {0., 0.};
46
0
    struct pj_lcc_data *Q = static_cast<struct pj_lcc_data *>(P->opaque);
47
0
    double rho;
48
49
0
    xy.x /= P->k0;
50
0
    xy.y /= P->k0;
51
52
0
    xy.y = Q->rho0 - xy.y;
53
0
    rho = hypot(xy.x, xy.y);
54
0
    if (rho != 0.) {
55
0
        if (Q->n < 0.) {
56
0
            rho = -rho;
57
0
            xy.x = -xy.x;
58
0
            xy.y = -xy.y;
59
0
        }
60
0
        if (P->es != 0.) {
61
0
            lp.phi = pj_phi2(P->ctx, pow(rho / Q->c, 1. / Q->n), P->e);
62
0
            if (lp.phi == HUGE_VAL) {
63
0
                proj_errno_set(
64
0
                    P, PROJ_ERR_COORD_TRANSFM_OUTSIDE_PROJECTION_DOMAIN);
65
0
                return lp;
66
0
            }
67
68
0
        } else
69
0
            lp.phi = 2. * atan(pow(Q->c / rho, 1. / Q->n)) - M_HALFPI;
70
0
        lp.lam = atan2(xy.x, xy.y) / Q->n;
71
0
    } else {
72
0
        lp.lam = 0.;
73
0
        lp.phi = Q->n > 0. ? M_HALFPI : -M_HALFPI;
74
0
    }
75
0
    return lp;
76
0
}
77
78
0
PJ *PJ_PROJECTION(lcc) {
79
0
    double cosphi, sinphi;
80
0
    int secant;
81
0
    struct pj_lcc_data *Q = static_cast<struct pj_lcc_data *>(
82
0
        calloc(1, sizeof(struct pj_lcc_data)));
83
84
0
    if (nullptr == Q)
85
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
86
0
    P->opaque = Q;
87
88
0
    Q->phi1 = pj_param(P->ctx, P->params, "rlat_1").f;
89
0
    if (pj_param(P->ctx, P->params, "tlat_2").i)
90
0
        Q->phi2 = pj_param(P->ctx, P->params, "rlat_2").f;
91
0
    else {
92
0
        Q->phi2 = Q->phi1;
93
0
        if (!pj_param(P->ctx, P->params, "tlat_0").i)
94
0
            P->phi0 = Q->phi1;
95
0
    }
96
97
0
    if (fabs(Q->phi1 + Q->phi2) < EPS10) {
98
0
        proj_log_error(P, _("Invalid value for lat_1 and lat_2: |lat_1 + "
99
0
                            "lat_2| should be > 0"));
100
0
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
101
0
    }
102
103
0
    Q->n = sinphi = sin(Q->phi1);
104
0
    cosphi = cos(Q->phi1);
105
106
0
    if (fabs(cosphi) < EPS10 || fabs(Q->phi1) >= M_PI_2) {
107
0
        proj_log_error(P,
108
0
                       _("Invalid value for lat_1: |lat_1| should be < 90°"));
109
0
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
110
0
    }
111
0
    if (fabs(cos(Q->phi2)) < EPS10 || fabs(Q->phi2) >= M_PI_2) {
112
0
        proj_log_error(P,
113
0
                       _("Invalid value for lat_2: |lat_2| should be < 90°"));
114
0
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
115
0
    }
116
117
0
    secant = fabs(Q->phi1 - Q->phi2) >= EPS10;
118
0
    if (P->es != 0.) {
119
0
        double ml1, m1;
120
121
0
        m1 = pj_msfn(sinphi, cosphi, P->es);
122
0
        ml1 = pj_tsfn(Q->phi1, sinphi, P->e);
123
0
        if (secant) { /* secant cone */
124
0
            sinphi = sin(Q->phi2);
125
0
            Q->n = log(m1 / pj_msfn(sinphi, cos(Q->phi2), P->es));
126
0
            if (Q->n == 0) {
127
                // Not quite, but es is very close to 1...
128
0
                proj_log_error(P, _("Invalid value for eccentricity"));
129
0
                return pj_default_destructor(
130
0
                    P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
131
0
            }
132
0
            const double ml2 = pj_tsfn(Q->phi2, sinphi, P->e);
133
0
            const double denom = log(ml1 / ml2);
134
0
            if (denom == 0) {
135
                // Not quite, but es is very close to 1...
136
0
                proj_log_error(P, _("Invalid value for eccentricity"));
137
0
                return pj_default_destructor(
138
0
                    P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
139
0
            }
140
0
            Q->n /= denom;
141
0
        }
142
0
        Q->rho0 = m1 * pow(ml1, -Q->n) / Q->n;
143
0
        Q->c = Q->rho0;
144
0
        Q->rho0 *= (fabs(fabs(P->phi0) - M_HALFPI) < EPS10)
145
0
                       ? 0.
146
0
                       : pow(pj_tsfn(P->phi0, sin(P->phi0), P->e), Q->n);
147
0
    } else {
148
0
        if (secant)
149
0
            Q->n =
150
0
                log(cosphi / cos(Q->phi2)) / log(tan(M_FORTPI + .5 * Q->phi2) /
151
0
                                                 tan(M_FORTPI + .5 * Q->phi1));
152
0
        if (Q->n == 0) {
153
            // Likely reason is that phi1 / phi2 are too close to zero.
154
            // Can be reproduced with +proj=lcc +a=1 +lat_2=.0000001
155
0
            proj_log_error(
156
0
                P, _("Invalid value for lat_1 and lat_2: |lat_1 + lat_2| "
157
0
                     "should be > 0"));
158
0
            return pj_default_destructor(P,
159
0
                                         PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
160
0
        }
161
0
        Q->c = cosphi * pow(tan(M_FORTPI + .5 * Q->phi1), Q->n) / Q->n;
162
0
        Q->rho0 = (fabs(fabs(P->phi0) - M_HALFPI) < EPS10)
163
0
                      ? 0.
164
0
                      : Q->c * pow(tan(M_FORTPI + .5 * P->phi0), -Q->n);
165
0
    }
166
167
0
    P->inv = lcc_e_inverse;
168
0
    P->fwd = lcc_e_forward;
169
170
0
    return P;
171
0
}
172
173
#undef EPS10