Age Owner Branch data TLA Line data Source code
1 : : /*
2 : : * GiST support for ltree
3 : : * Teodor Sigaev <teodor@stack.net>
4 : : * contrib/ltree/ltree_gist.c
5 : : */
6 : : #include "postgres.h"
7 : :
8 : : #include "access/gist.h"
9 : : #include "access/reloptions.h"
10 : : #include "access/stratnum.h"
11 : : #include "crc32.h"
12 : : #include "ltree.h"
13 : : #include "utils/array.h"
14 : :
15 : : #define NEXTVAL(x) ( (lquery*)( (char*)(x) + INTALIGN( VARSIZE(x) ) ) )
16 : : #define ISEQ(a,b) ( (a)->numlevel == (b)->numlevel && ltree_compare(a,b)==0 )
17 : :
8454 bruce@momjian.us 18 :CBC 2 : PG_FUNCTION_INFO_V1(ltree_gist_in);
19 : 2 : PG_FUNCTION_INFO_V1(ltree_gist_out);
20 : :
21 : : Datum
8454 bruce@momjian.us 22 :UBC 0 : ltree_gist_in(PG_FUNCTION_ARGS)
23 : : {
8131 tgl@sss.pgh.pa.us 24 [ # # ]: 0 : ereport(ERROR,
25 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
26 : : errmsg("cannot accept a value of type %s", "ltree_gist")));
27 : :
28 : : PG_RETURN_VOID(); /* keep compiler quiet */
29 : : }
30 : :
31 : : Datum
8454 bruce@momjian.us 32 : 0 : ltree_gist_out(PG_FUNCTION_ARGS)
33 : : {
8131 tgl@sss.pgh.pa.us 34 [ # # ]: 0 : ereport(ERROR,
35 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
36 : : errmsg("cannot display a value of type %s", "ltree_gist")));
37 : :
38 : : PG_RETURN_VOID(); /* keep compiler quiet */
39 : : }
40 : :
41 : : ltree_gist *
2037 akorotkov@postgresql 42 :CBC 17307 : ltree_gist_alloc(bool isalltrue, BITVECP sign, int siglen,
43 : : ltree *left, ltree *right)
44 : : {
45 [ + - + + ]: 22582 : int32 size = LTG_HDRSIZE + (isalltrue ? 0 : siglen) +
892 tgl@sss.pgh.pa.us 46 [ + + ]: 5275 : (left ? VARSIZE(left) + (right ? VARSIZE(right) : 0) : 0);
2037 akorotkov@postgresql 47 : 17307 : ltree_gist *result = palloc(size);
48 : :
49 : 17307 : SET_VARSIZE(result, size);
50 : :
51 [ + + ]: 17307 : if (siglen)
52 : : {
53 : 15295 : result->flag = 0;
54 : :
55 [ - + ]: 15295 : if (isalltrue)
2037 akorotkov@postgresql 56 :UBC 0 : result->flag |= LTG_ALLTRUE;
2037 akorotkov@postgresql 57 [ + + ]:CBC 15295 : else if (sign)
58 : 5089 : memcpy(LTG_SIGN(result), sign, siglen);
59 : : else
60 : 10206 : memset(LTG_SIGN(result), 0, siglen);
61 : :
62 [ + + ]: 15295 : if (left)
63 : : {
64 [ + - ]: 3263 : memcpy(LTG_LNODE(result, siglen), left, VARSIZE(left));
65 : :
66 [ + - + - : 3263 : if (!right || left == right || ISEQ(left, right))
+ + - + ]
2037 akorotkov@postgresql 67 :UBC 0 : result->flag |= LTG_NORIGHT;
68 : : else
2037 akorotkov@postgresql 69 [ - + - - :CBC 3263 : memcpy(LTG_RNODE(result, siglen), right, VARSIZE(right));
+ - + - ]
70 : : }
71 : : }
72 : : else
73 : : {
74 [ - + ]: 2012 : Assert(left);
75 : 2012 : result->flag = LTG_ONENODE;
76 : 2012 : memcpy(LTG_NODE(result), left, VARSIZE(left));
77 : : }
78 : :
79 : 17307 : return result;
80 : : }
81 : :
8454 bruce@momjian.us 82 : 3 : PG_FUNCTION_INFO_V1(ltree_compress);
83 : 3 : PG_FUNCTION_INFO_V1(ltree_decompress);
84 : 3 : PG_FUNCTION_INFO_V1(ltree_same);
85 : 3 : PG_FUNCTION_INFO_V1(ltree_union);
86 : 3 : PG_FUNCTION_INFO_V1(ltree_penalty);
87 : 3 : PG_FUNCTION_INFO_V1(ltree_picksplit);
88 : 3 : PG_FUNCTION_INFO_V1(ltree_consistent);
2037 akorotkov@postgresql 89 : 3 : PG_FUNCTION_INFO_V1(ltree_gist_options);
90 : :
91 : : #define GETENTRY(vec,pos) ((ltree_gist *) DatumGetPointer((vec)->vector[(pos)].key))
92 : :
93 : : Datum
8454 bruce@momjian.us 94 : 2106 : ltree_compress(PG_FUNCTION_ARGS)
95 : : {
96 : 2106 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
8490 97 : 2106 : GISTENTRY *retval = entry;
98 : :
8454 99 [ + + ]: 2106 : if (entry->leafkey)
100 : : { /* ltree */
2961 tgl@sss.pgh.pa.us 101 : 2012 : ltree *val = DatumGetLtreeP(entry->key);
2037 akorotkov@postgresql 102 : 2012 : ltree_gist *key = ltree_gist_alloc(false, NULL, 0, val, 0);
103 : :
8454 bruce@momjian.us 104 : 2012 : retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
8490 105 : 2012 : gistentryinit(*retval, PointerGetDatum(key),
106 : : entry->rel, entry->page,
107 : : entry->offset, false);
108 : : }
109 : 2106 : PG_RETURN_POINTER(retval);
110 : : }
111 : :
112 : : Datum
8454 113 : 100278 : ltree_decompress(PG_FUNCTION_ARGS)
114 : : {
115 : 100278 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
2961 tgl@sss.pgh.pa.us 116 : 100278 : ltree_gist *key = (ltree_gist *) PG_DETOAST_DATUM(entry->key);
117 : :
8454 bruce@momjian.us 118 [ - + ]: 100278 : if (PointerGetDatum(key) != entry->key)
119 : : {
8454 bruce@momjian.us 120 :UBC 0 : GISTENTRY *retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
121 : :
8490 122 : 0 : gistentryinit(*retval, PointerGetDatum(key),
123 : : entry->rel, entry->page,
124 : : entry->offset, false);
125 : 0 : PG_RETURN_POINTER(retval);
126 : : }
8454 bruce@momjian.us 127 :CBC 100278 : PG_RETURN_POINTER(entry);
128 : : }
129 : :
130 : : Datum
131 : 3199 : ltree_same(PG_FUNCTION_ARGS)
132 : : {
133 : 3199 : ltree_gist *a = (ltree_gist *) PG_GETARG_POINTER(0);
134 : 3199 : ltree_gist *b = (ltree_gist *) PG_GETARG_POINTER(1);
135 : 3199 : bool *result = (bool *) PG_GETARG_POINTER(2);
1321 akorotkov@postgresql 136 [ + - ]: 3199 : int siglen = LTREE_GET_SIGLEN();
137 : :
8490 bruce@momjian.us 138 : 3199 : *result = false;
8454 139 [ - + ]: 3199 : if (LTG_ISONENODE(a) != LTG_ISONENODE(b))
8454 bruce@momjian.us 140 :UBC 0 : PG_RETURN_POINTER(result);
141 : :
8454 bruce@momjian.us 142 [ - + ]:CBC 3199 : if (LTG_ISONENODE(a))
1510 michael@paquier.xyz 143 [ # # # # ]:UBC 0 : *result = ISEQ(LTG_NODE(a), LTG_NODE(b));
144 : : else
145 : : {
146 : : int32 i;
8454 bruce@momjian.us 147 :CBC 3199 : BITVECP sa = LTG_SIGN(a),
148 : 3199 : sb = LTG_SIGN(b);
149 : :
150 [ - + ]: 3199 : if (LTG_ISALLTRUE(a) != LTG_ISALLTRUE(b))
8454 bruce@momjian.us 151 :UBC 0 : PG_RETURN_POINTER(result);
152 : :
2037 akorotkov@postgresql 153 [ + - + - :CBC 3199 : if (!ISEQ(LTG_LNODE(a, siglen), LTG_LNODE(b, siglen)))
+ + + - +
- + + ]
8454 bruce@momjian.us 154 : 11 : PG_RETURN_POINTER(result);
2037 akorotkov@postgresql 155 [ - + - - : 3188 : if (!ISEQ(LTG_RNODE(a, siglen), LTG_RNODE(b, siglen)))
+ - + - -
+ - - + -
+ - + + -
+ - - + -
+ - - + -
- + - + -
+ + ]
8490 bruce@momjian.us 156 : 17 : PG_RETURN_POINTER(result);
157 : :
158 : 3171 : *result = true;
8454 159 [ + - ]: 3171 : if (!LTG_ISALLTRUE(a))
160 : : {
2037 akorotkov@postgresql 161 [ + + ]: 4596607 : LOOPBYTE(siglen)
162 : : {
6555 bruce@momjian.us 163 [ + + ]: 4593438 : if (sa[i] != sb[i])
164 : : {
165 : 2 : *result = false;
166 : 2 : break;
167 : : }
168 : : }
169 : : }
170 : : }
171 : :
8454 172 : 3171 : PG_RETURN_POINTER(result);
173 : : }
174 : :
175 : : static void
2037 akorotkov@postgresql 176 : 5590 : hashing(BITVECP sign, ltree *t, int siglen)
177 : : {
8454 bruce@momjian.us 178 : 5590 : int tlen = t->numlevel;
8490 179 : 5590 : ltree_level *cur = LTREE_FIRST(t);
180 : : int hash;
181 : :
8454 182 [ + + ]: 42274 : while (tlen > 0)
183 : : {
184 : 36684 : hash = ltree_crc32_sz(cur->name, cur->len);
2037 akorotkov@postgresql 185 : 36684 : HASH(sign, hash, siglen);
8490 bruce@momjian.us 186 : 36684 : cur = LEVEL_NEXT(cur);
187 : 36684 : tlen--;
188 : : }
189 : 5590 : }
190 : :
191 : : Datum
8454 192 : 3199 : ltree_union(PG_FUNCTION_ARGS)
193 : : {
7729 194 : 3199 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
8454 195 : 3199 : int *size = (int *) PG_GETARG_POINTER(1);
1321 akorotkov@postgresql 196 [ + - ]: 3199 : int siglen = LTREE_GET_SIGLEN();
2037 197 : 3199 : BITVECP base = palloc0(siglen);
198 : : int32 i,
199 : : j;
200 : : ltree_gist *result,
201 : : *cur;
8454 bruce@momjian.us 202 : 3199 : ltree *left = NULL,
203 : 3199 : *right = NULL,
204 : : *curtree;
205 : 3199 : bool isalltrue = false;
206 : :
7881 teodor@sigaev.ru 207 [ + + ]: 9597 : for (j = 0; j < entryvec->n; j++)
208 : : {
8490 bruce@momjian.us 209 : 6398 : cur = GETENTRY(entryvec, j);
8454 210 [ + + ]: 6398 : if (LTG_ISONENODE(cur))
211 : : {
8490 212 : 3199 : curtree = LTG_NODE(cur);
2037 akorotkov@postgresql 213 : 3199 : hashing(base, curtree, siglen);
8454 bruce@momjian.us 214 [ + - + + ]: 3199 : if (!left || ltree_compare(left, curtree) > 0)
215 : 11 : left = curtree;
216 [ + - + + ]: 3199 : if (!right || ltree_compare(right, curtree) < 0)
8490 217 : 17 : right = curtree;
218 : : }
219 : : else
220 : : {
8454 221 [ + - - + ]: 3199 : if (isalltrue || LTG_ISALLTRUE(cur))
8490 bruce@momjian.us 222 :UBC 0 : isalltrue = true;
223 : : else
224 : : {
8454 bruce@momjian.us 225 :CBC 3199 : BITVECP sc = LTG_SIGN(cur);
226 : :
2037 akorotkov@postgresql 227 [ + + ]: 4641399 : LOOPBYTE(siglen)
6555 bruce@momjian.us 228 : 4638200 : ((unsigned char *) base)[i] |= sc[i];
229 : : }
230 : :
2037 akorotkov@postgresql 231 [ + - ]: 3199 : curtree = LTG_LNODE(cur, siglen);
8454 bruce@momjian.us 232 [ - + - - ]: 3199 : if (!left || ltree_compare(left, curtree) > 0)
233 : 3199 : left = curtree;
2037 akorotkov@postgresql 234 [ - + - - : 3199 : curtree = LTG_RNODE(cur, siglen);
+ - + - ]
8454 bruce@momjian.us 235 [ - + - - ]: 3199 : if (!right || ltree_compare(right, curtree) < 0)
8490 236 : 3199 : right = curtree;
237 : : }
238 : : }
239 : :
8454 240 [ + - ]: 3199 : if (isalltrue == false)
241 : : {
8490 242 : 3199 : isalltrue = true;
2037 akorotkov@postgresql 243 [ + - ]: 3199 : LOOPBYTE(siglen)
244 : : {
6555 bruce@momjian.us 245 [ + - ]: 3199 : if (((unsigned char *) base)[i] != 0xff)
246 : : {
247 : 3199 : isalltrue = false;
248 : 3199 : break;
249 : : }
250 : : }
251 : : }
252 : :
2037 akorotkov@postgresql 253 : 3199 : result = ltree_gist_alloc(isalltrue, base, siglen, left, right);
254 : :
255 : 3199 : *size = VARSIZE(result);
256 : :
8454 bruce@momjian.us 257 : 3199 : PG_RETURN_POINTER(result);
258 : : }
259 : :
260 : : Datum
261 : 9662 : ltree_penalty(PG_FUNCTION_ARGS)
262 : : {
263 : 9662 : ltree_gist *origval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
264 : 9662 : ltree_gist *newval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
265 : 9662 : float *penalty = (float *) PG_GETARG_POINTER(2);
1321 akorotkov@postgresql 266 [ + - ]: 9662 : int siglen = LTREE_GET_SIGLEN();
267 : : int32 cmpr,
268 : : cmpl;
269 : :
2037 270 [ + - - - : 9662 : cmpl = ltree_compare(LTG_GETLNODE(origval, siglen), LTG_GETLNODE(newval, siglen));
- + + - ]
271 [ - + - + : 9662 : cmpr = ltree_compare(LTG_GETRNODE(newval, siglen), LTG_GETRNODE(origval, siglen));
- - + - +
- + - - -
- - - - -
- ]
272 : :
7676 tgl@sss.pgh.pa.us 273 : 9662 : *penalty = Max(cmpl, 0) + Max(cmpr, 0);
274 : :
8490 bruce@momjian.us 275 : 9662 : PG_RETURN_POINTER(penalty);
276 : : }
277 : :
278 : : /* used for sorting */
279 : : typedef struct rix
280 : : {
281 : : int index;
282 : : ltree *r;
283 : : } RIX;
284 : :
285 : : static int
8454 286 : 16822 : treekey_cmp(const void *a, const void *b)
287 : : {
2097 alvherre@alvh.no-ip. 288 : 33644 : return ltree_compare(((const RIX *) a)->r,
289 : 16822 : ((const RIX *) b)->r);
290 : : }
291 : :
292 : :
293 : : Datum
8454 bruce@momjian.us 294 : 32 : ltree_picksplit(PG_FUNCTION_ARGS)
295 : : {
7729 296 : 32 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
8454 297 : 32 : GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
1321 akorotkov@postgresql 298 [ + - ]: 32 : int siglen = LTREE_GET_SIGLEN();
299 : : OffsetNumber j;
300 : : int32 i;
301 : : RIX *array;
302 : : OffsetNumber maxoff;
303 : : int nbytes;
304 : : ltree *lu_l,
305 : : *lu_r,
306 : : *ru_l,
307 : : *ru_r;
308 : : ltree_gist *lu,
309 : : *ru;
2037 310 : 32 : BITVECP ls = palloc0(siglen),
311 : 32 : rs = palloc0(siglen);
8454 bruce@momjian.us 312 : 32 : bool lisat = false,
2037 akorotkov@postgresql 313 : 32 : risat = false;
314 : :
7881 teodor@sigaev.ru 315 : 32 : maxoff = entryvec->n - 1;
8490 bruce@momjian.us 316 : 32 : nbytes = (maxoff + 2) * sizeof(OffsetNumber);
317 : 32 : v->spl_left = (OffsetNumber *) palloc(nbytes);
318 : 32 : v->spl_right = (OffsetNumber *) palloc(nbytes);
319 : 32 : v->spl_nleft = 0;
320 : 32 : v->spl_nright = 0;
321 : 32 : array = (RIX *) palloc(sizeof(RIX) * (maxoff + 1));
322 : :
323 : : /* copy the data into RIXes, and sort the RIXes */
8454 324 [ + + ]: 2447 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
325 : : {
8490 326 : 2415 : array[j].index = j;
3050 tgl@sss.pgh.pa.us 327 : 2415 : lu = GETENTRY(entryvec, j); /* use as tmp val */
2037 akorotkov@postgresql 328 [ + + + - ]: 2415 : array[j].r = LTG_GETLNODE(lu, siglen);
329 : : }
330 : :
993 peter@eisentraut.org 331 : 32 : qsort(&array[FirstOffsetNumber], maxoff - FirstOffsetNumber + 1,
332 : : sizeof(RIX), treekey_cmp);
333 : :
8490 bruce@momjian.us 334 : 32 : lu_l = lu_r = ru_l = ru_r = NULL;
8454 335 [ + + ]: 2447 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
336 : : {
3050 tgl@sss.pgh.pa.us 337 : 2415 : lu = GETENTRY(entryvec, array[j].index); /* use as tmp val */
8454 bruce@momjian.us 338 [ + + ]: 2415 : if (j <= (maxoff - FirstOffsetNumber + 1) / 2)
339 : : {
8490 340 : 1202 : v->spl_left[v->spl_nleft] = array[j].index;
341 : 1202 : v->spl_nleft++;
2037 akorotkov@postgresql 342 [ + + + + : 1202 : if (lu_r == NULL || ltree_compare(LTG_GETRNODE(lu, siglen), lu_r) > 0)
- + - - +
- + - +
+ ]
343 [ + + - + : 1201 : lu_r = LTG_GETRNODE(lu, siglen);
- - + - +
- ]
8454 bruce@momjian.us 344 [ + + ]: 1202 : if (LTG_ISONENODE(lu))
2037 akorotkov@postgresql 345 : 1190 : hashing(ls, LTG_NODE(lu), siglen);
346 : : else
347 : : {
8454 bruce@momjian.us 348 [ + - - + ]: 12 : if (lisat || LTG_ISALLTRUE(lu))
8490 bruce@momjian.us 349 :UBC 0 : lisat = true;
350 : : else
351 : : {
8454 bruce@momjian.us 352 :CBC 12 : BITVECP sc = LTG_SIGN(lu);
353 : :
2037 akorotkov@postgresql 354 [ + + ]: 24300 : LOOPBYTE(siglen)
6555 bruce@momjian.us 355 : 24288 : ((unsigned char *) ls)[i] |= sc[i];
356 : : }
357 : : }
358 : : }
359 : : else
360 : : {
8490 361 : 1213 : v->spl_right[v->spl_nright] = array[j].index;
362 : 1213 : v->spl_nright++;
2037 akorotkov@postgresql 363 [ + + + + : 1213 : if (ru_r == NULL || ltree_compare(LTG_GETRNODE(lu, siglen), ru_r) > 0)
- + - - +
- + - +
+ ]
364 [ + + - + : 1212 : ru_r = LTG_GETRNODE(lu, siglen);
- - + - +
- ]
8454 bruce@momjian.us 365 [ + + ]: 1213 : if (LTG_ISONENODE(lu))
2037 akorotkov@postgresql 366 : 1201 : hashing(rs, LTG_NODE(lu), siglen);
367 : : else
368 : : {
8454 bruce@momjian.us 369 [ + - - + ]: 12 : if (risat || LTG_ISALLTRUE(lu))
8490 bruce@momjian.us 370 :UBC 0 : risat = true;
371 : : else
372 : : {
8454 bruce@momjian.us 373 :CBC 12 : BITVECP sc = LTG_SIGN(lu);
374 : :
2037 akorotkov@postgresql 375 [ + + ]: 24300 : LOOPBYTE(siglen)
6555 bruce@momjian.us 376 : 24288 : ((unsigned char *) rs)[i] |= sc[i];
377 : : }
378 : : }
379 : : }
380 : : }
381 : :
8454 382 [ + - ]: 32 : if (lisat == false)
383 : : {
8490 384 : 32 : lisat = true;
2037 akorotkov@postgresql 385 [ + - ]: 32 : LOOPBYTE(siglen)
386 : : {
6555 bruce@momjian.us 387 [ + - ]: 32 : if (((unsigned char *) ls)[i] != 0xff)
388 : : {
389 : 32 : lisat = false;
390 : 32 : break;
391 : : }
392 : : }
393 : : }
394 : :
8454 395 [ + - ]: 32 : if (risat == false)
396 : : {
8490 397 : 32 : risat = true;
2037 akorotkov@postgresql 398 [ + - ]: 32 : LOOPBYTE(siglen)
399 : : {
6555 bruce@momjian.us 400 [ + - ]: 32 : if (((unsigned char *) rs)[i] != 0xff)
401 : : {
402 : 32 : risat = false;
403 : 32 : break;
404 : : }
405 : : }
406 : : }
407 : :
2037 akorotkov@postgresql 408 [ + + + - ]: 32 : lu_l = LTG_GETLNODE(GETENTRY(entryvec, array[FirstOffsetNumber].index), siglen);
409 : 32 : lu = ltree_gist_alloc(lisat, ls, siglen, lu_l, lu_r);
410 : :
411 [ + + + - ]: 32 : ru_l = LTG_GETLNODE(GETENTRY(entryvec, array[1 + ((maxoff - FirstOffsetNumber + 1) / 2)].index), siglen);
412 : 32 : ru = ltree_gist_alloc(risat, rs, siglen, ru_l, ru_r);
413 : :
414 : 32 : pfree(ls);
415 : 32 : pfree(rs);
416 : :
8490 bruce@momjian.us 417 : 32 : v->spl_ldatum = PointerGetDatum(lu);
418 : 32 : v->spl_rdatum = PointerGetDatum(ru);
419 : :
420 : 32 : PG_RETURN_POINTER(v);
421 : : }
422 : :
423 : : static bool
2037 akorotkov@postgresql 424 : 22 : gist_isparent(ltree_gist *key, ltree *query, int siglen)
425 : : {
4872 peter_e@gmx.net 426 : 22 : int32 numlevel = query->numlevel;
427 : : int i;
428 : :
8454 bruce@momjian.us 429 [ + + ]: 94 : for (i = query->numlevel; i >= 0; i--)
430 : : {
431 : 76 : query->numlevel = i;
2037 akorotkov@postgresql 432 [ - + + - : 80 : if (ltree_compare(query, LTG_GETLNODE(key, siglen)) >= 0 &&
+ + + - ]
433 [ - + - + : 4 : ltree_compare(query, LTG_GETRNODE(key, siglen)) <= 0)
- - + - +
- ]
434 : : {
8490 bruce@momjian.us 435 : 4 : query->numlevel = numlevel;
436 : 4 : return true;
437 : : }
438 : : }
439 : :
440 : 18 : query->numlevel = numlevel;
441 : 18 : return false;
442 : : }
443 : :
444 : : static ltree *
5982 445 : 44 : copy_ltree(ltree *src)
446 : : {
3520 andres@anarazel.de 447 : 44 : ltree *dst = (ltree *) palloc0(VARSIZE(src));
448 : :
6816 tgl@sss.pgh.pa.us 449 : 44 : memcpy(dst, src, VARSIZE(src));
7020 teodor@sigaev.ru 450 : 44 : return dst;
451 : : }
452 : :
453 : : static bool
2037 akorotkov@postgresql 454 : 22 : gist_ischild(ltree_gist *key, ltree *query, int siglen)
455 : : {
456 [ - + + - ]: 22 : ltree *left = copy_ltree(LTG_GETLNODE(key, siglen));
457 [ - + - + : 22 : ltree *right = copy_ltree(LTG_GETRNODE(key, siglen));
- - + - +
- ]
6963 bruce@momjian.us 458 : 22 : bool res = true;
459 : :
7020 teodor@sigaev.ru 460 [ + + ]: 22 : if (left->numlevel > query->numlevel)
461 : 14 : left->numlevel = query->numlevel;
462 : :
463 [ + + ]: 22 : if (ltree_compare(query, left) < 0)
464 : 18 : res = false;
465 : :
466 [ + + ]: 22 : if (right->numlevel > query->numlevel)
467 : 20 : right->numlevel = query->numlevel;
468 : :
469 [ + + - + ]: 22 : if (res && ltree_compare(query, right) > 0)
7020 teodor@sigaev.ru 470 :UBC 0 : res = false;
471 : :
7020 teodor@sigaev.ru 472 :CBC 22 : pfree(left);
473 : 22 : pfree(right);
474 : :
475 : 22 : return res;
476 : : }
477 : :
478 : : static bool
2037 akorotkov@postgresql 479 : 190 : gist_qe(ltree_gist *key, lquery *query, int siglen)
480 : : {
8454 bruce@momjian.us 481 : 190 : lquery_level *curq = LQUERY_FIRST(query);
482 : 190 : BITVECP sign = LTG_SIGN(key);
483 : 190 : int qlen = query->numlevel;
484 : :
485 [ - + ]: 190 : if (LTG_ISALLTRUE(key))
8490 bruce@momjian.us 486 :UBC 0 : return true;
487 : :
8454 bruce@momjian.us 488 [ + + ]:CBC 746 : while (qlen > 0)
489 : : {
490 [ + + + - ]: 556 : if (curq->numvar && LQL_CANLOOKSIGN(curq))
491 : : {
492 : 366 : bool isexist = false;
493 : 366 : int vlen = curq->numvar;
8490 494 : 366 : lquery_variant *curv = LQL_FIRST(curq);
495 : :
8454 496 [ + - ]: 366 : while (vlen > 0)
497 : : {
2037 akorotkov@postgresql 498 [ + - ]: 366 : if (GETBIT(sign, HASHVAL(curv->val, siglen)))
499 : : {
8454 bruce@momjian.us 500 : 366 : isexist = true;
8490 501 : 366 : break;
502 : : }
8490 bruce@momjian.us 503 :UBC 0 : curv = LVAR_NEXT(curv);
504 : 0 : vlen--;
505 : : }
8454 bruce@momjian.us 506 [ - + ]:CBC 366 : if (!isexist)
8454 bruce@momjian.us 507 :UBC 0 : return false;
508 : : }
509 : :
8490 bruce@momjian.us 510 :CBC 556 : curq = LQL_NEXT(curq);
511 : 556 : qlen--;
512 : : }
513 : :
514 : 190 : return true;
515 : : }
516 : :
517 : : static int
5982 518 : 255 : gist_tqcmp(ltree *t, lquery *q)
519 : : {
8490 520 : 255 : ltree_level *al = LTREE_FIRST(t);
521 : 255 : lquery_level *ql = LQUERY_FIRST(q);
522 : : lquery_variant *bl;
8454 523 : 255 : int an = t->numlevel;
524 : 255 : int bn = q->firstgood;
525 : 255 : int res = 0;
526 : :
527 [ + + + + ]: 303 : while (an > 0 && bn > 0)
528 : : {
8490 529 : 237 : bl = LQL_FIRST(ql);
5424 rhaas@postgresql.org 530 [ + + ]: 237 : if ((res = memcmp(al->name, bl->name, Min(al->len, bl->len))) == 0)
531 : : {
8454 bruce@momjian.us 532 [ + + ]: 82 : if (al->len != bl->len)
8490 533 : 34 : return al->len - bl->len;
534 : : }
535 : : else
536 : 155 : return res;
8454 537 : 48 : an--;
538 : 48 : bn--;
8490 539 : 48 : al = LEVEL_NEXT(al);
540 : 48 : ql = LQL_NEXT(ql);
541 : : }
542 : :
7021 teodor@sigaev.ru 543 : 66 : return Min(t->numlevel, q->firstgood) - q->firstgood;
544 : : }
545 : :
546 : : static bool
2037 akorotkov@postgresql 547 : 190 : gist_between(ltree_gist *key, lquery *query, int siglen)
548 : : {
8454 bruce@momjian.us 549 [ + + ]: 190 : if (query->firstgood == 0)
8490 550 : 36 : return true;
551 : :
2037 akorotkov@postgresql 552 [ - + + - : 154 : if (gist_tqcmp(LTG_GETLNODE(key, siglen), query) > 0)
+ + ]
7021 teodor@sigaev.ru 553 : 53 : return false;
554 : :
2037 akorotkov@postgresql 555 [ - + - + : 101 : if (gist_tqcmp(LTG_GETRNODE(key, siglen), query) < 0)
- - + - +
- + + ]
7021 teodor@sigaev.ru 556 : 45 : return false;
557 : :
558 : 56 : return true;
559 : : }
560 : :
561 : : typedef struct LtreeSignature
562 : : {
563 : : BITVECP sign;
564 : : int siglen;
565 : : } LtreeSignature;
566 : :
567 : : static bool
2037 akorotkov@postgresql 568 : 72 : checkcondition_bit(void *cxt, ITEM *val)
569 : : {
570 : 72 : LtreeSignature *sig = cxt;
571 : :
572 [ + - + - ]: 72 : return (FLG_CANLOOKSIGN(val->flag)) ? GETBIT(sig->sign, HASHVAL(val->val, sig->siglen)) : true;
573 : : }
574 : :
575 : : static bool
576 : 36 : gist_qtxt(ltree_gist *key, ltxtquery *query, int siglen)
577 : : {
578 : : LtreeSignature sig;
579 : :
8454 bruce@momjian.us 580 [ - + ]: 36 : if (LTG_ISALLTRUE(key))
8490 bruce@momjian.us 581 :UBC 0 : return true;
582 : :
2037 akorotkov@postgresql 583 :CBC 36 : sig.sign = LTG_SIGN(key);
584 : 36 : sig.siglen = siglen;
585 : :
2097 alvherre@alvh.no-ip. 586 : 36 : return ltree_execute(GETQUERY(query),
587 : : &sig, false,
588 : : checkcondition_bit);
589 : : }
590 : :
591 : : static bool
2037 akorotkov@postgresql 592 : 29 : arrq_cons(ltree_gist *key, ArrayType *_query, int siglen)
593 : : {
7729 bruce@momjian.us 594 [ - + ]: 29 : lquery *query = (lquery *) ARR_DATA_PTR(_query);
595 : 29 : int num = ArrayGetNItems(ARR_NDIM(_query), ARR_DIMS(_query));
596 : :
5724 tgl@sss.pgh.pa.us 597 [ - + ]: 29 : if (ARR_NDIM(_query) > 1)
7729 bruce@momjian.us 598 [ # # ]:UBC 0 : ereport(ERROR,
599 : : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
600 : : errmsg("array must be one-dimensional")));
5405 tgl@sss.pgh.pa.us 601 [ - + ]:CBC 29 : if (array_contains_nulls(_query))
7282 tgl@sss.pgh.pa.us 602 [ # # ]:UBC 0 : ereport(ERROR,
603 : : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
604 : : errmsg("array must not contain nulls")));
605 : :
7729 bruce@momjian.us 606 [ + + ]:CBC 61 : while (num > 0)
607 : : {
2037 akorotkov@postgresql 608 [ + - + + ]: 45 : if (gist_qe(key, query, siglen) && gist_between(key, query, siglen))
7729 bruce@momjian.us 609 : 13 : return true;
610 : 32 : num--;
611 : 32 : query = NEXTVAL(query);
612 : : }
8286 613 : 16 : return false;
614 : : }
615 : :
616 : : Datum
8454 617 : 10068 : ltree_consistent(PG_FUNCTION_ARGS)
618 : : {
619 : 10068 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
8490 620 : 10068 : StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
621 : :
622 : : /* Oid subtype = PG_GETARG_OID(3); */
6405 tgl@sss.pgh.pa.us 623 : 10068 : bool *recheck = (bool *) PG_GETARG_POINTER(4);
1321 akorotkov@postgresql 624 [ + - ]: 10068 : int siglen = LTREE_GET_SIGLEN();
6405 tgl@sss.pgh.pa.us 625 : 10068 : ltree_gist *key = (ltree_gist *) DatumGetPointer(entry->key);
626 : 10068 : void *query = NULL;
8454 bruce@momjian.us 627 : 10068 : bool res = false;
628 : :
629 : : /* All cases served by this function are exact */
6405 tgl@sss.pgh.pa.us 630 : 10068 : *recheck = false;
631 : :
8454 bruce@momjian.us 632 [ + + + + : 10068 : switch (strategy)
+ + + + +
+ - ]
633 : : {
8490 634 : 330 : case BTLessStrategyNumber:
2961 tgl@sss.pgh.pa.us 635 : 330 : query = PG_GETARG_LTREE_P(1);
8454 bruce@momjian.us 636 : 330 : res = (GIST_LEAF(entry)) ?
637 : 316 : (ltree_compare((ltree *) query, LTG_NODE(key)) > 0)
8490 638 [ + + ]: 660 : :
2037 akorotkov@postgresql 639 [ - + + - ]: 14 : (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0);
8490 bruce@momjian.us 640 : 330 : break;
641 : 330 : case BTLessEqualStrategyNumber:
2961 tgl@sss.pgh.pa.us 642 : 330 : query = PG_GETARG_LTREE_P(1);
2037 akorotkov@postgresql 643 [ + + + - ]: 330 : res = (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0);
8490 bruce@momjian.us 644 : 330 : break;
645 : 311 : case BTEqualStrategyNumber:
2961 tgl@sss.pgh.pa.us 646 : 311 : query = PG_GETARG_LTREE_P(1);
8454 bruce@momjian.us 647 [ + + ]: 311 : if (GIST_LEAF(entry))
648 : 289 : res = (ltree_compare((ltree *) query, LTG_NODE(key)) == 0);
649 : : else
2037 akorotkov@postgresql 650 [ - + + - ]: 44 : res = (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0
8454 bruce@momjian.us 651 [ + + + + ]: 30 : &&
2037 akorotkov@postgresql 652 [ - + - + : 8 : ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8490 bruce@momjian.us 653 : 311 : break;
654 : 938 : case BTGreaterEqualStrategyNumber:
2961 tgl@sss.pgh.pa.us 655 : 938 : query = PG_GETARG_LTREE_P(1);
2037 akorotkov@postgresql 656 [ + + - + : 938 : res = (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8490 bruce@momjian.us 657 : 938 : break;
658 : 938 : case BTGreaterStrategyNumber:
2961 tgl@sss.pgh.pa.us 659 : 938 : query = PG_GETARG_LTREE_P(1);
8454 bruce@momjian.us 660 : 938 : res = (GIST_LEAF(entry)) ?
2037 akorotkov@postgresql 661 [ + - - - : 924 : (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) < 0)
- - - - -
- ]
8490 bruce@momjian.us 662 [ + + ]: 1876 : :
2037 akorotkov@postgresql 663 [ - + - + : 14 : (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8490 bruce@momjian.us 664 : 938 : break;
665 : 187 : case 10:
2961 tgl@sss.pgh.pa.us 666 : 187 : query = PG_GETARG_LTREE_P_COPY(1);
8454 bruce@momjian.us 667 : 187 : res = (GIST_LEAF(entry)) ?
668 : 165 : inner_isparent((ltree *) query, LTG_NODE(key))
8490 669 [ + + ]: 352 : :
2037 akorotkov@postgresql 670 : 22 : gist_isparent(key, (ltree *) query, siglen);
8490 bruce@momjian.us 671 : 187 : break;
672 : 187 : case 11:
2961 tgl@sss.pgh.pa.us 673 : 187 : query = PG_GETARG_LTREE_P(1);
8454 bruce@momjian.us 674 : 187 : res = (GIST_LEAF(entry)) ?
675 : 165 : inner_isparent(LTG_NODE(key), (ltree *) query)
8490 676 [ + + ]: 352 : :
2037 akorotkov@postgresql 677 : 22 : gist_ischild(key, (ltree *) query, siglen);
8490 bruce@momjian.us 678 : 187 : break;
679 : 4158 : case 12:
680 : : case 13:
2961 tgl@sss.pgh.pa.us 681 : 4158 : query = PG_GETARG_LQUERY_P(1);
8454 bruce@momjian.us 682 [ + + ]: 4158 : if (GIST_LEAF(entry))
683 : 4013 : res = DatumGetBool(DirectFunctionCall2(ltq_regex,
684 : : PointerGetDatum(LTG_NODE(key)),
685 : : PointerGetDatum((lquery *) query)
686 : : ));
687 : : else
2037 akorotkov@postgresql 688 [ + - + + ]: 290 : res = (gist_qe(key, (lquery *) query, siglen) &&
689 : 145 : gist_between(key, (lquery *) query, siglen));
8454 bruce@momjian.us 690 : 4158 : break;
8490 691 : 2048 : case 14:
692 : : case 15:
2961 tgl@sss.pgh.pa.us 693 : 2048 : query = PG_GETARG_LTXTQUERY_P(1);
8454 bruce@momjian.us 694 [ + + ]: 2048 : if (GIST_LEAF(entry))
695 : 2012 : res = DatumGetBool(DirectFunctionCall2(ltxtq_exec,
696 : : PointerGetDatum(LTG_NODE(key)),
697 : : PointerGetDatum((ltxtquery *) query)
698 : : ));
699 : : else
2037 akorotkov@postgresql 700 : 36 : res = gist_qtxt(key, (ltxtquery *) query, siglen);
8454 bruce@momjian.us 701 : 2048 : break;
8286 702 : 641 : case 16:
703 : : case 17:
2961 tgl@sss.pgh.pa.us 704 : 641 : query = PG_GETARG_ARRAYTYPE_P(1);
8286 bruce@momjian.us 705 [ + + ]: 641 : if (GIST_LEAF(entry))
706 : 612 : res = DatumGetBool(DirectFunctionCall2(lt_q_regex,
707 : : PointerGetDatum(LTG_NODE(key)),
708 : : PointerGetDatum((ArrayType *) query)
709 : : ));
710 : : else
2037 akorotkov@postgresql 711 : 29 : res = arrq_cons(key, (ArrayType *) query, siglen);
8286 bruce@momjian.us 712 : 641 : break;
8490 bruce@momjian.us 713 :UBC 0 : default:
714 : : /* internal error */
8131 tgl@sss.pgh.pa.us 715 [ # # ]: 0 : elog(ERROR, "unrecognized StrategyNumber: %d", strategy);
716 : : }
717 : :
6963 bruce@momjian.us 718 [ + + ]:CBC 10068 : PG_FREE_IF_COPY(query, 1);
8490 719 : 10068 : PG_RETURN_BOOL(res);
720 : : }
721 : :
722 : : static void
918 akorotkov@postgresql 723 : 3 : ltree_gist_relopts_validator(void *parsed_options, relopt_value *vals,
724 : : int nvals)
725 : : {
726 : 3 : LtreeGistOptions *options = (LtreeGistOptions *) parsed_options;
727 : :
728 [ + + ]: 3 : if (options->siglen != INTALIGN(options->siglen))
729 [ + - ]: 1 : ereport(ERROR,
730 : : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
731 : : errmsg("siglen value must be a multiple of %d", ALIGNOF_INT)));
732 : 2 : }
733 : :
734 : : Datum
2037 735 : 14 : ltree_gist_options(PG_FUNCTION_ARGS)
736 : : {
737 : 14 : local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
738 : :
739 : 14 : init_local_reloptions(relopts, sizeof(LtreeGistOptions));
740 : 14 : add_local_int_reloption(relopts, "siglen",
741 : : "signature length in bytes",
742 : : LTREE_SIGLEN_DEFAULT,
743 : : INTALIGN(1),
744 : : LTREE_SIGLEN_MAX,
745 : : offsetof(LtreeGistOptions, siglen));
918 746 : 14 : register_reloptions_validator(relopts, ltree_gist_relopts_validator);
747 : :
2037 748 : 14 : PG_RETURN_VOID();
749 : : }
|