/src/PROJ/src/projections/rouss.cpp
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | ** libproj -- library of cartographic projections |
3 | | ** |
4 | | ** Copyright (c) 2003, 2006 Gerald I. Evenden |
5 | | */ |
6 | | /* |
7 | | ** Permission is hereby granted, free of charge, to any person obtaining |
8 | | ** a copy of this software and associated documentation files (the |
9 | | ** "Software"), to deal in the Software without restriction, including |
10 | | ** without limitation the rights to use, copy, modify, merge, publish, |
11 | | ** distribute, sublicense, and/or sell copies of the Software, and to |
12 | | ** permit persons to whom the Software is furnished to do so, subject to |
13 | | ** the following conditions: |
14 | | ** |
15 | | ** The above copyright notice and this permission notice shall be |
16 | | ** included in all copies or substantial portions of the Software. |
17 | | ** |
18 | | ** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
19 | | ** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
20 | | ** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. |
21 | | ** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY |
22 | | ** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, |
23 | | ** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE |
24 | | ** SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
25 | | */ |
26 | | |
27 | | #include <errno.h> |
28 | | #include <math.h> |
29 | | |
30 | | #include "proj.h" |
31 | | #include "proj_internal.h" |
32 | | |
33 | | namespace { // anonymous namespace |
34 | | struct pj_rouss_data { |
35 | | double s0; |
36 | | double A1, A2, A3, A4, A5, A6; |
37 | | double B1, B2, B3, B4, B5, B6, B7, B8; |
38 | | double C1, C2, C3, C4, C5, C6, C7, C8; |
39 | | double D1, D2, D3, D4, D5, D6, D7, D8, D9, D10, D11; |
40 | | void *en; |
41 | | }; |
42 | | } // anonymous namespace |
43 | | PROJ_HEAD(rouss, "Roussilhe Stereographic") "\n\tAzi, Ell"; |
44 | | |
45 | 0 | static PJ_XY rouss_e_forward(PJ_LP lp, PJ *P) { /* Ellipsoidal, forward */ |
46 | 0 | PJ_XY xy = {0.0, 0.0}; |
47 | 0 | struct pj_rouss_data *Q = static_cast<struct pj_rouss_data *>(P->opaque); |
48 | 0 | double s, al, cp, sp, al2, s2; |
49 | |
|
50 | 0 | cp = cos(lp.phi); |
51 | 0 | sp = sin(lp.phi); |
52 | 0 | s = proj_mdist(lp.phi, sp, cp, Q->en) - Q->s0; |
53 | 0 | s2 = s * s; |
54 | 0 | al = lp.lam * cp / sqrt(1. - P->es * sp * sp); |
55 | 0 | al2 = al * al; |
56 | 0 | xy.x = P->k0 * al * |
57 | 0 | (1. + s2 * (Q->A1 + s2 * Q->A4) - |
58 | 0 | al2 * (Q->A2 + s * Q->A3 + s2 * Q->A5 + al2 * Q->A6)); |
59 | 0 | xy.y = P->k0 * |
60 | 0 | (al2 * (Q->B1 + al2 * Q->B4) + |
61 | 0 | s * (1. + al2 * (Q->B3 - al2 * Q->B6) + s2 * (Q->B2 + s2 * Q->B8) + |
62 | 0 | s * al2 * (Q->B5 + s * Q->B7))); |
63 | |
|
64 | 0 | return xy; |
65 | 0 | } |
66 | | |
67 | 0 | static PJ_LP rouss_e_inverse(PJ_XY xy, PJ *P) { /* Ellipsoidal, inverse */ |
68 | 0 | PJ_LP lp = {0.0, 0.0}; |
69 | 0 | struct pj_rouss_data *Q = static_cast<struct pj_rouss_data *>(P->opaque); |
70 | 0 | double s, al, x = xy.x / P->k0, y = xy.y / P->k0, x2, y2; |
71 | |
|
72 | 0 | x2 = x * x; |
73 | 0 | y2 = y * y; |
74 | 0 | al = x * |
75 | 0 | (1. - Q->C1 * y2 + |
76 | 0 | x2 * (Q->C2 + Q->C3 * y - Q->C4 * x2 + Q->C5 * y2 - Q->C7 * x2 * y) + |
77 | 0 | y2 * (Q->C6 * y2 - Q->C8 * x2 * y)); |
78 | 0 | s = Q->s0 + y * (1. + y2 * (-Q->D2 + Q->D8 * y2)) + |
79 | 0 | x2 * (-Q->D1 + y * (-Q->D3 + y * (-Q->D5 + y * (-Q->D7 + y * Q->D11))) + |
80 | 0 | x2 * (Q->D4 + y * (Q->D6 + y * Q->D10) - x2 * Q->D9)); |
81 | 0 | lp.phi = proj_inv_mdist(P->ctx, s, Q->en); |
82 | 0 | s = sin(lp.phi); |
83 | 0 | lp.lam = al * sqrt(1. - P->es * s * s) / cos(lp.phi); |
84 | |
|
85 | 0 | return lp; |
86 | 0 | } |
87 | | |
88 | 141 | static PJ *pj_rouss_destructor(PJ *P, int errlev) { |
89 | 141 | if (nullptr == P) |
90 | 0 | return nullptr; |
91 | | |
92 | 141 | if (nullptr == P->opaque) |
93 | 0 | return pj_default_destructor(P, errlev); |
94 | | |
95 | 141 | if (static_cast<struct pj_rouss_data *>(P->opaque)->en) |
96 | 141 | free(static_cast<struct pj_rouss_data *>(P->opaque)->en); |
97 | | |
98 | 141 | return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/); |
99 | 141 | } |
100 | | |
101 | 141 | PJ *PJ_PROJECTION(rouss) { |
102 | 141 | double N0, es2, t, t2, R_R0_2, R_R0_4; |
103 | | |
104 | 141 | struct pj_rouss_data *Q = static_cast<struct pj_rouss_data *>( |
105 | 141 | calloc(1, sizeof(struct pj_rouss_data))); |
106 | 141 | if (nullptr == Q) |
107 | 0 | return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/); |
108 | 141 | P->opaque = Q; |
109 | | |
110 | 141 | if (!((Q->en = proj_mdist_ini(P->es)))) |
111 | 0 | return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/); |
112 | | |
113 | 141 | es2 = sin(P->phi0); |
114 | 141 | Q->s0 = proj_mdist(P->phi0, es2, cos(P->phi0), Q->en); |
115 | 141 | t = 1. - (es2 = P->es * es2 * es2); |
116 | 141 | N0 = 1. / sqrt(t); |
117 | 141 | R_R0_2 = t * t / P->one_es; |
118 | 141 | R_R0_4 = R_R0_2 * R_R0_2; |
119 | 141 | t = tan(P->phi0); |
120 | 141 | t2 = t * t; |
121 | 141 | Q->C1 = Q->A1 = R_R0_2 / 4.; |
122 | 141 | Q->C2 = Q->A2 = R_R0_2 * (2 * t2 - 1. - 2. * es2) / 12.; |
123 | 141 | Q->A3 = R_R0_2 * t * (1. + 4. * t2) / (12. * N0); |
124 | 141 | Q->A4 = R_R0_4 / 24.; |
125 | 141 | Q->A5 = R_R0_4 * (-1. + t2 * (11. + 12. * t2)) / 24.; |
126 | 141 | Q->A6 = R_R0_4 * (-2. + t2 * (11. - 2. * t2)) / 240.; |
127 | 141 | Q->B1 = t / (2. * N0); |
128 | 141 | Q->B2 = R_R0_2 / 12.; |
129 | 141 | Q->B3 = R_R0_2 * (1. + 2. * t2 - 2. * es2) / 4.; |
130 | 141 | Q->B4 = R_R0_2 * t * (2. - t2) / (24. * N0); |
131 | 141 | Q->B5 = R_R0_2 * t * (5. + 4. * t2) / (8. * N0); |
132 | 141 | Q->B6 = R_R0_4 * (-2. + t2 * (-5. + 6. * t2)) / 48.; |
133 | 141 | Q->B7 = R_R0_4 * (5. + t2 * (19. + 12. * t2)) / 24.; |
134 | 141 | Q->B8 = R_R0_4 / 120.; |
135 | 141 | Q->C3 = R_R0_2 * t * (1. + t2) / (3. * N0); |
136 | 141 | Q->C4 = R_R0_4 * (-3. + t2 * (34. + 22. * t2)) / 240.; |
137 | 141 | Q->C5 = R_R0_4 * (4. + t2 * (13. + 12. * t2)) / 24.; |
138 | 141 | Q->C6 = R_R0_4 / 16.; |
139 | 141 | Q->C7 = R_R0_4 * t * (11. + t2 * (33. + t2 * 16.)) / (48. * N0); |
140 | 141 | Q->C8 = R_R0_4 * t * (1. + t2 * 4.) / (36. * N0); |
141 | 141 | Q->D1 = t / (2. * N0); |
142 | 141 | Q->D2 = R_R0_2 / 12.; |
143 | 141 | Q->D3 = R_R0_2 * (2 * t2 + 1. - 2. * es2) / 4.; |
144 | 141 | Q->D4 = R_R0_2 * t * (1. + t2) / (8. * N0); |
145 | 141 | Q->D5 = R_R0_2 * t * (1. + t2 * 2.) / (4. * N0); |
146 | 141 | Q->D6 = R_R0_4 * (1. + t2 * (6. + t2 * 6.)) / 16.; |
147 | 141 | Q->D7 = R_R0_4 * t2 * (3. + t2 * 4.) / 8.; |
148 | 141 | Q->D8 = R_R0_4 / 80.; |
149 | 141 | Q->D9 = R_R0_4 * t * (-21. + t2 * (178. - t2 * 26.)) / 720.; |
150 | 141 | Q->D10 = R_R0_4 * t * (29. + t2 * (86. + t2 * 48.)) / (96. * N0); |
151 | 141 | Q->D11 = R_R0_4 * t * (37. + t2 * 44.) / (96. * N0); |
152 | | |
153 | 141 | P->fwd = rouss_e_forward; |
154 | 141 | P->inv = rouss_e_inverse; |
155 | 141 | P->destructor = pj_rouss_destructor; |
156 | | |
157 | 141 | return P; |
158 | 141 | } |