proj4r 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/API.md +111 -0
- data/README.md +23 -0
- data/extconf.rb +12 -0
- data/lib/proj4r.rb +184 -0
- data/proj4r.gemspec +22 -0
- data/rb_geod.c +392 -0
- data/rb_proj.c +357 -0
- data/rb_proj4r.c +232 -0
- data/rb_proj4r.h +46 -0
- data/test/TEST.txt +5 -0
- data/test/attributes.rb +13 -0
- data/test/tenkizu.rb +29 -0
- data/test/test.rb +25 -0
- data/test/test_ca.rb +17 -0
- data/test/test_geod_ca.rb +25 -0
- data/test/test_proj.rb +14 -0
- data/test/test_proj_ca.rb +15 -0
- metadata +60 -0
data/rb_geod.c
ADDED
@@ -0,0 +1,392 @@
|
|
1
|
+
#include "ruby.h"
|
2
|
+
#include "rb_proj4r.h"
|
3
|
+
|
4
|
+
#define MAX_PARGS 50
|
5
|
+
|
6
|
+
VALUE rb_cGeod;
|
7
|
+
|
8
|
+
static void
|
9
|
+
free_geod (Geod *geod)
|
10
|
+
{
|
11
|
+
if ( geod->ref ) {
|
12
|
+
free(geod->ref);
|
13
|
+
}
|
14
|
+
free(geod);
|
15
|
+
}
|
16
|
+
|
17
|
+
static VALUE
|
18
|
+
rb_geod_s_allocate (VALUE klass)
|
19
|
+
{
|
20
|
+
Geod *geod;
|
21
|
+
return Data_Make_Struct(klass, Geod, 0, free_geod, geod);
|
22
|
+
}
|
23
|
+
|
24
|
+
|
25
|
+
/*
|
26
|
+
* call-seq:
|
27
|
+
* Geod.new(a = 6378137, f = 1.0/298.257223563) -> nil
|
28
|
+
*
|
29
|
+
* Sets parameters for geodesic computations.
|
30
|
+
* See manual of 'geod' for the details of paramters.
|
31
|
+
*
|
32
|
+
* Geod.init("+ellps=WGS84 +units=km")
|
33
|
+
*
|
34
|
+
*/
|
35
|
+
|
36
|
+
|
37
|
+
static VALUE
|
38
|
+
rb_geod_initialize (int argc, VALUE *argv, VALUE self)
|
39
|
+
{
|
40
|
+
VALUE ra, rf;
|
41
|
+
Geod *geod;
|
42
|
+
rb_scan_args(argc, argv, "02", &ra, &rf);
|
43
|
+
if ( NIL_P(ra) && NIL_P(rf) ) {
|
44
|
+
ra = rb_float_new(6378137); /* WGS84 : default */
|
45
|
+
rf = rb_float_new(1.0/298.257223563);
|
46
|
+
}
|
47
|
+
Data_Get_Struct(self, Geod, geod);
|
48
|
+
geod->ref = ALLOC(struct geod_geodesic);
|
49
|
+
geod_init(geod->ref, NUM2DBL(ra), NUM2DBL(rf));
|
50
|
+
return Qnil;
|
51
|
+
}
|
52
|
+
|
53
|
+
/*
|
54
|
+
* call-seq:
|
55
|
+
* Geod.invgeod(lon1,lat1,lon2,lat2) -> array
|
56
|
+
*
|
57
|
+
* Calculates the forward and back azimuths and distance
|
58
|
+
* between an initial and terminus point latitudes and longitudes.
|
59
|
+
* The latitude should be a floating point value in degrees, which
|
60
|
+
* positive value means northern direction. The longitude should be a
|
61
|
+
* floating point value in degrees, which positive value means western
|
62
|
+
* direction. The returned array contains the forward and back azimuths
|
63
|
+
* in degrees and distance in the metric specified by +Geod.init+.
|
64
|
+
*
|
65
|
+
* Geod.init("+ellps=WGS84 +units=km")
|
66
|
+
* Geod.invgeod(100,30,-100,-30)
|
67
|
+
* #=> [94.9053479513086, -85.0946520486914, 18101.7096098234]
|
68
|
+
*
|
69
|
+
*/
|
70
|
+
|
71
|
+
static VALUE
|
72
|
+
rb_geod_inverse (VALUE self, VALUE vl1, VALUE vp1, VALUE vl2, VALUE vp2)
|
73
|
+
{
|
74
|
+
Geod *geod;
|
75
|
+
double ps12, paz12, paz21;
|
76
|
+
Data_Get_Struct(self, Geod, geod);
|
77
|
+
geod_inverse(geod->ref,
|
78
|
+
NUM2DBL(vl1),
|
79
|
+
NUM2DBL(vp1),
|
80
|
+
NUM2DBL(vl2),
|
81
|
+
NUM2DBL(vp2),
|
82
|
+
&ps12, &paz12, &paz21);
|
83
|
+
return rb_ary_new3(3,
|
84
|
+
rb_float_new(ps12),
|
85
|
+
rb_float_new(paz12),
|
86
|
+
rb_float_new(paz21));
|
87
|
+
}
|
88
|
+
|
89
|
+
/*
|
90
|
+
* call-seq:
|
91
|
+
* Geod.distance(lon1,lat1,lon2,lat2) -> array
|
92
|
+
*
|
93
|
+
* Shortcut of +Geod.invgeod(lon1,lat1,lon2,lat2)[2]+
|
94
|
+
*
|
95
|
+
*/
|
96
|
+
|
97
|
+
static VALUE
|
98
|
+
rb_geod_distance (VALUE self, VALUE vl1, VALUE vp1, VALUE vl2, VALUE vp2)
|
99
|
+
{
|
100
|
+
Geod *geod;
|
101
|
+
double ps12;
|
102
|
+
Data_Get_Struct(self, Geod, geod);
|
103
|
+
geod_inverse(geod->ref,
|
104
|
+
NUM2DBL(vl1), NUM2DBL(vp1), NUM2DBL(vl2), NUM2DBL(vp2),
|
105
|
+
&ps12, 0, 0);
|
106
|
+
return rb_float_new(ps12);
|
107
|
+
}
|
108
|
+
|
109
|
+
/*
|
110
|
+
* call-seq:
|
111
|
+
* Geod.geod(lat1, lon1, az12, dist) -> array
|
112
|
+
*
|
113
|
+
* Calculates the latitude, longitude and back azimuth of a terminus point
|
114
|
+
* given a initial point latitude, longitude, azimuth and distance.
|
115
|
+
* The latitude should be a floating point value in degrees, which
|
116
|
+
* positive value means northern direction. The longitude should be a
|
117
|
+
* floating point value in degrees, which positive value means western
|
118
|
+
* direction. The returned array contains the latitude, longitude and
|
119
|
+
* back azimuth at the terminus point in degrees.
|
120
|
+
*
|
121
|
+
* Geod.init("+ellps=WGS84 +units=km")
|
122
|
+
* Geod.geod(100,30,94.9053479513086,18101.7096098234)
|
123
|
+
* #=> [-99.9999997537569, -30.0000001660067, -85.0946531598225]
|
124
|
+
*
|
125
|
+
*/
|
126
|
+
|
127
|
+
static VALUE
|
128
|
+
rb_geod_forward (VALUE self, VALUE vlat1, VALUE vlon1, VALUE vaz, VALUE vdist)
|
129
|
+
{
|
130
|
+
Geod *geod;
|
131
|
+
double l2, p2, az21;
|
132
|
+
Data_Get_Struct(self, Geod, geod);
|
133
|
+
geod_direct(geod->ref,
|
134
|
+
NUM2DBL(vlat1), NUM2DBL(vlon1), NUM2DBL(vaz), NUM2DBL(vdist),
|
135
|
+
&l2, &p2, &az21);
|
136
|
+
return rb_ary_new3(3,
|
137
|
+
rb_float_new(l2),
|
138
|
+
rb_float_new(p2),
|
139
|
+
rb_float_new(az21));
|
140
|
+
}
|
141
|
+
|
142
|
+
#ifdef HAVE_CARRAY_H
|
143
|
+
|
144
|
+
#include "carray.h"
|
145
|
+
|
146
|
+
|
147
|
+
/*
|
148
|
+
* call-seq:
|
149
|
+
* Geod#direct_ca(lat1, lon1, az12, dist, lat2, lon2, az21) -> nil
|
150
|
+
* Geod#forward_ca(lat1, lon1, az12, dist, lat2, lon2, az21) -> nil
|
151
|
+
*
|
152
|
+
* Calculates the latitude, longitude and back azimuth of a terminus point
|
153
|
+
* given a initial point latitude, longitude, azimuth and distance.
|
154
|
+
* The latitude should be a floating point value in degrees, which
|
155
|
+
* positive value means northern direction. The longitude should be a
|
156
|
+
* floating point value in degrees, which positive value means western
|
157
|
+
* direction. The returned array contains the latitude, longitude and
|
158
|
+
* back azimuth at the terminus point in degrees.
|
159
|
+
*
|
160
|
+
* input:
|
161
|
+
*
|
162
|
+
* lat1, lon1, az12, dist
|
163
|
+
* read only CArray.double or its convertible
|
164
|
+
*
|
165
|
+
* output:
|
166
|
+
*
|
167
|
+
* lat2, lon2, az21
|
168
|
+
* writable CArray.double or its convertible
|
169
|
+
*
|
170
|
+
*/
|
171
|
+
|
172
|
+
static VALUE
|
173
|
+
rb_geod_direct_ca (VALUE self,
|
174
|
+
volatile VALUE vlat1, volatile VALUE vlon1,
|
175
|
+
volatile VALUE vaz12, volatile VALUE vdist,
|
176
|
+
volatile VALUE vlat2, volatile VALUE vlon2,
|
177
|
+
volatile VALUE vaz21)
|
178
|
+
{
|
179
|
+
Geod *geod;
|
180
|
+
CArray *lat1, *lon1, *az12, *dist, *lat2, *lon2, *az21;
|
181
|
+
double *p1, *p2, *p3, *p4, *p5, *p6, *p7;
|
182
|
+
ca_size_t s1, s2, s3, s4, s5, s6, s7;
|
183
|
+
int8_t *m, *mp;
|
184
|
+
ca_size_t count;
|
185
|
+
ca_size_t i;
|
186
|
+
|
187
|
+
Data_Get_Struct(self, Geod, geod);
|
188
|
+
|
189
|
+
lat1 = ca_wrap_readonly(vlat1, CA_DOUBLE);
|
190
|
+
lon1 = ca_wrap_readonly(vlon1, CA_DOUBLE);
|
191
|
+
az12 = ca_wrap_readonly(vaz12, CA_DOUBLE);
|
192
|
+
dist = ca_wrap_readonly(vdist, CA_DOUBLE);
|
193
|
+
lat2 = ca_wrap_writable(vlat2, CA_DOUBLE);
|
194
|
+
lon2 = ca_wrap_writable(vlon2, CA_DOUBLE);
|
195
|
+
az21 = ca_wrap_writable(vaz21, CA_DOUBLE);
|
196
|
+
|
197
|
+
ca_attach_n(7, lat1, lon1, dist, az12, lat2, lon2, az21);
|
198
|
+
|
199
|
+
count = ca_set_iterator(7,
|
200
|
+
lat1, &p1, &s1,
|
201
|
+
lon1, &p2, &s2,
|
202
|
+
az12, &p3, &s3,
|
203
|
+
dist, &p4, &s4,
|
204
|
+
lat2, &p5, &s5,
|
205
|
+
lon2, &p6, &s6,
|
206
|
+
az21, &p7, &s7);
|
207
|
+
|
208
|
+
m = ca_allocate_mask_iterator(4, lat1, lon1, az12, dist);
|
209
|
+
|
210
|
+
mp = m;
|
211
|
+
for (i=0; i<count; i++) {
|
212
|
+
if ( *mp ) {
|
213
|
+
*p5 = 0.0/0.0;
|
214
|
+
*p6 = 0.0/0.0;
|
215
|
+
*p7 = 0.0/0.0;
|
216
|
+
}
|
217
|
+
else {
|
218
|
+
geod_direct(geod->ref, *p1, *p2, *p3, *p4, p5, p6, p7);
|
219
|
+
}
|
220
|
+
p1+=s1; p2+=s2; p3+=s3; p4+=s4; p5+=s5; p6+=s6; p7+=s7;
|
221
|
+
mp++;
|
222
|
+
}
|
223
|
+
|
224
|
+
free(m);
|
225
|
+
|
226
|
+
ca_sync_n(3, lat2, lon2, az21);
|
227
|
+
ca_detach_n(7, lat1, lon1, dist, az12, lat2, lon2, az21);
|
228
|
+
|
229
|
+
return Qnil;
|
230
|
+
}
|
231
|
+
|
232
|
+
/*
|
233
|
+
* call-seq:
|
234
|
+
* Geod.invgeod_d(lat1,lon1,lat2,lon2,dist,az12,az21) -> nil
|
235
|
+
*
|
236
|
+
* Calculates the forward and back azimuths and distance
|
237
|
+
* between an initial and terminus point latitudes and longitudes.
|
238
|
+
* The latitude should be a floating point value in degrees, which
|
239
|
+
* positive value means northern direction. The longitude should be a
|
240
|
+
* floating point value in degrees, which positive value means western
|
241
|
+
* direction. The returned array contains the forward and back azimuths
|
242
|
+
* in degrees and distance in the metric specified by +Geod.init+.
|
243
|
+
*
|
244
|
+
* input:
|
245
|
+
*
|
246
|
+
* lat1, lon1, lat2, lon2
|
247
|
+
* read only CArray.double or its convertible
|
248
|
+
*
|
249
|
+
* output:
|
250
|
+
*
|
251
|
+
* dist, az12, az21
|
252
|
+
* writable CArray.double or its convertible
|
253
|
+
*
|
254
|
+
*/
|
255
|
+
|
256
|
+
static VALUE
|
257
|
+
rb_geod_inverse_ca (VALUE self,
|
258
|
+
volatile VALUE vlat1, volatile VALUE vlon1,
|
259
|
+
volatile VALUE vlat2, volatile VALUE vlon2,
|
260
|
+
volatile VALUE vdist, volatile VALUE vaz12,
|
261
|
+
volatile VALUE vaz21)
|
262
|
+
{
|
263
|
+
Geod *geod;
|
264
|
+
CArray *lat1, *lon1, *lat2, *lon2, *dist, *az12, *az21;
|
265
|
+
double *p1, *p2, *p3, *p4, *p5, *p6, *p7;
|
266
|
+
ca_size_t s1, s2, s3, s4, s5, s6, s7;
|
267
|
+
int8_t *m, *mp;
|
268
|
+
ca_size_t count;
|
269
|
+
ca_size_t i;
|
270
|
+
|
271
|
+
Data_Get_Struct(self, Geod, geod);
|
272
|
+
|
273
|
+
lat1 = ca_wrap_readonly(vlat1, CA_DOUBLE);
|
274
|
+
lon1 = ca_wrap_readonly(vlon1, CA_DOUBLE);
|
275
|
+
lat2 = ca_wrap_readonly(vlat2, CA_DOUBLE);
|
276
|
+
lon2 = ca_wrap_readonly(vlon2, CA_DOUBLE);
|
277
|
+
dist = ca_wrap_writable(vdist, CA_DOUBLE);
|
278
|
+
az12 = ca_wrap_writable(vaz12, CA_DOUBLE);
|
279
|
+
az21 = ca_wrap_writable(vaz21, CA_DOUBLE);
|
280
|
+
|
281
|
+
ca_attach_n(7, lat1, lon1, lat2, lon2, dist, az12, az21);
|
282
|
+
|
283
|
+
count = ca_set_iterator(7,
|
284
|
+
lat1, &p1, &s1,
|
285
|
+
lon1, &p2, &s2,
|
286
|
+
lat2, &p3, &s3,
|
287
|
+
lon2, &p4, &s4,
|
288
|
+
dist, &p5, &s5,
|
289
|
+
az12, &p6, &s6,
|
290
|
+
az21, &p7, &s7);
|
291
|
+
|
292
|
+
m = ca_allocate_mask_iterator(4, lat1, lon1, lat2, lon2);
|
293
|
+
|
294
|
+
mp = m;
|
295
|
+
for (i=0; i<count; i++) {
|
296
|
+
if ( *mp ) {
|
297
|
+
*p5 = 0.0/0.0;
|
298
|
+
*p6 = 0.0/0.0;
|
299
|
+
*p7 = 0.0/0.0;
|
300
|
+
}
|
301
|
+
else {
|
302
|
+
geod_inverse(geod->ref, *p1, *p2, *p3, *p4, p5, p6, p7);
|
303
|
+
}
|
304
|
+
p1+=s1; p2+=s2; p3+=s3; p4+=s4; p5+=s5; p6+=s6; p7+=s7;
|
305
|
+
mp++;
|
306
|
+
}
|
307
|
+
|
308
|
+
free(m);
|
309
|
+
|
310
|
+
ca_sync_n(3, dist, az12, az21);
|
311
|
+
ca_detach_n(7, lat1, lon1, lat2, lon2, dist, az12, az21);
|
312
|
+
|
313
|
+
return Qnil;
|
314
|
+
}
|
315
|
+
|
316
|
+
static VALUE
|
317
|
+
rb_geod_distance_ca (VALUE self,
|
318
|
+
volatile VALUE vlat1, volatile VALUE vlon1,
|
319
|
+
volatile VALUE vlat2, volatile VALUE vlon2,
|
320
|
+
volatile VALUE vdist)
|
321
|
+
{
|
322
|
+
Geod *geod;
|
323
|
+
CArray *lat1, *lon1, *lat2, *lon2, *dist;
|
324
|
+
double *p1, *p2, *p3, *p4, *p5;
|
325
|
+
double d0;
|
326
|
+
ca_size_t s1, s2, s3, s4, s5;
|
327
|
+
int8_t *m, *mp;
|
328
|
+
ca_size_t count;
|
329
|
+
ca_size_t i;
|
330
|
+
|
331
|
+
Data_Get_Struct(self, Geod, geod);
|
332
|
+
|
333
|
+
lat1 = ca_wrap_readonly(vlat1, CA_DOUBLE);
|
334
|
+
lon1 = ca_wrap_readonly(vlon1, CA_DOUBLE);
|
335
|
+
lat2 = ca_wrap_readonly(vlat2, CA_DOUBLE);
|
336
|
+
lon2 = ca_wrap_readonly(vlon2, CA_DOUBLE);
|
337
|
+
dist = ca_wrap_writable(vdist, CA_DOUBLE);
|
338
|
+
|
339
|
+
ca_attach_n(5, lat1, lon1, lat2, lon2, dist);
|
340
|
+
|
341
|
+
count = ca_set_iterator(5,
|
342
|
+
lat1, &p1, &s1,
|
343
|
+
lon1, &p2, &s2,
|
344
|
+
lat2, &p3, &s3,
|
345
|
+
lon2, &p4, &s4,
|
346
|
+
dist, &p5, &s5);
|
347
|
+
|
348
|
+
m = ca_allocate_mask_iterator(4, lat1, lon1, lat2, lon2);
|
349
|
+
|
350
|
+
mp = m;
|
351
|
+
for (i=0; i<count; i++) {
|
352
|
+
if ( *mp ) {
|
353
|
+
*p5 = 0.0/0.0;
|
354
|
+
}
|
355
|
+
else {
|
356
|
+
geod_inverse(geod->ref, *p1, *p2, *p3, *p4, &d0, 0, 0);
|
357
|
+
*p5 = d0;
|
358
|
+
}
|
359
|
+
p1+=s1; p2+=s2; p3+=s3; p4+=s4; p5+=s5;
|
360
|
+
mp++;
|
361
|
+
}
|
362
|
+
|
363
|
+
free(m);
|
364
|
+
|
365
|
+
ca_sync(dist);
|
366
|
+
ca_detach_n(5, lat1, lon1, lat2, lon2, dist);
|
367
|
+
|
368
|
+
return Qnil;
|
369
|
+
}
|
370
|
+
|
371
|
+
|
372
|
+
#endif
|
373
|
+
|
374
|
+
void
|
375
|
+
Init_geod ()
|
376
|
+
{
|
377
|
+
rb_cGeod = rb_define_class_under(rb_mPROJ4, "Geod", rb_cObject);
|
378
|
+
rb_define_alloc_func(rb_cGeod, rb_geod_s_allocate);
|
379
|
+
rb_define_method(rb_cGeod, "_initialize", rb_geod_initialize, -1);
|
380
|
+
rb_define_private_method(rb_cGeod, "_inverse", rb_geod_inverse, 4);
|
381
|
+
rb_define_private_method(rb_cGeod, "_distance", rb_geod_distance, 4);
|
382
|
+
rb_define_private_method(rb_cGeod, "_forward", rb_geod_forward, 4);
|
383
|
+
|
384
|
+
#ifdef HAVE_CARRAY_H
|
385
|
+
rb_define_private_method(rb_cGeod, "_direct_ca", rb_geod_direct_ca, 7);
|
386
|
+
rb_define_private_method(rb_cGeod, "_forward_ca", rb_geod_direct_ca, 7);
|
387
|
+
rb_define_private_method(rb_cGeod, "_inverse_ca", rb_geod_inverse_ca, 7);
|
388
|
+
rb_define_private_method(rb_cGeod, "_distance_ca", rb_geod_distance_ca, 5);
|
389
|
+
#endif
|
390
|
+
|
391
|
+
}
|
392
|
+
|