Line data Source code
1 : /*
2 : * contrib/hstore/hstore_gist.c
3 : */
4 : #include "postgres.h"
5 :
6 : #include "access/gist.h"
7 : #include "access/reloptions.h"
8 : #include "access/stratnum.h"
9 : #include "catalog/pg_type.h"
10 : #include "common/int.h"
11 : #include "hstore.h"
12 : #include "utils/pg_crc.h"
13 :
14 : /* gist_hstore_ops opclass options */
15 : typedef struct
16 : {
17 : int32 vl_len_; /* varlena header (do not touch directly!) */
18 : int siglen; /* signature length in bytes */
19 : } GistHstoreOptions;
20 :
21 : /* bigint defines */
22 : #define BITBYTE 8
23 : #define SIGLEN_DEFAULT (sizeof(int32) * 4)
24 : #define SIGLEN_MAX GISTMaxIndexKeySize
25 : #define SIGLENBIT(siglen) ((siglen) * BITBYTE)
26 : #define GET_SIGLEN() (PG_HAS_OPCLASS_OPTIONS() ? \
27 : ((GistHstoreOptions *) PG_GET_OPCLASS_OPTIONS())->siglen : \
28 : SIGLEN_DEFAULT)
29 :
30 :
31 : typedef char *BITVECP;
32 :
33 : #define LOOPBYTE(siglen) \
34 : for (i = 0; i < (siglen); i++)
35 :
36 : #define LOOPBIT(siglen) \
37 : for (i = 0; i < SIGLENBIT(siglen); i++)
38 :
39 : /* beware of multiple evaluation of arguments to these macros! */
40 : #define GETBYTE(x,i) ( *( (BITVECP)(x) + (int)( (i) / BITBYTE ) ) )
41 : #define GETBITBYTE(x,i) ( (*((char*)(x)) >> (i)) & 0x01 )
42 : #define CLRBIT(x,i) GETBYTE(x,i) &= ~( 0x01 << ( (i) % BITBYTE ) )
43 : #define SETBIT(x,i) GETBYTE(x,i) |= ( 0x01 << ( (i) % BITBYTE ) )
44 : #define GETBIT(x,i) ( (GETBYTE(x,i) >> ( (i) % BITBYTE )) & 0x01 )
45 : #define HASHVAL(val, siglen) (((unsigned int)(val)) % SIGLENBIT(siglen))
46 : #define HASH(sign, val, siglen) SETBIT((sign), HASHVAL(val, siglen))
47 :
48 : typedef struct
49 : {
50 : int32 vl_len_; /* varlena header (do not touch directly!) */
51 : int32 flag;
52 : char data[FLEXIBLE_ARRAY_MEMBER];
53 : } GISTTYPE;
54 :
55 : #define ALLISTRUE 0x04
56 :
57 : #define ISALLTRUE(x) ( ((GISTTYPE*)x)->flag & ALLISTRUE )
58 :
59 : #define GTHDRSIZE (VARHDRSZ + sizeof(int32))
60 : #define CALCGTSIZE(flag, siglen) ( GTHDRSIZE+(((flag) & ALLISTRUE) ? 0 : (siglen)) )
61 :
62 : #define GETSIGN(x) ( (BITVECP)( (char*)x+GTHDRSIZE ) )
63 :
64 : #define SUMBIT(val) ( \
65 : GETBITBYTE((val),0) + \
66 : GETBITBYTE((val),1) + \
67 : GETBITBYTE((val),2) + \
68 : GETBITBYTE((val),3) + \
69 : GETBITBYTE((val),4) + \
70 : GETBITBYTE((val),5) + \
71 : GETBITBYTE((val),6) + \
72 : GETBITBYTE((val),7) \
73 : )
74 :
75 : #define GETENTRY(vec,pos) ((GISTTYPE *) DatumGetPointer((vec)->vector[(pos)].key))
76 :
77 : #define WISH_F(a,b,c) (double)( -(double)(((a)-(b))*((a)-(b))*((a)-(b)))*(c) )
78 :
79 : /* shorthand for calculating CRC-32 of a single chunk of data. */
80 : static pg_crc32
81 66132 : crc32_sz(const char *buf, int size)
82 : {
83 : pg_crc32 crc;
84 :
85 66132 : INIT_TRADITIONAL_CRC32(crc);
86 352534 : COMP_TRADITIONAL_CRC32(crc, buf, size);
87 66132 : FIN_TRADITIONAL_CRC32(crc);
88 :
89 66132 : return crc;
90 : }
91 :
92 :
93 14 : PG_FUNCTION_INFO_V1(ghstore_in);
94 14 : PG_FUNCTION_INFO_V1(ghstore_out);
95 :
96 :
97 : Datum
98 0 : ghstore_in(PG_FUNCTION_ARGS)
99 : {
100 0 : ereport(ERROR,
101 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
102 : errmsg("cannot accept a value of type %s", "ghstore")));
103 :
104 : PG_RETURN_VOID(); /* keep compiler quiet */
105 : }
106 :
107 : Datum
108 0 : ghstore_out(PG_FUNCTION_ARGS)
109 : {
110 0 : ereport(ERROR,
111 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
112 : errmsg("cannot display a value of type %s", "ghstore")));
113 :
114 : PG_RETURN_VOID(); /* keep compiler quiet */
115 : }
116 :
117 : static GISTTYPE *
118 18932 : ghstore_alloc(bool allistrue, int siglen, BITVECP sign)
119 : {
120 18932 : int flag = allistrue ? ALLISTRUE : 0;
121 18932 : int size = CALCGTSIZE(flag, siglen);
122 18932 : GISTTYPE *res = palloc(size);
123 :
124 18932 : SET_VARSIZE(res, size);
125 18932 : res->flag = flag;
126 :
127 18932 : if (!allistrue)
128 : {
129 18932 : if (sign)
130 2708 : memcpy(GETSIGN(res), sign, siglen);
131 : else
132 16224 : memset(GETSIGN(res), 0, siglen);
133 : }
134 :
135 18932 : return res;
136 : }
137 :
138 16 : PG_FUNCTION_INFO_V1(ghstore_consistent);
139 16 : PG_FUNCTION_INFO_V1(ghstore_compress);
140 16 : PG_FUNCTION_INFO_V1(ghstore_decompress);
141 16 : PG_FUNCTION_INFO_V1(ghstore_penalty);
142 16 : PG_FUNCTION_INFO_V1(ghstore_picksplit);
143 16 : PG_FUNCTION_INFO_V1(ghstore_union);
144 16 : PG_FUNCTION_INFO_V1(ghstore_same);
145 16 : PG_FUNCTION_INFO_V1(ghstore_options);
146 :
147 : Datum
148 16280 : ghstore_compress(PG_FUNCTION_ARGS)
149 : {
150 16280 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
151 16280 : int siglen = GET_SIGLEN();
152 16280 : GISTENTRY *retval = entry;
153 :
154 16280 : if (entry->leafkey)
155 : {
156 4004 : GISTTYPE *res = ghstore_alloc(false, siglen, NULL);
157 4004 : HStore *val = DatumGetHStoreP(entry->key);
158 4004 : HEntry *hsent = ARRPTR(val);
159 4004 : char *ptr = STRPTR(val);
160 4004 : int count = HS_COUNT(val);
161 : int i;
162 :
163 23128 : for (i = 0; i < count; ++i)
164 : {
165 : int h;
166 :
167 19124 : h = crc32_sz((char *) HSTORE_KEY(hsent, ptr, i),
168 19124 : HSTORE_KEYLEN(hsent, i));
169 19124 : HASH(GETSIGN(res), h, siglen);
170 19124 : if (!HSTORE_VALISNULL(hsent, i))
171 : {
172 19120 : h = crc32_sz((char *) HSTORE_VAL(hsent, ptr, i),
173 19120 : HSTORE_VALLEN(hsent, i));
174 19120 : HASH(GETSIGN(res), h, siglen);
175 : }
176 : }
177 :
178 4004 : retval = palloc_object(GISTENTRY);
179 4004 : gistentryinit(*retval, PointerGetDatum(res),
180 : entry->rel, entry->page,
181 : entry->offset,
182 : false);
183 : }
184 12276 : else if (!ISALLTRUE(DatumGetPointer(entry->key)))
185 : {
186 : int32 i;
187 : GISTTYPE *res;
188 12276 : BITVECP sign = GETSIGN(DatumGetPointer(entry->key));
189 :
190 12276 : LOOPBYTE(siglen)
191 : {
192 12276 : if ((sign[i] & 0xff) != 0xff)
193 12276 : PG_RETURN_POINTER(retval);
194 : }
195 :
196 0 : res = ghstore_alloc(true, siglen, NULL);
197 :
198 0 : retval = palloc_object(GISTENTRY);
199 0 : gistentryinit(*retval, PointerGetDatum(res),
200 : entry->rel, entry->page,
201 : entry->offset,
202 : false);
203 : }
204 :
205 4004 : PG_RETURN_POINTER(retval);
206 : }
207 :
208 : /*
209 : * Since type ghstore isn't toastable (and doesn't need to be),
210 : * this function can be a no-op.
211 : */
212 : Datum
213 96814 : ghstore_decompress(PG_FUNCTION_ARGS)
214 : {
215 96814 : PG_RETURN_POINTER(PG_GETARG_POINTER(0));
216 : }
217 :
218 : Datum
219 12220 : ghstore_same(PG_FUNCTION_ARGS)
220 : {
221 12220 : GISTTYPE *a = (GISTTYPE *) PG_GETARG_POINTER(0);
222 12220 : GISTTYPE *b = (GISTTYPE *) PG_GETARG_POINTER(1);
223 12220 : bool *result = (bool *) PG_GETARG_POINTER(2);
224 12220 : int siglen = GET_SIGLEN();
225 :
226 :
227 12220 : if (ISALLTRUE(a) && ISALLTRUE(b))
228 0 : *result = true;
229 12220 : else if (ISALLTRUE(a))
230 0 : *result = false;
231 12220 : else if (ISALLTRUE(b))
232 0 : *result = false;
233 : else
234 : {
235 : int32 i;
236 12220 : BITVECP sa = GETSIGN(a),
237 12220 : sb = GETSIGN(b);
238 :
239 12220 : *result = true;
240 9048244 : LOOPBYTE(siglen)
241 : {
242 9045592 : if (sa[i] != sb[i])
243 : {
244 9568 : *result = false;
245 9568 : break;
246 : }
247 : }
248 : }
249 12220 : PG_RETURN_POINTER(result);
250 : }
251 :
252 : static int32
253 0 : sizebitvec(BITVECP sign, int siglen)
254 : {
255 0 : int32 size = 0,
256 : i;
257 :
258 0 : LOOPBYTE(siglen)
259 : {
260 0 : size += SUMBIT(sign);
261 0 : sign = (BITVECP) (((char *) sign) + 1);
262 : }
263 0 : return size;
264 : }
265 :
266 : static int
267 265044 : hemdistsign(BITVECP a, BITVECP b, int siglen)
268 : {
269 : int i,
270 265044 : dist = 0;
271 :
272 773295316 : LOOPBIT(siglen)
273 : {
274 773030272 : if (GETBIT(a, i) != GETBIT(b, i))
275 6722970 : dist++;
276 : }
277 265044 : return dist;
278 : }
279 :
280 : static int
281 265044 : hemdist(GISTTYPE *a, GISTTYPE *b, int siglen)
282 : {
283 265044 : if (ISALLTRUE(a))
284 : {
285 0 : if (ISALLTRUE(b))
286 0 : return 0;
287 : else
288 0 : return SIGLENBIT(siglen) - sizebitvec(GETSIGN(b), siglen);
289 : }
290 265044 : else if (ISALLTRUE(b))
291 0 : return SIGLENBIT(siglen) - sizebitvec(GETSIGN(a), siglen);
292 :
293 265044 : return hemdistsign(GETSIGN(a), GETSIGN(b), siglen);
294 : }
295 :
296 : static int32
297 24440 : unionkey(BITVECP sbase, GISTTYPE *add, int siglen)
298 : {
299 : int32 i;
300 24440 : BITVECP sadd = GETSIGN(add);
301 :
302 24440 : if (ISALLTRUE(add))
303 0 : return 1;
304 43089496 : LOOPBYTE(siglen)
305 43065056 : sbase[i] |= sadd[i];
306 24440 : return 0;
307 : }
308 :
309 : Datum
310 12220 : ghstore_union(PG_FUNCTION_ARGS)
311 : {
312 12220 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
313 12220 : int32 len = entryvec->n;
314 :
315 12220 : int *size = (int *) PG_GETARG_POINTER(1);
316 12220 : int siglen = GET_SIGLEN();
317 : int32 i;
318 12220 : GISTTYPE *result = ghstore_alloc(false, siglen, NULL);
319 12220 : BITVECP base = GETSIGN(result);
320 :
321 36660 : for (i = 0; i < len; i++)
322 : {
323 24440 : if (unionkey(base, GETENTRY(entryvec, i), siglen))
324 : {
325 0 : result->flag |= ALLISTRUE;
326 0 : SET_VARSIZE(result, CALCGTSIZE(ALLISTRUE, siglen));
327 0 : break;
328 : }
329 : }
330 :
331 12220 : *size = VARSIZE(result);
332 :
333 12220 : PG_RETURN_POINTER(result);
334 : }
335 :
336 : Datum
337 31734 : ghstore_penalty(PG_FUNCTION_ARGS)
338 : {
339 31734 : GISTENTRY *origentry = (GISTENTRY *) PG_GETARG_POINTER(0); /* always ISSIGNKEY */
340 31734 : GISTENTRY *newentry = (GISTENTRY *) PG_GETARG_POINTER(1);
341 31734 : float *penalty = (float *) PG_GETARG_POINTER(2);
342 31734 : int siglen = GET_SIGLEN();
343 31734 : GISTTYPE *origval = (GISTTYPE *) DatumGetPointer(origentry->key);
344 31734 : GISTTYPE *newval = (GISTTYPE *) DatumGetPointer(newentry->key);
345 :
346 31734 : *penalty = hemdist(origval, newval, siglen);
347 31734 : PG_RETURN_POINTER(penalty);
348 : }
349 :
350 :
351 : typedef struct
352 : {
353 : OffsetNumber pos;
354 : int32 cost;
355 : } SPLITCOST;
356 :
357 : static int
358 12044 : comparecost(const void *a, const void *b)
359 : {
360 24088 : return pg_cmp_s32(((const SPLITCOST *) a)->cost,
361 12044 : ((const SPLITCOST *) b)->cost);
362 : }
363 :
364 :
365 : Datum
366 1354 : ghstore_picksplit(PG_FUNCTION_ARGS)
367 : {
368 1354 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
369 1354 : OffsetNumber maxoff = entryvec->n - 2;
370 :
371 1354 : GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
372 1354 : int siglen = GET_SIGLEN();
373 : OffsetNumber k,
374 : j;
375 : GISTTYPE *datum_l,
376 : *datum_r;
377 : BITVECP union_l,
378 : union_r;
379 : int32 size_alpha,
380 : size_beta;
381 : int32 size_waste,
382 1354 : waste = -1;
383 : int32 nbytes;
384 1354 : OffsetNumber seed_1 = 0,
385 1354 : seed_2 = 0;
386 : OffsetNumber *left,
387 : *right;
388 : BITVECP ptr;
389 : int i;
390 : SPLITCOST *costvector;
391 : GISTTYPE *_k,
392 : *_j;
393 :
394 1354 : nbytes = (maxoff + 2) * sizeof(OffsetNumber);
395 1354 : v->spl_left = (OffsetNumber *) palloc(nbytes);
396 1354 : v->spl_right = (OffsetNumber *) palloc(nbytes);
397 :
398 6062 : for (k = FirstOffsetNumber; k < maxoff; k = OffsetNumberNext(k))
399 : {
400 4708 : _k = GETENTRY(entryvec, k);
401 213770 : for (j = OffsetNumberNext(k); j <= maxoff; j = OffsetNumberNext(j))
402 : {
403 209062 : size_waste = hemdist(_k, GETENTRY(entryvec, j), siglen);
404 209062 : if (size_waste > waste)
405 : {
406 2258 : waste = size_waste;
407 2258 : seed_1 = k;
408 2258 : seed_2 = j;
409 : }
410 : }
411 : }
412 :
413 1354 : left = v->spl_left;
414 1354 : v->spl_nleft = 0;
415 1354 : right = v->spl_right;
416 1354 : v->spl_nright = 0;
417 :
418 1354 : if (seed_1 == 0 || seed_2 == 0)
419 : {
420 0 : seed_1 = 1;
421 0 : seed_2 = 2;
422 : }
423 :
424 : /* form initial .. */
425 1354 : datum_l = ghstore_alloc(ISALLTRUE(GETENTRY(entryvec, seed_1)), siglen,
426 1354 : GETSIGN(GETENTRY(entryvec, seed_1)));
427 1354 : datum_r = ghstore_alloc(ISALLTRUE(GETENTRY(entryvec, seed_2)), siglen,
428 1354 : GETSIGN(GETENTRY(entryvec, seed_2)));
429 :
430 1354 : maxoff = OffsetNumberNext(maxoff);
431 : /* sort before ... */
432 1354 : costvector = palloc_array(SPLITCOST, maxoff);
433 8770 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
434 : {
435 7416 : costvector[j - 1].pos = j;
436 7416 : _j = GETENTRY(entryvec, j);
437 7416 : size_alpha = hemdist(datum_l, _j, siglen);
438 7416 : size_beta = hemdist(datum_r, _j, siglen);
439 7416 : costvector[j - 1].cost = abs(size_alpha - size_beta);
440 : }
441 1354 : qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
442 :
443 1354 : union_l = GETSIGN(datum_l);
444 1354 : union_r = GETSIGN(datum_r);
445 :
446 8770 : for (k = 0; k < maxoff; k++)
447 : {
448 7416 : j = costvector[k].pos;
449 7416 : if (j == seed_1)
450 : {
451 1354 : *left++ = j;
452 1354 : v->spl_nleft++;
453 1354 : continue;
454 : }
455 6062 : else if (j == seed_2)
456 : {
457 1354 : *right++ = j;
458 1354 : v->spl_nright++;
459 1354 : continue;
460 : }
461 4708 : _j = GETENTRY(entryvec, j);
462 4708 : size_alpha = hemdist(datum_l, _j, siglen);
463 4708 : size_beta = hemdist(datum_r, _j, siglen);
464 :
465 4708 : if (size_alpha < size_beta + WISH_F(v->spl_nleft, v->spl_nright, 0.0001))
466 : {
467 2222 : if (ISALLTRUE(datum_l) || ISALLTRUE(_j))
468 : {
469 0 : if (!ISALLTRUE(datum_l))
470 0 : memset(union_l, 0xff, siglen);
471 : }
472 : else
473 : {
474 2222 : ptr = GETSIGN(_j);
475 2507614 : LOOPBYTE(siglen)
476 2505392 : union_l[i] |= ptr[i];
477 : }
478 2222 : *left++ = j;
479 2222 : v->spl_nleft++;
480 : }
481 : else
482 : {
483 2486 : if (ISALLTRUE(datum_r) || ISALLTRUE(_j))
484 : {
485 0 : if (!ISALLTRUE(datum_r))
486 0 : memset(union_r, 0xff, siglen);
487 : }
488 : else
489 : {
490 2486 : ptr = GETSIGN(_j);
491 2969926 : LOOPBYTE(siglen)
492 2967440 : union_r[i] |= ptr[i];
493 : }
494 2486 : *right++ = j;
495 2486 : v->spl_nright++;
496 : }
497 : }
498 :
499 1354 : *right = *left = FirstOffsetNumber;
500 :
501 1354 : v->spl_ldatum = PointerGetDatum(datum_l);
502 1354 : v->spl_rdatum = PointerGetDatum(datum_r);
503 :
504 1354 : PG_RETURN_POINTER(v);
505 : }
506 :
507 :
508 : Datum
509 21004 : ghstore_consistent(PG_FUNCTION_ARGS)
510 : {
511 21004 : GISTTYPE *entry = (GISTTYPE *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
512 21004 : StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
513 : #ifdef NOT_USED
514 : Oid subtype = PG_GETARG_OID(3);
515 : #endif
516 21004 : bool *recheck = (bool *) PG_GETARG_POINTER(4);
517 21004 : int siglen = GET_SIGLEN();
518 21004 : bool res = true;
519 : BITVECP sign;
520 :
521 : /* All cases served by this function are inexact */
522 21004 : *recheck = true;
523 :
524 21004 : if (ISALLTRUE(entry))
525 0 : PG_RETURN_BOOL(true);
526 :
527 21004 : sign = GETSIGN(entry);
528 :
529 21004 : if (strategy == HStoreContainsStrategyNumber ||
530 : strategy == HStoreOldContainsStrategyNumber)
531 9426 : {
532 9426 : HStore *query = PG_GETARG_HSTORE_P(1);
533 9426 : HEntry *qe = ARRPTR(query);
534 9426 : char *qv = STRPTR(query);
535 9426 : int count = HS_COUNT(query);
536 : int i;
537 :
538 19266 : for (i = 0; res && i < count; ++i)
539 : {
540 9840 : int crc = crc32_sz((char *) HSTORE_KEY(qe, qv, i),
541 9840 : HSTORE_KEYLEN(qe, i));
542 :
543 9840 : if (GETBIT(sign, HASHVAL(crc, siglen)))
544 : {
545 3922 : if (!HSTORE_VALISNULL(qe, i))
546 : {
547 2346 : crc = crc32_sz((char *) HSTORE_VAL(qe, qv, i),
548 2346 : HSTORE_VALLEN(qe, i));
549 2346 : if (!GETBIT(sign, HASHVAL(crc, siglen)))
550 1158 : res = false;
551 : }
552 : }
553 : else
554 5918 : res = false;
555 : }
556 : }
557 11578 : else if (strategy == HStoreExistsStrategyNumber)
558 : {
559 3906 : text *query = PG_GETARG_TEXT_PP(1);
560 3906 : int crc = crc32_sz(VARDATA_ANY(query), VARSIZE_ANY_EXHDR(query));
561 :
562 3906 : res = (GETBIT(sign, HASHVAL(crc, siglen))) ? true : false;
563 : }
564 7672 : else if (strategy == HStoreExistsAllStrategyNumber)
565 : {
566 3196 : ArrayType *query = PG_GETARG_ARRAYTYPE_P(1);
567 : Datum *key_datums;
568 : bool *key_nulls;
569 : int key_count;
570 : int i;
571 :
572 3196 : deconstruct_array_builtin(query, TEXTOID, &key_datums, &key_nulls, &key_count);
573 :
574 7632 : for (i = 0; res && i < key_count; ++i)
575 : {
576 : int crc;
577 :
578 4436 : if (key_nulls[i])
579 0 : continue;
580 4436 : crc = crc32_sz(VARDATA(DatumGetPointer(key_datums[i])), VARSIZE(DatumGetPointer(key_datums[i])) - VARHDRSZ);
581 4436 : if (!(GETBIT(sign, HASHVAL(crc, siglen))))
582 2560 : res = false;
583 : }
584 : }
585 4476 : else if (strategy == HStoreExistsAnyStrategyNumber)
586 : {
587 4476 : ArrayType *query = PG_GETARG_ARRAYTYPE_P(1);
588 : Datum *key_datums;
589 : bool *key_nulls;
590 : int key_count;
591 : int i;
592 :
593 4476 : deconstruct_array_builtin(query, TEXTOID, &key_datums, &key_nulls, &key_count);
594 :
595 4476 : res = false;
596 :
597 11836 : for (i = 0; !res && i < key_count; ++i)
598 : {
599 : int crc;
600 :
601 7360 : if (key_nulls[i])
602 0 : continue;
603 7360 : crc = crc32_sz(VARDATA(DatumGetPointer(key_datums[i])), VARSIZE(DatumGetPointer(key_datums[i])) - VARHDRSZ);
604 7360 : if (GETBIT(sign, HASHVAL(crc, siglen)))
605 2418 : res = true;
606 : }
607 : }
608 : else
609 0 : elog(ERROR, "Unsupported strategy number: %d", strategy);
610 :
611 21004 : PG_RETURN_BOOL(res);
612 : }
613 :
614 : Datum
615 20 : ghstore_options(PG_FUNCTION_ARGS)
616 : {
617 20 : local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
618 :
619 20 : init_local_reloptions(relopts, sizeof(GistHstoreOptions));
620 20 : add_local_int_reloption(relopts, "siglen",
621 : "signature length in bytes",
622 : SIGLEN_DEFAULT, 1, SIGLEN_MAX,
623 : offsetof(GistHstoreOptions, siglen));
624 :
625 20 : PG_RETURN_VOID();
626 : }
|