aboutsummaryrefslogtreecommitdiff
path: root/src/apps/geod_set.cpp
blob: d6516f22b2b0c44a65ac4ca332d6d912156e76d2 (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
78
79
80
#define _IN_GEOD_SET

#include <math.h>
#include <stdlib.h>
#include <string.h>

#include "proj.h"
#include "proj_internal.h"
#include "geod_interface.h"
#include "emess.h"

	void
geod_set(int argc, char **argv) {
	paralist *start = nullptr, *curr;
	double es;
	char *name;

    /* put arguments into internal linked list */
	if (argc <= 0)
		emess(1, "no arguments in initialization list");
	start = curr = pj_mkparam(argv[0]);
	if (!curr)
		emess(1, "memory allocation failed");
	for (int i = 1; curr != nullptr && i < argc; ++i) {
		curr->next = pj_mkparam(argv[i]);
		if (!curr->next)
			emess(1, "memory allocation failed");
		curr = curr->next;
	}
	/* set elliptical parameters */
	if (pj_ell_set(pj_get_default_ctx(),start, &geod_a, &es)) emess(1,"ellipse setup failure");
	/* set units */
	if ((name = pj_param(nullptr,start, "sunits").s) != nullptr) {
            bool unit_found = false;
            auto units = proj_get_units_from_database(nullptr, nullptr, "linear", false, nullptr);
            for( int i = 0; units && units[i]; i++ )
            {
                if( units[i]->proj_short_name &&
                    strcmp(units[i]->proj_short_name, name) == 0 ) {
                    unit_found = true;
                    to_meter = units[i]->conv_factor;
                    fr_meter = 1 / to_meter;
                }
            }
            proj_unit_list_destroy(units);
            if( !unit_found )
                emess(1,"%s unknown unit conversion id", name);
	} else
		to_meter = fr_meter = 1;
	geod_f = es/(1 + sqrt(1 - es));
	geod_ini();
	/* check if line or arc mode */
	if (pj_param(nullptr,start, "tlat_1").i) {
		double del_S;
		phi1 = pj_param(nullptr,start, "rlat_1").f;
		lam1 = pj_param(nullptr,start, "rlon_1").f;
		if (pj_param(nullptr,start, "tlat_2").i) {
			phi2 = pj_param(nullptr,start, "rlat_2").f;
			lam2 = pj_param(nullptr,start, "rlon_2").f;
			geod_inv();
			geod_pre();
		} else if ((geod_S = pj_param(nullptr,start, "dS").f) != 0.) {
			al12 = pj_param(nullptr,start, "rA").f;
			geod_pre();
			geod_for();
		} else emess(1,"incomplete geodesic/arc info");
		if ((n_alpha = pj_param(nullptr,start, "in_A").i) > 0) {
			if ((del_alpha = pj_param(nullptr,start, "rdel_A").f) == 0.0)
				emess(1,"del azimuth == 0");
		} else if ((del_S = fabs(pj_param(nullptr,start, "ddel_S").f)) != 0.) {
			n_S = (int)(geod_S / del_S + .5);
		} else if ((n_S = pj_param(nullptr,start, "in_S").i) <= 0)
			emess(1,"no interval divisor selected");
	}
	/* free up linked list */
	for ( ; start; start = curr) {
		curr = start->next;
		free(start);
	}
}