Coverage Report

Created: 2025-07-11 06:33

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