aboutsummaryrefslogtreecommitdiff
path: root/src/nad_cvt.cpp
blob: 76d2ccd0f8f6ff71bc223098177fa346be342d89 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
#define PJ_LIB__

#include <stdio.h>
#include <stdlib.h>

#include "proj.h"
#include "proj_internal.h"
#include "proj_math.h"

#define MAX_ITERATIONS 10
#define TOL 1e-12

PJ_LP nad_cvt(PJ_LP in, int inverse, struct CTABLE *ct) {
    PJ_LP t, tb,del, dif;
    int i = MAX_ITERATIONS;
    const double toltol = TOL*TOL;

    if (in.lam == HUGE_VAL)
        return in;

    /* normalize input to ll origin */
    tb = in;
    tb.lam -= ct->ll.lam;
    tb.phi -= ct->ll.phi;
    tb.lam = adjlon (tb.lam - M_PI) + M_PI;

    t = nad_intr (tb, ct);
    if (t.lam == HUGE_VAL)
        return t;

    if (!inverse) {
        in.lam -= t.lam;
        in.phi += t.phi;
        return in;
    }

    t.lam = tb.lam + t.lam;
    t.phi = tb.phi - t.phi;

    do {
        del = nad_intr(t, ct);

        /* This case used to return failure, but I have
           changed it to return the first order approximation
           of the inverse shift.  This avoids cases where the
           grid shift *into* this grid came from another grid.
           While we aren't returning optimally correct results
           I feel a close result in this case is better than
           no result.  NFW
           To demonstrate use -112.5839956 49.4914451 against
           the NTv2 grid shift file from Canada. */
        if (del.lam == HUGE_VAL)
            break;

        dif.lam = t.lam - del.lam - tb.lam;
           dif.phi = t.phi + del.phi - tb.phi;
        t.lam -= dif.lam;
        t.phi -= dif.phi;

    } while (--i && (dif.lam*dif.lam + dif.phi*dif.phi > toltol)); /* prob. slightly faster than hypot() */

    if (i==0) {
        /* If we had access to a context, this should go through pj_log, and we should set ctx->errno */
        if (getenv ("PROJ_DEBUG"))
            fprintf( stderr, "Inverse grid shift iterator failed to converge.\n" );
        t.lam = t.phi = HUGE_VAL;
        return t;
    }

    /* and again: pj_log and ctx->errno */
    if (del.lam==HUGE_VAL && getenv ("PROJ_DEBUG"))
        fprintf (stderr, "Inverse grid shift iteration failed, presumably at grid edge.\nUsing first approximation.\n");

    in.lam = adjlon (t.lam + ct->ll.lam);
    in.phi = t.phi + ct->ll.phi;
    return in;
}