PostgreSQL Source Code git master
btree_ts.c
Go to the documentation of this file.
1/*
2 * contrib/btree_gist/btree_ts.c
3 */
4#include "postgres.h"
5
6#include <limits.h>
7
8#include "btree_gist.h"
9#include "btree_utils_num.h"
10#include "utils/fmgrprotos.h"
11#include "utils/timestamp.h"
12#include "utils/float.h"
13#include "utils/sortsupport.h"
14
15typedef struct
16{
19} tsKEY;
20
21/* GiST support functions */
34
35
36#ifdef USE_FLOAT8_BYVAL
37#define TimestampGetDatumFast(X) TimestampGetDatum(X)
38#else
39#define TimestampGetDatumFast(X) PointerGetDatum(&(X))
40#endif
41
42
43/* define for comparison */
44
45static bool
46gbt_tsgt(const void *a, const void *b, FmgrInfo *flinfo)
47{
48 const Timestamp *aa = (const Timestamp *) a;
49 const Timestamp *bb = (const Timestamp *) b;
50
54}
55
56static bool
57gbt_tsge(const void *a, const void *b, FmgrInfo *flinfo)
58{
59 const Timestamp *aa = (const Timestamp *) a;
60 const Timestamp *bb = (const Timestamp *) b;
61
65}
66
67static bool
68gbt_tseq(const void *a, const void *b, FmgrInfo *flinfo)
69{
70 const Timestamp *aa = (const Timestamp *) a;
71 const Timestamp *bb = (const Timestamp *) b;
72
76}
77
78static bool
79gbt_tsle(const void *a, const void *b, FmgrInfo *flinfo)
80{
81 const Timestamp *aa = (const Timestamp *) a;
82 const Timestamp *bb = (const Timestamp *) b;
83
87}
88
89static bool
90gbt_tslt(const void *a, const void *b, FmgrInfo *flinfo)
91{
92 const Timestamp *aa = (const Timestamp *) a;
93 const Timestamp *bb = (const Timestamp *) b;
94
98}
99
100static int
101gbt_tskey_cmp(const void *a, const void *b, FmgrInfo *flinfo)
102{
103 tsKEY *ia = (tsKEY *) (((const Nsrt *) a)->t);
104 tsKEY *ib = (tsKEY *) (((const Nsrt *) b)->t);
105 int res;
106
108 if (res == 0)
110
111 return res;
112}
113
114static float8
115gbt_ts_dist(const void *a, const void *b, FmgrInfo *flinfo)
116{
117 const Timestamp *aa = (const Timestamp *) a;
118 const Timestamp *bb = (const Timestamp *) b;
119 Interval *i;
120
122 return get_float8_infinity();
123
127 return fabs(INTERVAL_TO_SEC(i));
128}
129
130static const gbtree_ninfo tinfo =
131{
132 gbt_t_ts,
133 sizeof(Timestamp),
134 16, /* sizeof(gbtreekey16) */
135 gbt_tsgt,
136 gbt_tsge,
137 gbt_tseq,
138 gbt_tsle,
139 gbt_tslt,
142};
143
144
146Datum
148{
151 Interval *r;
152
154 {
155 Interval *p = palloc(sizeof(Interval));
156
157 p->day = INT_MAX;
158 p->month = INT_MAX;
159 p->time = PG_INT64_MAX;
161 }
162 else
165 PG_GETARG_DATUM(1)));
167}
168
170Datum
172{
175 Interval *r;
176
178 {
179 Interval *p = palloc(sizeof(Interval));
180
181 p->day = INT_MAX;
182 p->month = INT_MAX;
183 p->time = PG_INT64_MAX;
185 }
186
189 PG_GETARG_DATUM(1)));
191}
192
193/**************************************************
194 * GiST support functions
195 **************************************************/
196
197static inline Timestamp
199{
200 /* No timezone correction is needed, since GMT is offset 0 by definition */
201 return (Timestamp) ts;
202}
203
204
205Datum
207{
208 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
209
211}
212
213Datum
215{
216 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
217 GISTENTRY *retval;
218
219 if (entry->leafkey)
220 {
221 tsKEY *r = (tsKEY *) palloc(sizeof(tsKEY));
224
225 gmt = tstz_to_ts_gmt(ts);
226
227 retval = palloc(sizeof(GISTENTRY));
228 r->lower = r->upper = gmt;
229 gistentryinit(*retval, PointerGetDatum(r),
230 entry->rel, entry->page,
231 entry->offset, false);
232 }
233 else
234 retval = entry;
235
236 PG_RETURN_POINTER(retval);
237}
238
239Datum
241{
242 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
243
245}
246
247Datum
249{
250 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
253
254 /* Oid subtype = PG_GETARG_OID(3); */
255 bool *recheck = (bool *) PG_GETARG_POINTER(4);
256 tsKEY *kkk = (tsKEY *) DatumGetPointer(entry->key);
258
259 /* All cases served by this function are exact */
260 *recheck = false;
261
262 key.lower = (GBT_NUMKEY *) &kkk->lower;
263 key.upper = (GBT_NUMKEY *) &kkk->upper;
264
265 PG_RETURN_BOOL(gbt_num_consistent(&key, &query, &strategy,
266 GIST_LEAF(entry), &tinfo, fcinfo->flinfo));
267}
268
269Datum
271{
272 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
274
275 /* Oid subtype = PG_GETARG_OID(3); */
276 tsKEY *kkk = (tsKEY *) DatumGetPointer(entry->key);
278
279 key.lower = (GBT_NUMKEY *) &kkk->lower;
280 key.upper = (GBT_NUMKEY *) &kkk->upper;
281
283 &tinfo, fcinfo->flinfo));
284}
285
286Datum
288{
289 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
292
293 /* Oid subtype = PG_GETARG_OID(3); */
294 bool *recheck = (bool *) PG_GETARG_POINTER(4);
295 char *kkk = (char *) DatumGetPointer(entry->key);
297 Timestamp qqq;
298
299 /* All cases served by this function are exact */
300 *recheck = false;
301
302 key.lower = (GBT_NUMKEY *) &kkk[0];
303 key.upper = (GBT_NUMKEY *) &kkk[MAXALIGN(tinfo.size)];
304 qqq = tstz_to_ts_gmt(query);
305
306 PG_RETURN_BOOL(gbt_num_consistent(&key, &qqq, &strategy,
307 GIST_LEAF(entry), &tinfo, fcinfo->flinfo));
308}
309
310Datum
312{
313 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
315
316 /* Oid subtype = PG_GETARG_OID(3); */
317 char *kkk = (char *) DatumGetPointer(entry->key);
319 Timestamp qqq;
320
321 key.lower = (GBT_NUMKEY *) &kkk[0];
322 key.upper = (GBT_NUMKEY *) &kkk[MAXALIGN(tinfo.size)];
323 qqq = tstz_to_ts_gmt(query);
324
326 &tinfo, fcinfo->flinfo));
327}
328
329
330Datum
332{
334 void *out = palloc(sizeof(tsKEY));
335
336 *(int *) PG_GETARG_POINTER(1) = sizeof(tsKEY);
337 PG_RETURN_POINTER(gbt_num_union(out, entryvec, &tinfo, fcinfo->flinfo));
338}
339
340
341#define penalty_check_max_float(val) \
342 do { \
343 if ( val > FLT_MAX ) \
344 val = FLT_MAX; \
345 if ( val < -FLT_MAX ) \
346 val = -FLT_MAX; \
347 } while (0)
348
349
350Datum
352{
353 tsKEY *origentry = (tsKEY *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
354 tsKEY *newentry = (tsKEY *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
355 float *result = (float *) PG_GETARG_POINTER(2);
356
357 double orgdbl[2],
358 newdbl[2];
359
360 /*
361 * We are always using "double" timestamps here. Precision should be good
362 * enough.
363 */
364 orgdbl[0] = ((double) origentry->lower);
365 orgdbl[1] = ((double) origentry->upper);
366 newdbl[0] = ((double) newentry->lower);
367 newdbl[1] = ((double) newentry->upper);
368
369 penalty_check_max_float(orgdbl[0]);
370 penalty_check_max_float(orgdbl[1]);
371 penalty_check_max_float(newdbl[0]);
372 penalty_check_max_float(newdbl[1]);
373
374 penalty_num(result, orgdbl[0], orgdbl[1], newdbl[0], newdbl[1]);
375
376 PG_RETURN_POINTER(result);
377}
378
379
380Datum
382{
385 &tinfo, fcinfo->flinfo));
386}
387
388Datum
390{
391 tsKEY *b1 = (tsKEY *) PG_GETARG_POINTER(0);
392 tsKEY *b2 = (tsKEY *) PG_GETARG_POINTER(1);
393 bool *result = (bool *) PG_GETARG_POINTER(2);
394
395 *result = gbt_num_same((void *) b1, (void *) b2, &tinfo, fcinfo->flinfo);
396 PG_RETURN_POINTER(result);
397}
398
399static int
401{
402 tsKEY *arg1 = (tsKEY *) DatumGetPointer(x);
403 tsKEY *arg2 = (tsKEY *) DatumGetPointer(y);
404
405 /* for leaf items we expect lower == upper, so only compare lower */
409}
410
411Datum
413{
415
417 ssup->ssup_extra = NULL;
418
420}
Datum timestamp_cmp(PG_FUNCTION_ARGS)
Definition: timestamp.c:2270
Datum timestamp_ge(PG_FUNCTION_ARGS)
Definition: timestamp.c:2261
Datum timestamp_lt(PG_FUNCTION_ARGS)
Definition: timestamp.c:2234
Datum timestamp_le(PG_FUNCTION_ARGS)
Definition: timestamp.c:2252
Datum timestamp_gt(PG_FUNCTION_ARGS)
Definition: timestamp.c:2243
Datum timestamp_mi(PG_FUNCTION_ARGS)
Definition: timestamp.c:2845
Datum timestamp_eq(PG_FUNCTION_ARGS)
Definition: timestamp.c:2216
@ gbt_t_ts
Definition: btree_gist.h:23
Interval * abs_interval(Interval *a)
Datum gbt_ts_compress(PG_FUNCTION_ARGS)
Definition: btree_ts.c:206
static bool gbt_tsle(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:79
#define TimestampGetDatumFast(X)
Definition: btree_ts.c:39
Datum gbt_ts_penalty(PG_FUNCTION_ARGS)
Definition: btree_ts.c:351
Datum gbt_tstz_compress(PG_FUNCTION_ARGS)
Definition: btree_ts.c:214
static int gbt_ts_ssup_cmp(Datum x, Datum y, SortSupport ssup)
Definition: btree_ts.c:400
static bool gbt_tsgt(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:46
Datum gbt_ts_same(PG_FUNCTION_ARGS)
Definition: btree_ts.c:389
Datum gbt_tstz_consistent(PG_FUNCTION_ARGS)
Definition: btree_ts.c:287
Datum gbt_tstz_distance(PG_FUNCTION_ARGS)
Definition: btree_ts.c:311
Datum gbt_ts_distance(PG_FUNCTION_ARGS)
Definition: btree_ts.c:270
Datum gbt_ts_picksplit(PG_FUNCTION_ARGS)
Definition: btree_ts.c:381
#define penalty_check_max_float(val)
Definition: btree_ts.c:341
Datum ts_dist(PG_FUNCTION_ARGS)
Definition: btree_ts.c:147
Datum gbt_ts_union(PG_FUNCTION_ARGS)
Definition: btree_ts.c:331
Datum gbt_ts_fetch(PG_FUNCTION_ARGS)
Definition: btree_ts.c:240
static const gbtree_ninfo tinfo
Definition: btree_ts.c:130
static bool gbt_tslt(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:90
static bool gbt_tsge(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:57
Datum tstz_dist(PG_FUNCTION_ARGS)
Definition: btree_ts.c:171
static Timestamp tstz_to_ts_gmt(TimestampTz ts)
Definition: btree_ts.c:198
Datum gbt_ts_consistent(PG_FUNCTION_ARGS)
Definition: btree_ts.c:248
Datum gbt_ts_sortsupport(PG_FUNCTION_ARGS)
Definition: btree_ts.c:412
static int gbt_tskey_cmp(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:101
static bool gbt_tseq(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:68
PG_FUNCTION_INFO_V1(gbt_ts_compress)
static float8 gbt_ts_dist(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:115
GISTENTRY * gbt_num_compress(GISTENTRY *entry, const gbtree_ninfo *tinfo)
float8 gbt_num_distance(const GBT_NUMKEY_R *key, const void *query, bool is_leaf, const gbtree_ninfo *tinfo, FmgrInfo *flinfo)
bool gbt_num_consistent(const GBT_NUMKEY_R *key, const void *query, const StrategyNumber *strategy, bool is_leaf, const gbtree_ninfo *tinfo, FmgrInfo *flinfo)
bool gbt_num_same(const GBT_NUMKEY *a, const GBT_NUMKEY *b, const gbtree_ninfo *tinfo, FmgrInfo *flinfo)
GISTENTRY * gbt_num_fetch(GISTENTRY *entry, const gbtree_ninfo *tinfo)
GIST_SPLITVEC * gbt_num_picksplit(const GistEntryVector *entryvec, GIST_SPLITVEC *v, const gbtree_ninfo *tinfo, FmgrInfo *flinfo)
void * gbt_num_union(GBT_NUMKEY *out, const GistEntryVector *entryvec, const gbtree_ninfo *tinfo, FmgrInfo *flinfo)
#define penalty_num(result, olower, oupper, nlower, nupper)
#define INTERVAL_TO_SEC(ivp)
char GBT_NUMKEY
#define MAXALIGN(LEN)
Definition: c.h:782
double float8
Definition: c.h:601
#define PG_INT64_MAX
Definition: c.h:563
int64 Timestamp
Definition: timestamp.h:38
int64 TimestampTz
Definition: timestamp.h:39
#define TIMESTAMP_NOT_FINITE(j)
Definition: timestamp.h:169
static float8 get_float8_infinity(void)
Definition: float.h:94
#define PG_RETURN_VOID()
Definition: fmgr.h:349
#define DirectFunctionCall2(func, arg1, arg2)
Definition: fmgr.h:684
#define PG_RETURN_FLOAT8(x)
Definition: fmgr.h:367
#define PG_GETARG_POINTER(n)
Definition: fmgr.h:276
#define PG_GETARG_DATUM(n)
Definition: fmgr.h:268
#define PG_GETARG_UINT16(n)
Definition: fmgr.h:272
#define PG_RETURN_POINTER(x)
Definition: fmgr.h:361
#define PG_FUNCTION_ARGS
Definition: fmgr.h:193
#define PG_RETURN_BOOL(x)
Definition: fmgr.h:359
#define GIST_LEAF(entry)
Definition: gist.h:171
#define gistentryinit(e, k, r, pg, o, l)
Definition: gist.h:245
int y
Definition: isn.c:76
int b
Definition: isn.c:74
int x
Definition: isn.c:75
int a
Definition: isn.c:73
int i
Definition: isn.c:77
static const char gmt[]
Definition: localtime.c:53
void * palloc(Size size)
Definition: mcxt.c:1945
static bool DatumGetBool(Datum X)
Definition: postgres.h:95
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:327
uintptr_t Datum
Definition: postgres.h:69
static Pointer DatumGetPointer(Datum X)
Definition: postgres.h:317
static int32 DatumGetInt32(Datum X)
Definition: postgres.h:207
struct SortSupportData * SortSupport
Definition: sortsupport.h:58
uint16 StrategyNumber
Definition: stratnum.h:22
Definition: fmgr.h:57
OffsetNumber offset
Definition: gist.h:164
Datum key
Definition: gist.h:161
Page page
Definition: gist.h:163
Relation rel
Definition: gist.h:162
bool leafkey
Definition: gist.h:165
int32 day
Definition: timestamp.h:51
int32 month
Definition: timestamp.h:52
TimeOffset time
Definition: timestamp.h:49
int(* comparator)(Datum x, Datum y, SortSupport ssup)
Definition: sortsupport.h:106
void * ssup_extra
Definition: sortsupport.h:87
Definition: btree_ts.c:16
Timestamp lower
Definition: btree_ts.c:17
Timestamp upper
Definition: btree_ts.c:18
#define PG_GETARG_TIMESTAMP(n)
Definition: timestamp.h:63
#define PG_GETARG_TIMESTAMPTZ(n)
Definition: timestamp.h:64
static Interval * DatumGetIntervalP(Datum X)
Definition: timestamp.h:40
#define PG_RETURN_INTERVAL_P(x)
Definition: timestamp.h:69
static TimestampTz DatumGetTimestampTz(Datum X)
Definition: timestamp.h:34