-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathcommon.h
363 lines (287 loc) · 12.2 KB
/
common.h
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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
/*
Copyright (C) 2004-2023 Sergey Koposov
Email: skoposov AT ed DOT ac DOT uk
This file is part of Q3C.
Q3C is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
Q3C is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Q3C; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <math.h>
#include <stdlib.h>
#include "postgres.h"
#ifndef __STDC_FORMAT_MACROS
#define __STDC_FORMAT_MACROS
#endif
#include <inttypes.h>
#include <stdint.h>
#ifndef Q3C_COMMON_H
#define Q3C_COMMON_H
#ifndef q3c_ipix_t
/*#define q3c_ipix_t long long
typedef long long q3c_ipix_t ;*/
typedef int64 q3c_ipix_t;
#endif /* q3c_ipix_t */
#ifndef Q3C_IPIX_FMT
#define Q3C_IPIX_FMT INT64_FORMAT
#define Q3C_CONST(c) INT64CONST(c)
#endif /* Q3C_IPIX_FMT */
#ifndef Q3C_MAX_IPIX
#define Q3C_MAX_IPIX ( (((q3c_ipix_t)1 << 60) - 1) + ((q3c_ipix_t)1 << 60) + ((q3c_ipix_t)1 << 62))
/* This is maximum allowed ipix value
101 + 1111111111111....111
60 ones
*/
#endif /* Q3C_MAX_IPIX */
/* If You have not specified the Q3C_LONG_DOUBLE macro then we will use simple
double functions */
#ifndef Q3C_LONG_DOUBLE
typedef double q3c_coord_t;
#define q3c_pow(a,b) pow(a, b)
#define q3c_fabs(a) fabs(a)
#define q3c_fmod(a,b) fmod(a, b)
#define q3c_cos(a) cos(a)
#define q3c_sin(a) sin(a)
#define q3c_asin(a) asin(a)
#define q3c_acos(a) acos(a)
#define q3c_sincos0(a, b, c) sincos(a, b, c)
#define q3c_tan(a) tan(a)
#define q3c_atan(a) atan(a)
#define q3c_atan2(a,b) atan2(a, b)
#define q3c_sqrt(a) sqrt(a)
#define q3c_log(a) log(a)
#define q3c_ceil(a) ceil(a)
#define Q3C_HALF 0.5
#define Q3C_COORD_FMT "%f"
#define Q3C_PI 3.1415926535897932384626433832795028841968
#define Q3C_2PI 6.2831853071795864769252867665590057683936
#define Q3C_PI_2 1.5707963267948966192313216916397514420984
#define Q3C_DEGRA 0.0174532925199432957692369076848861271344
#define Q3C_RADEG 57.2957795130823208767981548141051703324122
#define Q3C_LG2 0.6931471805599453094172321214581765680755
/*
I really do not see the reason to set them as const ...
const q3c_coord_t Q3C_PI=3.1415926535897932384626433832795028841968;
const q3c_coord_t Q3C_2PI=6.2831853071795864769252867665590057683936;
const q3c_coord_t Q3C_PI_2=1.5707963267948966192313216916397514420984;
const q3c_coord_t Q3C_DEGRA=0.0174532925199432957692369076848861271344;
const q3c_coord_t Q3C_RADEG=57.2957795130823208767981548141051703324122;
const q3c_coord_t Q3C_LG2=0.6931471805599453094172321214581765680755;
*/
#else /* Q3C_LONG_DOUBLE */
typedef long double q3c_coord_t;
#define q3c_pow(a,b) powl(a, b)
#define q3c_fabs(a) fabsl(a)
#define q3c_fmod(a,b) fmodl(a, b)
#define q3c_cos(a) cosl(a)
#define q3c_sin(a) sinl(a)
#define q3c_asin(a) asinl(a)
#define q3c_acos(a) acosl(a)
#define q3c_sincos0(a, b, c) sincosl(a, b, c)
#define q3c_tan(a) tanl(a)
#define q3c_atan(a) atanl(a)
#define q3c_atan2(a,b) atan2l(a, b)
#define q3c_sqrt(a) sqrtl(a)
#define q3c_log(a) logl(a)
#define q3c_ceil(a) ceill(a)
#define Q3C_HALF 0.5L
#define Q3C_COORD_FMT "%.15Lf"
#define Q3C_PI 3.1415926535897932384626433832795028841968L
#define Q3C_2PI 6.2831853071795864769252867665590057683936L
#define Q3C_PI_2 1.5707963267948966192313216916397514420984L
#define Q3C_DEGRA 0.0174532925199432957692369076848861271344L
#define Q3C_RADEG 57.2957795130823208767981548141051703324122L
#define Q3C_LG2 0.6931471805599453094172321214581765680755L
#endif /* Q3C_LONG_DOUBLE */
#ifndef Q3C_CBITS
#define Q3C_CBITS sizeof(q3c_coord_t)
#endif /* Q3C_CBITS */
#ifndef Q3C_IBITS
#define Q3C_IBITS sizeof(q3c_ipix_t)
#endif /* Q3C_IBITS */
#ifndef Q3C_INTERLEAVED_NBITS
#define Q3C_INTERLEAVED_NBITS 16
#endif /* Q3C_INTERLEAVED_NBITS */
#ifndef Q3C_I1
#define Q3C_I1 (1 << (Q3C_INTERLEAVED_NBITS))
#endif /* Q3C_INTERLEAVED_NBITS */
/* this probably will work only in the case of long double variables ?? */
/*#define Q3C_PI 3.1415926535897932384626433832795029L
#define q3c_2_PI 3.1415926535897932384626433832795029L
#define Q3C_2PI 6.2831853071795864769252867665590058L
#define Q3C_PI_2 1.5707963267948966192313216916397514L
#define Q3C_DEGRA 0.01745329251994329576923690768488612L
#define Q3C_RADEG 57.295779513082320876798154814105170L
#define Q3C_LG2 0.301029995663981195213738894724493026768L
#define Q3C_LG2 0.6931471805599453094172321214581765680755L*/
#define Q3C_BOX_INTERSECT(x0, x1, y0, y1, a0, a1, b0, b1) ((((x0 < a0) && \
(x1 >= a0)) || \
(x0 <= a1)) && \
(((y0 < b0) && \
(y1 >= b0)) || \
(y0 <= b1)))
/* True if square(x0,x1,y0,y1) and square(a0,a1,b0,b1) have any common point*/
#define Q3C_INTERSECT(a, b, x, y) (((a <= y) && (a >= x)) || \
((a < x) && (b >= x)))
#define Q3C_DISJUNCT 0 /* disjunct areas */
#define Q3C_PARTIAL 1 /* partial coverage */
#define Q3C_EDGE 1 /* point lie on the edge of the region */
#define Q3C_COVER 2 /* the point or the region is completely inside other */
#define Q3C_MAX_N_POLY_VERTEX 100
/* Maximal number of vertices in the polygon */
#define Q3C_NPARTIALS 50
/*length of the list of the partially covered ipix ranges*/
#define Q3C_NFULLS 50
/*length of the list of the fully covered ipix ranges*/
#define Q3C_MAX_DEPTH 4
/*the maximum depth of going down the quadtree when doing spatial searches */
#define Q3C_STACK_SIZE 11000
/* the size of the stacks for quadtrees when doing spatial searches */
/* !!!!!!!!!!!!! IMPORTANT !!!!!!!!!!!!!!!
* Consider that the size of the stacks should directly depend on the
* value of res_depth variable !
* It seems that each of stacks should have the size 4*(2^(depth-1))
*/
#define Q3C_MAXRAD 35
/* maximum allowed radius for circles and ellipses */
#define Q3C_MINDISCR 1e-10
/* minimum possible value of the discriminant of the 2nd order curves,
before we start assuming parabola or hyperbola
*/
#define UNWRAP_RA(ra) ( (ra < 0) ? \
(q3c_fmod(ra, 360) + 360) : \
( (ra > 360) ? q3c_fmod(ra, 360) : ra ) \
)
#ifdef __USE_GNU
#define q3c_sincos(a,b,c) q3c_sincos0(a,&b,&c)
#else
#define q3c_sincos(a,b,c) do { \
b = q3c_sin(a); \
c = q3c_cos(a); \
} while(0);
#endif
struct q3c_prm
{
const q3c_ipix_t nside;
const q3c_ipix_t *xbits;
const q3c_ipix_t *ybits;
const q3c_ipix_t *xbits1;
const q3c_ipix_t *ybits1;
};
struct q3c_prm_write
{
q3c_ipix_t nside;
q3c_ipix_t *xbits;
q3c_ipix_t *ybits;
q3c_ipix_t *xbits1;
q3c_ipix_t *ybits1;
};
struct q3c_square
{
q3c_ipix_t x0, y0; /* Integer coordinates of the center of the square for */
int nside0; /* the nside0 segmentation */
char status;
};
#define SET_SQUARE(sq, x, y, n) do { \
sq->x0 = x; \
sq->y0 = y; \
sq->nside0 = n; \
} while(0);
typedef struct
{
int n;
q3c_coord_t *ra; /* array of RAs of vertices */
q3c_coord_t *dec; /* array of DECs of vertices */
q3c_coord_t *x; /* array of X coords on the cube face of vertices */
q3c_coord_t *y; /* array of Y coords on the cube face of vertices */
q3c_coord_t *ax; /* array of x projections of the edge between vertices */
q3c_coord_t *ay; /* array of y projections of the edge between vertices */
} q3c_poly;
typedef struct
{
q3c_coord_t ra;
q3c_coord_t dec;
q3c_coord_t rad;
} q3c_circle_region;
typedef struct
{
q3c_coord_t ra;
q3c_coord_t dec;
q3c_coord_t rad; /* major axis */
q3c_coord_t e; /* eccentricity */
q3c_coord_t PA;
} q3c_ellipse_region;
typedef enum {Q3C_CIRCLE, Q3C_POLYGON, Q3C_ELLIPSE} q3c_region;
void init_q3c(struct q3c_prm_write *, q3c_ipix_t);
void init_q3c1(struct q3c_prm_write *, q3c_ipix_t);
void q3c_dump_prm(struct q3c_prm_write *,char *);
void q3c_ang2ipix(struct q3c_prm *, q3c_coord_t, q3c_coord_t, q3c_ipix_t *);
void q3c_ipix2ang(struct q3c_prm *, q3c_ipix_t, q3c_coord_t *, q3c_coord_t *);
q3c_coord_t q3c_pixarea(struct q3c_prm *hprm, q3c_ipix_t ipix, int depth);
void q3c_get_nearby_split(struct q3c_prm *, q3c_coord_t, q3c_coord_t,
q3c_coord_t, q3c_ipix_t *, int);
void q3c_get_nearby(struct q3c_prm *, q3c_region, void *, q3c_ipix_t *);
void q3c_get_xy_minmax(q3c_coord_t, q3c_coord_t, q3c_coord_t, q3c_coord_t,
q3c_coord_t, q3c_coord_t, q3c_coord_t *, q3c_coord_t *,
q3c_coord_t *, q3c_coord_t *, char *);
void q3c_ang2ipix_xy(struct q3c_prm *hprm, q3c_coord_t ra, q3c_coord_t dec,
char *out_face_num, q3c_ipix_t *ipix, q3c_coord_t *x_out,
q3c_coord_t *y_out);
void q3c_get_poly_coefs(char, q3c_coord_t, q3c_coord_t, q3c_coord_t,
q3c_coord_t *, q3c_coord_t *, q3c_coord_t *,
q3c_coord_t *, q3c_coord_t *, q3c_coord_t *);
char q3c_xy2facenum(q3c_coord_t, q3c_coord_t, char);
char q3c_get_facenum(q3c_coord_t, q3c_coord_t);
char q3c_in_ellipse(q3c_coord_t alpha, q3c_coord_t delta0, q3c_coord_t alpha1,
q3c_coord_t delta01, q3c_coord_t d0, q3c_coord_t e,
q3c_coord_t PA0);
q3c_coord_t q3c_dist(q3c_coord_t, q3c_coord_t, q3c_coord_t, q3c_coord_t);
q3c_coord_t q3c_sindist(q3c_coord_t, q3c_coord_t, q3c_coord_t, q3c_coord_t);
void q3c_radial_query(struct q3c_prm *hprm, q3c_coord_t ra0,
q3c_coord_t dec0, q3c_coord_t rad,
q3c_ipix_t *out_ipix_arr_fulls,
q3c_ipix_t *out_ipix_arr_partials);
void q3c_ellipse_query(struct q3c_prm *hprm, q3c_coord_t ra0,
q3c_coord_t dec0, q3c_coord_t majax, q3c_coord_t PA,
q3c_coord_t ell, q3c_ipix_t *out_ipix_arr_fulls,
q3c_ipix_t *out_ipix_arr_partials);
void q3c_init_poly(q3c_poly *qp, int n);
void q3c_prepare_poly(q3c_poly *qp);
void q3c_project_poly(q3c_poly *qp, char facenum, char *large_flag);
char q3c_get_facenum_poly(q3c_poly *qp);
int q3c_check_point_in_poly(q3c_poly *qp, q3c_coord_t x0, q3c_coord_t y0);
int q3c_poly_cover_check(q3c_poly *qp, q3c_coord_t xc_cur,
q3c_coord_t yc_cur, q3c_coord_t cur_size);
void q3c_get_minmax_poly(q3c_poly *qp, q3c_coord_t *xmin, q3c_coord_t *xmax,
q3c_coord_t *ymin, q3c_coord_t *ymax);
void q3c_poly_query(struct q3c_prm *hprm, q3c_poly *qp,
q3c_ipix_t *out_ipix_arr_fulls,
q3c_ipix_t *out_ipix_arr_partials,
char *too_large);
int q3c_check_sphere_point_in_poly(struct q3c_prm *hprm, int n,
q3c_coord_t in_ra[], q3c_coord_t in_dec[],
q3c_coord_t ra0, q3c_coord_t dec0,
char *too_large,
int invocation,
q3c_coord_t (*)[Q3C_MAX_N_POLY_VERTEX],
q3c_coord_t (*)[Q3C_MAX_N_POLY_VERTEX],
q3c_coord_t (*)[Q3C_MAX_N_POLY_VERTEX],
q3c_coord_t (*)[Q3C_MAX_N_POLY_VERTEX],
char *, char *);
char q3c_get_region_facenum(q3c_region region, void *data);
q3c_ipix_t q3c_xiyi2ipix(const q3c_ipix_t nside, const q3c_ipix_t *xbits,
const q3c_ipix_t *ybits, char face_num,
q3c_ipix_t xi, q3c_ipix_t yi);
void q3c_multi_face_check(q3c_coord_t *xmin0, q3c_coord_t *ymin0,
q3c_coord_t *xmax0, q3c_coord_t *ymax0,
q3c_coord_t *points, char *multi_flag);
char q3c_too_big_check(q3c_region region, void * region_data);
void q3c_get_version(char *, int);
#endif/* Q3C_COMMON_H */