/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 |