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 : :
8403 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
8403 bruce@momjian.us 22 :UBC 0 : ltree_gist_in(PG_FUNCTION_ARGS)
23 : : {
8080 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
8403 bruce@momjian.us 32 : 0 : ltree_gist_out(PG_FUNCTION_ARGS)
33 : : {
8080 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 *
1986 akorotkov@postgresql 42 :CBC 16964 : ltree_gist_alloc(bool isalltrue, BITVECP sign, int siglen,
43 : : ltree *left, ltree *right)
44 : : {
45 [ + - + + ]: 22243 : int32 size = LTG_HDRSIZE + (isalltrue ? 0 : siglen) +
841 tgl@sss.pgh.pa.us 46 [ + + ]: 5279 : (left ? VARSIZE(left) + (right ? VARSIZE(right) : 0) : 0);
1986 akorotkov@postgresql 47 : 16964 : ltree_gist *result = palloc(size);
48 : :
49 : 16964 : SET_VARSIZE(result, size);
50 : :
51 [ + + ]: 16964 : if (siglen)
52 : : {
53 : 14952 : result->flag = 0;
54 : :
55 [ - + ]: 14952 : if (isalltrue)
1986 akorotkov@postgresql 56 :UBC 0 : result->flag |= LTG_ALLTRUE;
1986 akorotkov@postgresql 57 [ + + ]:CBC 14952 : else if (sign)
58 : 5065 : memcpy(LTG_SIGN(result), sign, siglen);
59 : : else
60 : 9887 : memset(LTG_SIGN(result), 0, siglen);
61 : :
62 [ + + ]: 14952 : if (left)
63 : : {
64 [ + - ]: 3267 : memcpy(LTG_LNODE(result, siglen), left, VARSIZE(left));
65 : :
66 [ + - + - : 3267 : if (!right || left == right || ISEQ(left, right))
+ + - + ]
1986 akorotkov@postgresql 67 :UBC 0 : result->flag |= LTG_NORIGHT;
68 : : else
1986 akorotkov@postgresql 69 [ - + - - :CBC 3267 : 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 : 16964 : return result;
80 : : }
81 : :
8403 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);
1986 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
8403 bruce@momjian.us 94 : 2109 : ltree_compress(PG_FUNCTION_ARGS)
95 : : {
96 : 2109 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
8439 97 : 2109 : GISTENTRY *retval = entry;
98 : :
8403 99 [ + + ]: 2109 : if (entry->leafkey)
100 : : { /* ltree */
2910 tgl@sss.pgh.pa.us 101 : 2012 : ltree *val = DatumGetLtreeP(entry->key);
1986 akorotkov@postgresql 102 : 2012 : ltree_gist *key = ltree_gist_alloc(false, NULL, 0, val, 0);
103 : :
8403 bruce@momjian.us 104 : 2012 : retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
8439 105 : 2012 : gistentryinit(*retval, PointerGetDatum(key),
106 : : entry->rel, entry->page,
107 : : entry->offset, false);
108 : : }
109 : 2109 : PG_RETURN_POINTER(retval);
110 : : }
111 : :
112 : : Datum
8403 113 : 97925 : ltree_decompress(PG_FUNCTION_ARGS)
114 : : {
115 : 97925 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
2910 tgl@sss.pgh.pa.us 116 : 97925 : ltree_gist *key = (ltree_gist *) PG_DETOAST_DATUM(entry->key);
117 : :
8403 bruce@momjian.us 118 [ - + ]: 97925 : if (PointerGetDatum(key) != entry->key)
119 : : {
8403 bruce@momjian.us 120 :UBC 0 : GISTENTRY *retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
121 : :
8439 122 : 0 : gistentryinit(*retval, PointerGetDatum(key),
123 : : entry->rel, entry->page,
124 : : entry->offset, false);
125 : 0 : PG_RETURN_POINTER(retval);
126 : : }
8403 bruce@momjian.us 127 :CBC 97925 : 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);
1270 akorotkov@postgresql 136 [ + - ]: 3199 : int siglen = LTREE_GET_SIGLEN();
137 : :
8439 bruce@momjian.us 138 : 3199 : *result = false;
8403 139 [ - + ]: 3199 : if (LTG_ISONENODE(a) != LTG_ISONENODE(b))
8403 bruce@momjian.us 140 :UBC 0 : PG_RETURN_POINTER(result);
141 : :
8403 bruce@momjian.us 142 [ - + ]:CBC 3199 : if (LTG_ISONENODE(a))
1459 michael@paquier.xyz 143 [ # # # # ]:UBC 0 : *result = ISEQ(LTG_NODE(a), LTG_NODE(b));
144 : : else
145 : : {
146 : : int32 i;
8403 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))
8403 bruce@momjian.us 151 :UBC 0 : PG_RETURN_POINTER(result);
152 : :
1986 akorotkov@postgresql 153 [ + - + - :CBC 3199 : if (!ISEQ(LTG_LNODE(a, siglen), LTG_LNODE(b, siglen)))
+ + + - +
- - + ]
8403 bruce@momjian.us 154 : 12 : PG_RETURN_POINTER(result);
1986 akorotkov@postgresql 155 [ - + - - : 3187 : if (!ISEQ(LTG_RNODE(a, siglen), LTG_RNODE(b, siglen)))
+ - + - -
+ - - + -
+ - + + -
+ - - + -
+ - - + -
- + - + -
+ + ]
8439 bruce@momjian.us 156 : 15 : PG_RETURN_POINTER(result);
157 : :
158 : 3172 : *result = true;
8403 159 [ + - ]: 3172 : if (!LTG_ISALLTRUE(a))
160 : : {
1986 akorotkov@postgresql 161 [ + + ]: 4600648 : LOOPBYTE(siglen)
162 : : {
6504 bruce@momjian.us 163 [ + + ]: 4597478 : if (sa[i] != sb[i])
164 : : {
165 : 2 : *result = false;
166 : 2 : break;
167 : : }
168 : : }
169 : : }
170 : : }
171 : :
8403 172 : 3172 : PG_RETURN_POINTER(result);
173 : : }
174 : :
175 : : static void
1986 akorotkov@postgresql 176 : 5765 : hashing(BITVECP sign, ltree *t, int siglen)
177 : : {
8403 bruce@momjian.us 178 : 5765 : int tlen = t->numlevel;
8439 179 : 5765 : ltree_level *cur = LTREE_FIRST(t);
180 : : int hash;
181 : :
8403 182 [ + + ]: 43663 : while (tlen > 0)
183 : : {
184 : 37898 : hash = ltree_crc32_sz(cur->name, cur->len);
1986 akorotkov@postgresql 185 : 37898 : HASH(sign, hash, siglen);
8439 bruce@momjian.us 186 : 37898 : cur = LEVEL_NEXT(cur);
187 : 37898 : tlen--;
188 : : }
189 : 5765 : }
190 : :
191 : : Datum
8403 192 : 3199 : ltree_union(PG_FUNCTION_ARGS)
193 : : {
7678 194 : 3199 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
8403 195 : 3199 : int *size = (int *) PG_GETARG_POINTER(1);
1270 akorotkov@postgresql 196 [ + - ]: 3199 : int siglen = LTREE_GET_SIGLEN();
1986 197 : 3199 : BITVECP base = palloc0(siglen);
198 : : int32 i,
199 : : j;
200 : : ltree_gist *result,
201 : : *cur;
8403 bruce@momjian.us 202 : 3199 : ltree *left = NULL,
203 : 3199 : *right = NULL,
204 : : *curtree;
205 : 3199 : bool isalltrue = false;
206 : :
7830 teodor@sigaev.ru 207 [ + + ]: 9597 : for (j = 0; j < entryvec->n; j++)
208 : : {
8439 bruce@momjian.us 209 : 6398 : cur = GETENTRY(entryvec, j);
8403 210 [ + + ]: 6398 : if (LTG_ISONENODE(cur))
211 : : {
8439 212 : 3199 : curtree = LTG_NODE(cur);
1986 akorotkov@postgresql 213 : 3199 : hashing(base, curtree, siglen);
8403 bruce@momjian.us 214 [ + - + + ]: 3199 : if (!left || ltree_compare(left, curtree) > 0)
215 : 12 : left = curtree;
216 [ + - + + ]: 3199 : if (!right || ltree_compare(right, curtree) < 0)
8439 217 : 15 : right = curtree;
218 : : }
219 : : else
220 : : {
8403 221 [ + - - + ]: 3199 : if (isalltrue || LTG_ISALLTRUE(cur))
8439 bruce@momjian.us 222 :UBC 0 : isalltrue = true;
223 : : else
224 : : {
8403 bruce@momjian.us 225 :CBC 3199 : BITVECP sc = LTG_SIGN(cur);
226 : :
1986 akorotkov@postgresql 227 [ + + ]: 4641399 : LOOPBYTE(siglen)
6504 bruce@momjian.us 228 : 4638200 : ((unsigned char *) base)[i] |= sc[i];
229 : : }
230 : :
1986 akorotkov@postgresql 231 [ + - ]: 3199 : curtree = LTG_LNODE(cur, siglen);
8403 bruce@momjian.us 232 [ - + - - ]: 3199 : if (!left || ltree_compare(left, curtree) > 0)
233 : 3199 : left = curtree;
1986 akorotkov@postgresql 234 [ - + - - : 3199 : curtree = LTG_RNODE(cur, siglen);
+ - + - ]
8403 bruce@momjian.us 235 [ - + - - ]: 3199 : if (!right || ltree_compare(right, curtree) < 0)
8439 236 : 3199 : right = curtree;
237 : : }
238 : : }
239 : :
8403 240 [ + - ]: 3199 : if (isalltrue == false)
241 : : {
8439 242 : 3199 : isalltrue = true;
1986 akorotkov@postgresql 243 [ + - ]: 3199 : LOOPBYTE(siglen)
244 : : {
6504 bruce@momjian.us 245 [ + - ]: 3199 : if (((unsigned char *) base)[i] != 0xff)
246 : : {
247 : 3199 : isalltrue = false;
248 : 3199 : break;
249 : : }
250 : : }
251 : : }
252 : :
1986 akorotkov@postgresql 253 : 3199 : result = ltree_gist_alloc(isalltrue, base, siglen, left, right);
254 : :
255 : 3199 : *size = VARSIZE(result);
256 : :
8403 bruce@momjian.us 257 : 3199 : PG_RETURN_POINTER(result);
258 : : }
259 : :
260 : : Datum
261 : 9636 : ltree_penalty(PG_FUNCTION_ARGS)
262 : : {
263 : 9636 : ltree_gist *origval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
264 : 9636 : ltree_gist *newval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
265 : 9636 : float *penalty = (float *) PG_GETARG_POINTER(2);
1270 akorotkov@postgresql 266 [ + - ]: 9636 : int siglen = LTREE_GET_SIGLEN();
267 : : int32 cmpr,
268 : : cmpl;
269 : :
1986 270 [ + - - - : 9636 : cmpl = ltree_compare(LTG_GETLNODE(origval, siglen), LTG_GETLNODE(newval, siglen));
- + + - ]
271 [ - + - + : 9636 : cmpr = ltree_compare(LTG_GETRNODE(newval, siglen), LTG_GETRNODE(origval, siglen));
- - + - +
- + - - -
- - - - -
- ]
272 : :
7625 tgl@sss.pgh.pa.us 273 : 9636 : *penalty = Max(cmpl, 0) + Max(cmpr, 0);
274 : :
8439 bruce@momjian.us 275 : 9636 : 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
8403 286 : 18271 : treekey_cmp(const void *a, const void *b)
287 : : {
2046 alvherre@alvh.no-ip. 288 : 36542 : return ltree_compare(((const RIX *) a)->r,
289 : 18271 : ((const RIX *) b)->r);
290 : : }
291 : :
292 : :
293 : : Datum
8403 bruce@momjian.us 294 : 34 : ltree_picksplit(PG_FUNCTION_ARGS)
295 : : {
7678 296 : 34 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
8403 297 : 34 : GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
1270 akorotkov@postgresql 298 [ + - ]: 34 : 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;
1986 310 : 34 : BITVECP ls = palloc0(siglen),
311 : 34 : rs = palloc0(siglen);
8403 bruce@momjian.us 312 : 34 : bool lisat = false,
1986 akorotkov@postgresql 313 : 34 : risat = false;
314 : :
7830 teodor@sigaev.ru 315 : 34 : maxoff = entryvec->n - 1;
8439 bruce@momjian.us 316 : 34 : nbytes = (maxoff + 2) * sizeof(OffsetNumber);
317 : 34 : v->spl_left = (OffsetNumber *) palloc(nbytes);
318 : 34 : v->spl_right = (OffsetNumber *) palloc(nbytes);
319 : 34 : v->spl_nleft = 0;
320 : 34 : v->spl_nright = 0;
321 : 34 : array = (RIX *) palloc(sizeof(RIX) * (maxoff + 1));
322 : :
323 : : /* copy the data into RIXes, and sort the RIXes */
8403 324 [ + + ]: 2624 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
325 : : {
8439 326 : 2590 : array[j].index = j;
2999 tgl@sss.pgh.pa.us 327 : 2590 : lu = GETENTRY(entryvec, j); /* use as tmp val */
1986 akorotkov@postgresql 328 [ + + + - ]: 2590 : array[j].r = LTG_GETLNODE(lu, siglen);
329 : : }
330 : :
942 peter@eisentraut.org 331 : 34 : qsort(&array[FirstOffsetNumber], maxoff - FirstOffsetNumber + 1,
332 : : sizeof(RIX), treekey_cmp);
333 : :
8439 bruce@momjian.us 334 : 34 : lu_l = lu_r = ru_l = ru_r = NULL;
8403 335 [ + + ]: 2624 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
336 : : {
2999 tgl@sss.pgh.pa.us 337 : 2590 : lu = GETENTRY(entryvec, array[j].index); /* use as tmp val */
8403 bruce@momjian.us 338 [ + + ]: 2590 : if (j <= (maxoff - FirstOffsetNumber + 1) / 2)
339 : : {
8439 340 : 1290 : v->spl_left[v->spl_nleft] = array[j].index;
341 : 1290 : v->spl_nleft++;
1986 akorotkov@postgresql 342 [ + + + + : 1290 : if (lu_r == NULL || ltree_compare(LTG_GETRNODE(lu, siglen), lu_r) > 0)
- + - - +
- + - +
+ ]
343 [ + + - + : 1289 : lu_r = LTG_GETRNODE(lu, siglen);
- - + - +
- ]
8403 bruce@momjian.us 344 [ + + ]: 1290 : if (LTG_ISONENODE(lu))
1986 akorotkov@postgresql 345 : 1278 : hashing(ls, LTG_NODE(lu), siglen);
346 : : else
347 : : {
8403 bruce@momjian.us 348 [ + - - + ]: 12 : if (lisat || LTG_ISALLTRUE(lu))
8439 bruce@momjian.us 349 :UBC 0 : lisat = true;
350 : : else
351 : : {
8403 bruce@momjian.us 352 :CBC 12 : BITVECP sc = LTG_SIGN(lu);
353 : :
1986 akorotkov@postgresql 354 [ + + ]: 24300 : LOOPBYTE(siglen)
6504 bruce@momjian.us 355 : 24288 : ((unsigned char *) ls)[i] |= sc[i];
356 : : }
357 : : }
358 : : }
359 : : else
360 : : {
8439 361 : 1300 : v->spl_right[v->spl_nright] = array[j].index;
362 : 1300 : v->spl_nright++;
1986 akorotkov@postgresql 363 [ + + + + : 1300 : if (ru_r == NULL || ltree_compare(LTG_GETRNODE(lu, siglen), ru_r) > 0)
- + - - +
- + - +
+ ]
364 [ + + - + : 1299 : ru_r = LTG_GETRNODE(lu, siglen);
- - + - +
- ]
8403 bruce@momjian.us 365 [ + + ]: 1300 : if (LTG_ISONENODE(lu))
1986 akorotkov@postgresql 366 : 1288 : hashing(rs, LTG_NODE(lu), siglen);
367 : : else
368 : : {
8403 bruce@momjian.us 369 [ + - - + ]: 12 : if (risat || LTG_ISALLTRUE(lu))
8439 bruce@momjian.us 370 :UBC 0 : risat = true;
371 : : else
372 : : {
8403 bruce@momjian.us 373 :CBC 12 : BITVECP sc = LTG_SIGN(lu);
374 : :
1986 akorotkov@postgresql 375 [ + + ]: 24300 : LOOPBYTE(siglen)
6504 bruce@momjian.us 376 : 24288 : ((unsigned char *) rs)[i] |= sc[i];
377 : : }
378 : : }
379 : : }
380 : : }
381 : :
8403 382 [ + - ]: 34 : if (lisat == false)
383 : : {
8439 384 : 34 : lisat = true;
1986 akorotkov@postgresql 385 [ + - ]: 34 : LOOPBYTE(siglen)
386 : : {
6504 bruce@momjian.us 387 [ + - ]: 34 : if (((unsigned char *) ls)[i] != 0xff)
388 : : {
389 : 34 : lisat = false;
390 : 34 : break;
391 : : }
392 : : }
393 : : }
394 : :
8403 395 [ + - ]: 34 : if (risat == false)
396 : : {
8439 397 : 34 : risat = true;
1986 akorotkov@postgresql 398 [ + - ]: 34 : LOOPBYTE(siglen)
399 : : {
6504 bruce@momjian.us 400 [ + - ]: 34 : if (((unsigned char *) rs)[i] != 0xff)
401 : : {
402 : 34 : risat = false;
403 : 34 : break;
404 : : }
405 : : }
406 : : }
407 : :
1986 akorotkov@postgresql 408 [ + + + - ]: 34 : lu_l = LTG_GETLNODE(GETENTRY(entryvec, array[FirstOffsetNumber].index), siglen);
409 : 34 : lu = ltree_gist_alloc(lisat, ls, siglen, lu_l, lu_r);
410 : :
411 [ + + + - ]: 34 : ru_l = LTG_GETLNODE(GETENTRY(entryvec, array[1 + ((maxoff - FirstOffsetNumber + 1) / 2)].index), siglen);
412 : 34 : ru = ltree_gist_alloc(risat, rs, siglen, ru_l, ru_r);
413 : :
414 : 34 : pfree(ls);
415 : 34 : pfree(rs);
416 : :
8439 bruce@momjian.us 417 : 34 : v->spl_ldatum = PointerGetDatum(lu);
418 : 34 : v->spl_rdatum = PointerGetDatum(ru);
419 : :
420 : 34 : PG_RETURN_POINTER(v);
421 : : }
422 : :
423 : : static bool
1986 akorotkov@postgresql 424 : 23 : gist_isparent(ltree_gist *key, ltree *query, int siglen)
425 : : {
4821 peter_e@gmx.net 426 : 23 : int32 numlevel = query->numlevel;
427 : : int i;
428 : :
8403 bruce@momjian.us 429 [ + + ]: 99 : for (i = query->numlevel; i >= 0; i--)
430 : : {
431 : 80 : query->numlevel = i;
1986 akorotkov@postgresql 432 [ - + + - : 84 : if (ltree_compare(query, LTG_GETLNODE(key, siglen)) >= 0 &&
+ + + - ]
433 [ - + - + : 4 : ltree_compare(query, LTG_GETRNODE(key, siglen)) <= 0)
- - + - +
- ]
434 : : {
8439 bruce@momjian.us 435 : 4 : query->numlevel = numlevel;
436 : 4 : return true;
437 : : }
438 : : }
439 : :
440 : 19 : query->numlevel = numlevel;
441 : 19 : return false;
442 : : }
443 : :
444 : : static ltree *
5931 445 : 46 : copy_ltree(ltree *src)
446 : : {
3469 andres@anarazel.de 447 : 46 : ltree *dst = (ltree *) palloc0(VARSIZE(src));
448 : :
6765 tgl@sss.pgh.pa.us 449 : 46 : memcpy(dst, src, VARSIZE(src));
6969 teodor@sigaev.ru 450 : 46 : return dst;
451 : : }
452 : :
453 : : static bool
1986 akorotkov@postgresql 454 : 23 : gist_ischild(ltree_gist *key, ltree *query, int siglen)
455 : : {
456 [ - + + - ]: 23 : ltree *left = copy_ltree(LTG_GETLNODE(key, siglen));
457 [ - + - + : 23 : ltree *right = copy_ltree(LTG_GETRNODE(key, siglen));
- - + - +
- ]
6912 bruce@momjian.us 458 : 23 : bool res = true;
459 : :
6969 teodor@sigaev.ru 460 [ + + ]: 23 : if (left->numlevel > query->numlevel)
461 : 17 : left->numlevel = query->numlevel;
462 : :
463 [ + + ]: 23 : if (ltree_compare(query, left) < 0)
464 : 19 : res = false;
465 : :
466 [ + + ]: 23 : if (right->numlevel > query->numlevel)
467 : 20 : right->numlevel = query->numlevel;
468 : :
469 [ + + - + ]: 23 : if (res && ltree_compare(query, right) > 0)
6969 teodor@sigaev.ru 470 :UBC 0 : res = false;
471 : :
6969 teodor@sigaev.ru 472 :CBC 23 : pfree(left);
473 : 23 : pfree(right);
474 : :
475 : 23 : return res;
476 : : }
477 : :
478 : : static bool
1986 akorotkov@postgresql 479 : 203 : gist_qe(ltree_gist *key, lquery *query, int siglen)
480 : : {
8403 bruce@momjian.us 481 : 203 : lquery_level *curq = LQUERY_FIRST(query);
482 : 203 : BITVECP sign = LTG_SIGN(key);
483 : 203 : int qlen = query->numlevel;
484 : :
485 [ - + ]: 203 : if (LTG_ISALLTRUE(key))
8439 bruce@momjian.us 486 :UBC 0 : return true;
487 : :
8403 bruce@momjian.us 488 [ + + ]:CBC 797 : while (qlen > 0)
489 : : {
490 [ + + + - ]: 594 : if (curq->numvar && LQL_CANLOOKSIGN(curq))
491 : : {
492 : 391 : bool isexist = false;
493 : 391 : int vlen = curq->numvar;
8439 494 : 391 : lquery_variant *curv = LQL_FIRST(curq);
495 : :
8403 496 [ + - ]: 391 : while (vlen > 0)
497 : : {
1986 akorotkov@postgresql 498 [ + - ]: 391 : if (GETBIT(sign, HASHVAL(curv->val, siglen)))
499 : : {
8403 bruce@momjian.us 500 : 391 : isexist = true;
8439 501 : 391 : break;
502 : : }
8439 bruce@momjian.us 503 :UBC 0 : curv = LVAR_NEXT(curv);
504 : 0 : vlen--;
505 : : }
8403 bruce@momjian.us 506 [ - + ]:CBC 391 : if (!isexist)
8403 bruce@momjian.us 507 :UBC 0 : return false;
508 : : }
509 : :
8439 bruce@momjian.us 510 :CBC 594 : curq = LQL_NEXT(curq);
511 : 594 : qlen--;
512 : : }
513 : :
514 : 203 : return true;
515 : : }
516 : :
517 : : static int
5931 518 : 266 : gist_tqcmp(ltree *t, lquery *q)
519 : : {
8439 520 : 266 : ltree_level *al = LTREE_FIRST(t);
521 : 266 : lquery_level *ql = LQUERY_FIRST(q);
522 : : lquery_variant *bl;
8403 523 : 266 : int an = t->numlevel;
524 : 266 : int bn = q->firstgood;
525 : 266 : int res = 0;
526 : :
527 [ + + + + ]: 314 : while (an > 0 && bn > 0)
528 : : {
8439 529 : 248 : bl = LQL_FIRST(ql);
5373 rhaas@postgresql.org 530 [ + + ]: 248 : if ((res = memcmp(al->name, bl->name, Min(al->len, bl->len))) == 0)
531 : : {
8403 bruce@momjian.us 532 [ + + ]: 82 : if (al->len != bl->len)
8439 533 : 34 : return al->len - bl->len;
534 : : }
535 : : else
536 : 166 : return res;
8403 537 : 48 : an--;
538 : 48 : bn--;
8439 539 : 48 : al = LEVEL_NEXT(al);
540 : 48 : ql = LQL_NEXT(ql);
541 : : }
542 : :
6970 teodor@sigaev.ru 543 : 66 : return Min(t->numlevel, q->firstgood) - q->firstgood;
544 : : }
545 : :
546 : : static bool
1986 akorotkov@postgresql 547 : 203 : gist_between(ltree_gist *key, lquery *query, int siglen)
548 : : {
8403 bruce@momjian.us 549 [ + + ]: 203 : if (query->firstgood == 0)
8439 550 : 38 : return true;
551 : :
1986 akorotkov@postgresql 552 [ - + + - : 165 : if (gist_tqcmp(LTG_GETLNODE(key, siglen), query) > 0)
+ + ]
6970 teodor@sigaev.ru 553 : 64 : return false;
554 : :
1986 akorotkov@postgresql 555 [ - + - + : 101 : if (gist_tqcmp(LTG_GETRNODE(key, siglen), query) < 0)
- - + - +
- + + ]
6970 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
1986 akorotkov@postgresql 568 : 76 : checkcondition_bit(void *cxt, ITEM *val)
569 : : {
570 : 76 : LtreeSignature *sig = cxt;
571 : :
572 [ + - + - ]: 76 : return (FLG_CANLOOKSIGN(val->flag)) ? GETBIT(sig->sign, HASHVAL(val->val, sig->siglen)) : true;
573 : : }
574 : :
575 : : static bool
576 : 38 : gist_qtxt(ltree_gist *key, ltxtquery *query, int siglen)
577 : : {
578 : : LtreeSignature sig;
579 : :
8403 bruce@momjian.us 580 [ - + ]: 38 : if (LTG_ISALLTRUE(key))
8439 bruce@momjian.us 581 :UBC 0 : return true;
582 : :
1986 akorotkov@postgresql 583 :CBC 38 : sig.sign = LTG_SIGN(key);
584 : 38 : sig.siglen = siglen;
585 : :
2046 alvherre@alvh.no-ip. 586 : 38 : return ltree_execute(GETQUERY(query),
587 : : &sig, false,
588 : : checkcondition_bit);
589 : : }
590 : :
591 : : static bool
1986 akorotkov@postgresql 592 : 31 : arrq_cons(ltree_gist *key, ArrayType *_query, int siglen)
593 : : {
7678 bruce@momjian.us 594 [ - + ]: 31 : lquery *query = (lquery *) ARR_DATA_PTR(_query);
595 : 31 : int num = ArrayGetNItems(ARR_NDIM(_query), ARR_DIMS(_query));
596 : :
5673 tgl@sss.pgh.pa.us 597 [ - + ]: 31 : if (ARR_NDIM(_query) > 1)
7678 bruce@momjian.us 598 [ # # ]:UBC 0 : ereport(ERROR,
599 : : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
600 : : errmsg("array must be one-dimensional")));
5354 tgl@sss.pgh.pa.us 601 [ - + ]:CBC 31 : if (array_contains_nulls(_query))
7231 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 : :
7678 bruce@momjian.us 606 [ + + ]:CBC 67 : while (num > 0)
607 : : {
1986 akorotkov@postgresql 608 [ + - + + ]: 49 : if (gist_qe(key, query, siglen) && gist_between(key, query, siglen))
7678 bruce@momjian.us 609 : 13 : return true;
610 : 36 : num--;
611 : 36 : query = NEXTVAL(query);
612 : : }
8235 613 : 18 : return false;
614 : : }
615 : :
616 : : Datum
8403 617 : 9785 : ltree_consistent(PG_FUNCTION_ARGS)
618 : : {
619 : 9785 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
8439 620 : 9785 : StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
621 : :
622 : : /* Oid subtype = PG_GETARG_OID(3); */
6354 tgl@sss.pgh.pa.us 623 : 9785 : bool *recheck = (bool *) PG_GETARG_POINTER(4);
1270 akorotkov@postgresql 624 [ + - ]: 9785 : int siglen = LTREE_GET_SIGLEN();
6354 tgl@sss.pgh.pa.us 625 : 9785 : ltree_gist *key = (ltree_gist *) DatumGetPointer(entry->key);
626 : 9785 : void *query = NULL;
8403 bruce@momjian.us 627 : 9785 : bool res = false;
628 : :
629 : : /* All cases served by this function are exact */
6354 tgl@sss.pgh.pa.us 630 : 9785 : *recheck = false;
631 : :
8403 bruce@momjian.us 632 [ + + + + : 9785 : switch (strategy)
+ + + + +
+ - ]
633 : : {
8439 634 : 311 : case BTLessStrategyNumber:
2910 tgl@sss.pgh.pa.us 635 : 311 : query = PG_GETARG_LTREE_P(1);
8403 bruce@momjian.us 636 : 311 : res = (GIST_LEAF(entry)) ?
637 : 296 : (ltree_compare((ltree *) query, LTG_NODE(key)) > 0)
8439 638 [ + + ]: 622 : :
1986 akorotkov@postgresql 639 [ - + + - ]: 15 : (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0);
8439 bruce@momjian.us 640 : 311 : break;
641 : 311 : case BTLessEqualStrategyNumber:
2910 tgl@sss.pgh.pa.us 642 : 311 : query = PG_GETARG_LTREE_P(1);
1986 akorotkov@postgresql 643 [ + + + - ]: 311 : res = (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0);
8439 bruce@momjian.us 644 : 311 : break;
645 : 292 : case BTEqualStrategyNumber:
2910 tgl@sss.pgh.pa.us 646 : 292 : query = PG_GETARG_LTREE_P(1);
8403 bruce@momjian.us 647 [ + + ]: 292 : if (GIST_LEAF(entry))
648 : 269 : res = (ltree_compare((ltree *) query, LTG_NODE(key)) == 0);
649 : : else
1986 akorotkov@postgresql 650 [ - + + - ]: 46 : res = (ltree_compare((ltree *) query, LTG_GETLNODE(key, siglen)) >= 0
8403 bruce@momjian.us 651 [ + + + + ]: 31 : &&
1986 akorotkov@postgresql 652 [ - + - + : 8 : ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8439 bruce@momjian.us 653 : 292 : break;
654 : 939 : case BTGreaterEqualStrategyNumber:
2910 tgl@sss.pgh.pa.us 655 : 939 : query = PG_GETARG_LTREE_P(1);
1986 akorotkov@postgresql 656 [ + + - + : 939 : res = (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8439 bruce@momjian.us 657 : 939 : break;
658 : 939 : case BTGreaterStrategyNumber:
2910 tgl@sss.pgh.pa.us 659 : 939 : query = PG_GETARG_LTREE_P(1);
8403 bruce@momjian.us 660 : 939 : res = (GIST_LEAF(entry)) ?
1986 akorotkov@postgresql 661 [ + - - - : 924 : (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) < 0)
- - - - -
- ]
8439 bruce@momjian.us 662 [ + + ]: 1878 : :
1986 akorotkov@postgresql 663 [ - + - + : 15 : (ltree_compare((ltree *) query, LTG_GETRNODE(key, siglen)) <= 0);
- - + - +
- ]
8439 bruce@momjian.us 664 : 939 : break;
665 : 187 : case 10:
2910 tgl@sss.pgh.pa.us 666 : 187 : query = PG_GETARG_LTREE_P_COPY(1);
8403 bruce@momjian.us 667 : 187 : res = (GIST_LEAF(entry)) ?
668 : 164 : inner_isparent((ltree *) query, LTG_NODE(key))
8439 669 [ + + ]: 351 : :
1986 akorotkov@postgresql 670 : 23 : gist_isparent(key, (ltree *) query, siglen);
8439 bruce@momjian.us 671 : 187 : break;
672 : 187 : case 11:
2910 tgl@sss.pgh.pa.us 673 : 187 : query = PG_GETARG_LTREE_P(1);
8403 bruce@momjian.us 674 : 187 : res = (GIST_LEAF(entry)) ?
675 : 164 : inner_isparent(LTG_NODE(key), (ltree *) query)
8439 676 [ + + ]: 351 : :
1986 akorotkov@postgresql 677 : 23 : gist_ischild(key, (ltree *) query, siglen);
8439 bruce@momjian.us 678 : 187 : break;
679 : 3986 : case 12:
680 : : case 13:
2910 tgl@sss.pgh.pa.us 681 : 3986 : query = PG_GETARG_LQUERY_P(1);
8403 bruce@momjian.us 682 [ + + ]: 3986 : if (GIST_LEAF(entry))
683 : 3832 : res = DatumGetBool(DirectFunctionCall2(ltq_regex,
684 : : PointerGetDatum(LTG_NODE(key)),
685 : : PointerGetDatum((lquery *) query)
686 : : ));
687 : : else
1986 akorotkov@postgresql 688 [ + - + + ]: 308 : res = (gist_qe(key, (lquery *) query, siglen) &&
689 : 154 : gist_between(key, (lquery *) query, siglen));
8403 bruce@momjian.us 690 : 3986 : break;
8439 691 : 2050 : case 14:
692 : : case 15:
2910 tgl@sss.pgh.pa.us 693 : 2050 : query = PG_GETARG_LTXTQUERY_P(1);
8403 bruce@momjian.us 694 [ + + ]: 2050 : if (GIST_LEAF(entry))
695 : 2012 : res = DatumGetBool(DirectFunctionCall2(ltxtq_exec,
696 : : PointerGetDatum(LTG_NODE(key)),
697 : : PointerGetDatum((ltxtquery *) query)
698 : : ));
699 : : else
1986 akorotkov@postgresql 700 : 38 : res = gist_qtxt(key, (ltxtquery *) query, siglen);
8403 bruce@momjian.us 701 : 2050 : break;
8235 702 : 583 : case 16:
703 : : case 17:
2910 tgl@sss.pgh.pa.us 704 : 583 : query = PG_GETARG_ARRAYTYPE_P(1);
8235 bruce@momjian.us 705 [ + + ]: 583 : if (GIST_LEAF(entry))
706 : 552 : res = DatumGetBool(DirectFunctionCall2(lt_q_regex,
707 : : PointerGetDatum(LTG_NODE(key)),
708 : : PointerGetDatum((ArrayType *) query)
709 : : ));
710 : : else
1986 akorotkov@postgresql 711 : 31 : res = arrq_cons(key, (ArrayType *) query, siglen);
8235 bruce@momjian.us 712 : 583 : break;
8439 bruce@momjian.us 713 :UBC 0 : default:
714 : : /* internal error */
8080 tgl@sss.pgh.pa.us 715 [ # # ]: 0 : elog(ERROR, "unrecognized StrategyNumber: %d", strategy);
716 : : }
717 : :
6912 bruce@momjian.us 718 [ + + ]:CBC 9785 : PG_FREE_IF_COPY(query, 1);
8439 719 : 9785 : PG_RETURN_BOOL(res);
720 : : }
721 : :
722 : : static void
867 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
1986 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));
867 746 : 14 : register_reloptions_validator(relopts, ltree_gist_relopts_validator);
747 : :
1986 748 : 14 : PG_RETURN_VOID();
749 : : }
|