Coverage Report

Created: 2026-05-30 06:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/PROJ/src/projections/col_urban.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
PROJ_HEAD(col_urban, "Colombia Urban")
10
"\n\tMisc\n\th_0=";
11
12
// Notations and formulas taken from IOGP Publication 373-7-2 -
13
// Geomatics Guidance Note number 7, part 2 - March 2020
14
15
namespace { // anonymous namespace
16
17
struct pj_col_urban {
18
    double h0;   // height of projection origin, divided by semi-major axis (a)
19
    double rho0; // adimensional value, contrary to Guidance note 7.2
20
    double A;
21
    double B; // adimensional value, contrary to Guidance note 7.2
22
    double C;
23
    double D; // adimensional value, contrary to Guidance note 7.2
24
};
25
} // anonymous namespace
26
27
336
static PJ_XY col_urban_forward(PJ_LP lp, PJ *P) {
28
336
    PJ_XY xy;
29
336
    struct pj_col_urban *Q = static_cast<struct pj_col_urban *>(P->opaque);
30
31
336
    const double cosphi = cos(lp.phi);
32
336
    const double sinphi = sin(lp.phi);
33
336
    const double nu = 1. / sqrt(1 - P->es * sinphi * sinphi);
34
336
    const double lam_nu_cosphi = lp.lam * nu * cosphi;
35
336
    xy.x = Q->A * lam_nu_cosphi;
36
336
    const double sinphi_m = sin(0.5 * (lp.phi + P->phi0));
37
336
    const double rho_m =
38
336
        (1 - P->es) / pow(1 - P->es * sinphi_m * sinphi_m, 1.5);
39
336
    const double G = 1 + Q->h0 / rho_m;
40
336
    xy.y = G * Q->rho0 *
41
336
           ((lp.phi - P->phi0) + Q->B * lam_nu_cosphi * lam_nu_cosphi);
42
43
336
    return xy;
44
336
}
45
46
0
static PJ_LP col_urban_inverse(PJ_XY xy, PJ *P) {
47
0
    PJ_LP lp;
48
0
    struct pj_col_urban *Q = static_cast<struct pj_col_urban *>(P->opaque);
49
50
0
    lp.phi = P->phi0 + xy.y / Q->D - Q->B * (xy.x / Q->C) * (xy.x / Q->C);
51
0
    const double sinphi = sin(lp.phi);
52
0
    const double nu = 1. / sqrt(1 - P->es * sinphi * sinphi);
53
0
    lp.lam = xy.x / (Q->C * nu * cos(lp.phi));
54
55
0
    return lp;
56
0
}
57
58
33
PJ *PJ_PROJECTION(col_urban) {
59
33
    struct pj_col_urban *Q = static_cast<struct pj_col_urban *>(
60
33
        calloc(1, sizeof(struct pj_col_urban)));
61
33
    if (nullptr == Q)
62
0
        return pj_default_destructor(P, PROJ_ERR_OTHER /*ENOMEM*/);
63
33
    P->opaque = Q;
64
65
33
    const double h0_unscaled = pj_param(P->ctx, P->params, "dh_0").f;
66
33
    Q->h0 = h0_unscaled / P->a;
67
33
    const double sinphi0 = sin(P->phi0);
68
33
    const double nu0 = 1. / sqrt(1 - P->es * sinphi0 * sinphi0);
69
33
    Q->A = 1 + Q->h0 / nu0;
70
33
    Q->rho0 = (1 - P->es) / pow(1 - P->es * sinphi0 * sinphi0, 1.5);
71
33
    Q->B = tan(P->phi0) / (2 * Q->rho0 * nu0);
72
33
    Q->C = 1 + Q->h0;
73
33
    Q->D = Q->rho0 * (1 + Q->h0 / (1 - P->es));
74
75
33
    P->fwd = col_urban_forward;
76
33
    P->inv = col_urban_inverse;
77
78
33
    return P;
79
33
}