Coverage Report

Created: 2025-10-10 06:31

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/PROJ/src/projections/putp6.cpp
Line
Count
Source
1
2
3
#include <errno.h>
4
#include <math.h>
5
6
#include "proj.h"
7
#include "proj_internal.h"
8
9
namespace { // anonymous namespace
10
struct pj_putp6 {
11
    double C_x, C_y, A, B, D;
12
};
13
} // anonymous namespace
14
15
PROJ_HEAD(putp6, "Putnins P6") "\n\tPCyl, Sph";
16
PROJ_HEAD(putp6p, "Putnins P6'") "\n\tPCyl, Sph";
17
18
120k
#define EPS 1e-10
19
27.9k
#define NITER 10
20
0
#define CON_POLE 1.732050807568877 /* sqrt(3) */
21
22
27.9k
static PJ_XY putp6_s_forward(PJ_LP lp, PJ *P) { /* Spheroidal, forward */
23
27.9k
    PJ_XY xy = {0.0, 0.0};
24
27.9k
    struct pj_putp6 *Q = static_cast<struct pj_putp6 *>(P->opaque);
25
27.9k
    int i;
26
27
27.9k
    const double p = Q->B * sin(lp.phi);
28
27.9k
    lp.phi *= 1.10265779;
29
120k
    for (i = NITER; i; --i) {
30
120k
        const double r = sqrt(1. + lp.phi * lp.phi);
31
120k
        const double V =
32
120k
            ((Q->A - r) * lp.phi - log(lp.phi + r) - p) / (Q->A - 2. * r);
33
120k
        lp.phi -= V;
34
120k
        if (fabs(V) < EPS)
35
27.9k
            break;
36
120k
    }
37
27.9k
    double sqrt_1_plus_phi2;
38
27.9k
    if (!i) {
39
        // Note: it seems this case is rarely reached as from experimenting,
40
        // i seems to be >= 6
41
0
        lp.phi = p < 0. ? -CON_POLE : CON_POLE;
42
        // the formula of the else case would also work, but this makes
43
        // some cppcheck versions happier.
44
0
        sqrt_1_plus_phi2 = 2;
45
27.9k
    } else {
46
27.9k
        sqrt_1_plus_phi2 = sqrt(1. + lp.phi * lp.phi);
47
27.9k
    }
48
27.9k
    xy.x = Q->C_x * lp.lam * (Q->D - sqrt_1_plus_phi2);
49
27.9k
    xy.y = Q->C_y * lp.phi;
50
51
27.9k
    return xy;
52
27.9k
}
53
54
0
static PJ_LP putp6_s_inverse(PJ_XY xy, PJ *P) { /* Spheroidal, inverse */
55
0
    PJ_LP lp = {0.0, 0.0};
56
0
    struct pj_putp6 *Q = static_cast<struct pj_putp6 *>(P->opaque);
57
0
    double r;
58
59
0
    lp.phi = xy.y / Q->C_y;
60
0
    r = sqrt(1. + lp.phi * lp.phi);
61
0
    lp.lam = xy.x / (Q->C_x * (Q->D - r));
62
0
    lp.phi = aasin(P->ctx, ((Q->A - r) * lp.phi - log(lp.phi + r)) / Q->B);
63
64
0
    return lp;
65
0
}
66
67
17
PJ *PJ_PROJECTION(putp6) {
68
17
    struct pj_putp6 *Q =
69
17
        static_cast<struct pj_putp6 *>(calloc(1, sizeof(struct pj_putp6)));
70
17
    if (nullptr == Q)
71
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
72
17
    P->opaque = Q;
73
74
17
    Q->C_x = 1.01346;
75
17
    Q->C_y = 0.91910;
76
17
    Q->A = 4.;
77
17
    Q->B = 2.1471437182129378784;
78
17
    Q->D = 2.;
79
80
17
    P->es = 0.;
81
17
    P->inv = putp6_s_inverse;
82
17
    P->fwd = putp6_s_forward;
83
84
17
    return P;
85
17
}
86
87
344
PJ *PJ_PROJECTION(putp6p) {
88
344
    struct pj_putp6 *Q =
89
344
        static_cast<struct pj_putp6 *>(calloc(1, sizeof(struct pj_putp6)));
90
344
    if (nullptr == Q)
91
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
92
344
    P->opaque = Q;
93
94
344
    Q->C_x = 0.44329;
95
344
    Q->C_y = 0.80404;
96
344
    Q->A = 6.;
97
344
    Q->B = 5.61125;
98
344
    Q->D = 3.;
99
100
344
    P->es = 0.;
101
344
    P->inv = putp6_s_inverse;
102
344
    P->fwd = putp6_s_forward;
103
104
344
    return P;
105
344
}
106
107
#undef EPS
108
#undef NITER
109
#undef CON_POLE