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
14typedef struct
15{
18} tsKEY;
19
20/*
21** timestamp ops
22*/
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
43static bool
44gbt_tsgt(const void *a, const void *b, FmgrInfo *flinfo)
45{
46 const Timestamp *aa = (const Timestamp *) a;
47 const Timestamp *bb = (const Timestamp *) b;
48
52}
53
54static bool
55gbt_tsge(const void *a, const void *b, FmgrInfo *flinfo)
56{
57 const Timestamp *aa = (const Timestamp *) a;
58 const Timestamp *bb = (const Timestamp *) b;
59
63}
64
65static bool
66gbt_tseq(const void *a, const void *b, FmgrInfo *flinfo)
67{
68 const Timestamp *aa = (const Timestamp *) a;
69 const Timestamp *bb = (const Timestamp *) b;
70
74}
75
76static bool
77gbt_tsle(const void *a, const void *b, FmgrInfo *flinfo)
78{
79 const Timestamp *aa = (const Timestamp *) a;
80 const Timestamp *bb = (const Timestamp *) b;
81
85}
86
87static bool
88gbt_tslt(const void *a, const void *b, FmgrInfo *flinfo)
89{
90 const Timestamp *aa = (const Timestamp *) a;
91 const Timestamp *bb = (const Timestamp *) b;
92
96}
97
98
99static int
100gbt_tskey_cmp(const void *a, const void *b, FmgrInfo *flinfo)
101{
102 tsKEY *ia = (tsKEY *) (((const Nsrt *) a)->t);
103 tsKEY *ib = (tsKEY *) (((const Nsrt *) b)->t);
104 int res;
105
107 if (res == 0)
109
110 return res;
111}
112
113static float8
114gbt_ts_dist(const void *a, const void *b, FmgrInfo *flinfo)
115{
116 const Timestamp *aa = (const Timestamp *) a;
117 const Timestamp *bb = (const Timestamp *) b;
118 Interval *i;
119
121 return get_float8_infinity();
122
126 return fabs(INTERVAL_TO_SEC(i));
127}
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/**************************************************
195 * timestamp ops
196 **************************************************/
197
198
199static inline Timestamp
201{
202 /* No timezone correction is needed, since GMT is offset 0 by definition */
203 return (Timestamp) ts;
204}
205
206
207Datum
209{
210 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
211
213}
214
215
216Datum
218{
219 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
220 GISTENTRY *retval;
221
222 if (entry->leafkey)
223 {
224 tsKEY *r = (tsKEY *) palloc(sizeof(tsKEY));
227
228 gmt = tstz_to_ts_gmt(ts);
229
230 retval = palloc(sizeof(GISTENTRY));
231 r->lower = r->upper = gmt;
232 gistentryinit(*retval, PointerGetDatum(r),
233 entry->rel, entry->page,
234 entry->offset, false);
235 }
236 else
237 retval = entry;
238
239 PG_RETURN_POINTER(retval);
240}
241
242Datum
244{
245 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
246
248}
249
250Datum
252{
253 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
256
257 /* Oid subtype = PG_GETARG_OID(3); */
258 bool *recheck = (bool *) PG_GETARG_POINTER(4);
259 tsKEY *kkk = (tsKEY *) DatumGetPointer(entry->key);
261
262 /* All cases served by this function are exact */
263 *recheck = false;
264
265 key.lower = (GBT_NUMKEY *) &kkk->lower;
266 key.upper = (GBT_NUMKEY *) &kkk->upper;
267
268 PG_RETURN_BOOL(gbt_num_consistent(&key, &query, &strategy,
269 GIST_LEAF(entry), &tinfo, fcinfo->flinfo));
270}
271
272Datum
274{
275 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
277
278 /* Oid subtype = PG_GETARG_OID(3); */
279 tsKEY *kkk = (tsKEY *) DatumGetPointer(entry->key);
281
282 key.lower = (GBT_NUMKEY *) &kkk->lower;
283 key.upper = (GBT_NUMKEY *) &kkk->upper;
284
286 &tinfo, fcinfo->flinfo));
287}
288
289Datum
291{
292 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
295
296 /* Oid subtype = PG_GETARG_OID(3); */
297 bool *recheck = (bool *) PG_GETARG_POINTER(4);
298 char *kkk = (char *) DatumGetPointer(entry->key);
300 Timestamp qqq;
301
302 /* All cases served by this function are exact */
303 *recheck = false;
304
305 key.lower = (GBT_NUMKEY *) &kkk[0];
306 key.upper = (GBT_NUMKEY *) &kkk[MAXALIGN(tinfo.size)];
307 qqq = tstz_to_ts_gmt(query);
308
309 PG_RETURN_BOOL(gbt_num_consistent(&key, &qqq, &strategy,
310 GIST_LEAF(entry), &tinfo, fcinfo->flinfo));
311}
312
313Datum
315{
316 GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
318
319 /* Oid subtype = PG_GETARG_OID(3); */
320 char *kkk = (char *) DatumGetPointer(entry->key);
322 Timestamp qqq;
323
324 key.lower = (GBT_NUMKEY *) &kkk[0];
325 key.upper = (GBT_NUMKEY *) &kkk[MAXALIGN(tinfo.size)];
326 qqq = tstz_to_ts_gmt(query);
327
329 &tinfo, fcinfo->flinfo));
330}
331
332
333Datum
335{
337 void *out = palloc(sizeof(tsKEY));
338
339 *(int *) PG_GETARG_POINTER(1) = sizeof(tsKEY);
340 PG_RETURN_POINTER(gbt_num_union(out, entryvec, &tinfo, fcinfo->flinfo));
341}
342
343
344#define penalty_check_max_float(val) \
345 do { \
346 if ( val > FLT_MAX ) \
347 val = FLT_MAX; \
348 if ( val < -FLT_MAX ) \
349 val = -FLT_MAX; \
350 } while (0)
351
352
353Datum
355{
356 tsKEY *origentry = (tsKEY *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
357 tsKEY *newentry = (tsKEY *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
358 float *result = (float *) PG_GETARG_POINTER(2);
359
360 double orgdbl[2],
361 newdbl[2];
362
363 /*
364 * We are always using "double" timestamps here. Precision should be good
365 * enough.
366 */
367 orgdbl[0] = ((double) origentry->lower);
368 orgdbl[1] = ((double) origentry->upper);
369 newdbl[0] = ((double) newentry->lower);
370 newdbl[1] = ((double) newentry->upper);
371
372 penalty_check_max_float(orgdbl[0]);
373 penalty_check_max_float(orgdbl[1]);
374 penalty_check_max_float(newdbl[0]);
375 penalty_check_max_float(newdbl[1]);
376
377 penalty_num(result, orgdbl[0], orgdbl[1], newdbl[0], newdbl[1]);
378
379 PG_RETURN_POINTER(result);
380}
381
382
383Datum
385{
388 &tinfo, fcinfo->flinfo));
389}
390
391Datum
393{
394 tsKEY *b1 = (tsKEY *) PG_GETARG_POINTER(0);
395 tsKEY *b2 = (tsKEY *) PG_GETARG_POINTER(1);
396 bool *result = (bool *) PG_GETARG_POINTER(2);
397
398 *result = gbt_num_same((void *) b1, (void *) b2, &tinfo, fcinfo->flinfo);
399 PG_RETURN_POINTER(result);
400}
Datum timestamp_cmp(PG_FUNCTION_ARGS)
Definition: timestamp.c:2251
Datum timestamp_ge(PG_FUNCTION_ARGS)
Definition: timestamp.c:2242
Datum timestamp_lt(PG_FUNCTION_ARGS)
Definition: timestamp.c:2215
Datum timestamp_le(PG_FUNCTION_ARGS)
Definition: timestamp.c:2233
Datum timestamp_gt(PG_FUNCTION_ARGS)
Definition: timestamp.c:2224
Datum timestamp_mi(PG_FUNCTION_ARGS)
Definition: timestamp.c:2779
Datum timestamp_eq(PG_FUNCTION_ARGS)
Definition: timestamp.c:2197
@ gbt_t_ts
Definition: btree_gist.h:23
Interval * abs_interval(Interval *a)
Datum gbt_ts_compress(PG_FUNCTION_ARGS)
Definition: btree_ts.c:208
static bool gbt_tsle(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:77
#define TimestampGetDatumFast(X)
Definition: btree_ts.c:39
Datum gbt_ts_penalty(PG_FUNCTION_ARGS)
Definition: btree_ts.c:354
Datum gbt_tstz_compress(PG_FUNCTION_ARGS)
Definition: btree_ts.c:217
static bool gbt_tsgt(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:44
Datum gbt_ts_same(PG_FUNCTION_ARGS)
Definition: btree_ts.c:392
Datum gbt_tstz_consistent(PG_FUNCTION_ARGS)
Definition: btree_ts.c:290
Datum gbt_tstz_distance(PG_FUNCTION_ARGS)
Definition: btree_ts.c:314
Datum gbt_ts_distance(PG_FUNCTION_ARGS)
Definition: btree_ts.c:273
Datum gbt_ts_picksplit(PG_FUNCTION_ARGS)
Definition: btree_ts.c:384
#define penalty_check_max_float(val)
Definition: btree_ts.c:344
Datum ts_dist(PG_FUNCTION_ARGS)
Definition: btree_ts.c:147
Datum gbt_ts_union(PG_FUNCTION_ARGS)
Definition: btree_ts.c:334
Datum gbt_ts_fetch(PG_FUNCTION_ARGS)
Definition: btree_ts.c:243
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:88
static bool gbt_tsge(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:55
Datum tstz_dist(PG_FUNCTION_ARGS)
Definition: btree_ts.c:171
static Timestamp tstz_to_ts_gmt(TimestampTz ts)
Definition: btree_ts.c:200
Datum gbt_ts_consistent(PG_FUNCTION_ARGS)
Definition: btree_ts.c:251
static int gbt_tskey_cmp(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:100
static bool gbt_tseq(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:66
PG_FUNCTION_INFO_V1(gbt_ts_compress)
static float8 gbt_ts_dist(const void *a, const void *b, FmgrInfo *flinfo)
Definition: btree_ts.c:114
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:768
double float8
Definition: c.h:587
#define PG_INT64_MAX
Definition: c.h:549
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 DirectFunctionCall2(func, arg1, arg2)
Definition: fmgr.h:643
#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 b
Definition: isn.c:69
int a
Definition: isn.c:68
int i
Definition: isn.c:72
static const char gmt[]
Definition: localtime.c:53
void * palloc(Size size)
Definition: mcxt.c:1317
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
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
Definition: btree_ts.c:15
Timestamp lower
Definition: btree_ts.c:16
Timestamp upper
Definition: btree_ts.c:17
#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