Coverage Report

Created: 2026-03-22 06:35

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/PROJ/src/projections/lcc.cpp
Line
Count
Source
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
4.59k
#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
3.10k
static PJ_XY lcc_e_forward(PJ_LP lp, PJ *P) { /* Ellipsoidal, forward */
23
3.10k
    PJ_XY xy = {0., 0.};
24
3.10k
    struct pj_lcc_data *Q = static_cast<struct pj_lcc_data *>(P->opaque);
25
3.10k
    double rho;
26
27
3.10k
    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
3.10k
    } else {
34
3.10k
        rho =
35
3.10k
            Q->c * (P->es != 0. ? pow(pj_tsfn(lp.phi, sin(lp.phi), P->e), Q->n)
36
3.10k
                                : pow(tan(M_FORTPI + .5 * lp.phi), -Q->n));
37
3.10k
    }
38
3.10k
    lp.lam *= Q->n;
39
3.10k
    xy.x = P->k0 * (rho * sin(lp.lam));
40
3.10k
    xy.y = P->k0 * (Q->rho0 - rho * cos(lp.lam));
41
3.10k
    return xy;
42
3.10k
}
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
264
PJ *PJ_PROJECTION(lcc) {
79
264
    double cosphi, sinphi;
80
264
    int secant;
81
264
    struct pj_lcc_data *Q = static_cast<struct pj_lcc_data *>(
82
264
        calloc(1, sizeof(struct pj_lcc_data)));
83
84
264
    if (nullptr == Q)
85
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
86
264
    P->opaque = Q;
87
88
264
    Q->phi1 = pj_param(P->ctx, P->params, "rlat_1").f;
89
264
    if (pj_param(P->ctx, P->params, "tlat_2").i)
90
169
        Q->phi2 = pj_param(P->ctx, P->params, "rlat_2").f;
91
95
    else {
92
95
        Q->phi2 = Q->phi1;
93
95
        if (!pj_param(P->ctx, P->params, "tlat_0").i)
94
81
            P->phi0 = Q->phi1;
95
95
    }
96
97
264
    if (fabs(Q->phi1 + Q->phi2) < EPS10) {
98
57
        proj_log_error(P, _("Invalid value for lat_1 and lat_2: |lat_1 + "
99
57
                            "lat_2| should be > 0"));
100
57
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
101
57
    }
102
103
207
    Q->n = sinphi = sin(Q->phi1);
104
207
    cosphi = cos(Q->phi1);
105
106
207
    if (fabs(cosphi) < EPS10 || fabs(Q->phi1) >= M_PI_2) {
107
3
        proj_log_error(P,
108
3
                       _("Invalid value for lat_1: |lat_1| should be < 90°"));
109
3
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
110
3
    }
111
204
    if (fabs(cos(Q->phi2)) < EPS10 || fabs(Q->phi2) >= M_PI_2) {
112
3
        proj_log_error(P,
113
3
                       _("Invalid value for lat_2: |lat_2| should be < 90°"));
114
3
        return pj_default_destructor(P, PROJ_ERR_INVALID_OP_ILLEGAL_ARG_VALUE);
115
3
    }
116
117
201
    secant = fabs(Q->phi1 - Q->phi2) >= EPS10;
118
201
    if (P->es != 0.) {
119
170
        double ml1, m1;
120
121
170
        m1 = pj_msfn(sinphi, cosphi, P->es);
122
170
        ml1 = pj_tsfn(Q->phi1, sinphi, P->e);
123
170
        if (secant) { /* secant cone */
124
73
            sinphi = sin(Q->phi2);
125
73
            Q->n = log(m1 / pj_msfn(sinphi, cos(Q->phi2), P->es));
126
73
            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
73
            const double ml2 = pj_tsfn(Q->phi2, sinphi, P->e);
133
73
            const double denom = log(ml1 / ml2);
134
73
            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
73
            Q->n /= denom;
141
73
        }
142
170
        Q->rho0 = m1 * pow(ml1, -Q->n) / Q->n;
143
170
        Q->c = Q->rho0;
144
170
        Q->rho0 *= (fabs(fabs(P->phi0) - M_HALFPI) < EPS10)
145
170
                       ? 0.
146
170
                       : pow(pj_tsfn(P->phi0, sin(P->phi0), P->e), Q->n);
147
170
    } else {
148
31
        if (secant)
149
12
            Q->n =
150
12
                log(cosphi / cos(Q->phi2)) / log(tan(M_FORTPI + .5 * Q->phi2) /
151
12
                                                 tan(M_FORTPI + .5 * Q->phi1));
152
31
        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
31
        Q->c = cosphi * pow(tan(M_FORTPI + .5 * Q->phi1), Q->n) / Q->n;
162
31
        Q->rho0 = (fabs(fabs(P->phi0) - M_HALFPI) < EPS10)
163
31
                      ? 0.
164
31
                      : Q->c * pow(tan(M_FORTPI + .5 * P->phi0), -Q->n);
165
31
    }
166
167
201
    P->inv = lcc_e_inverse;
168
201
    P->fwd = lcc_e_forward;
169
170
201
    return P;
171
201
}
172
173
#undef EPS10