Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * tsgistidx.c
4 : * GiST support functions for tsvector_ops
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : *
8 : *
9 : * IDENTIFICATION
10 : * src/backend/utils/adt/tsgistidx.c
11 : *
12 : *-------------------------------------------------------------------------
13 : */
14 :
15 : #include "postgres.h"
16 :
17 : #include "access/gist.h"
18 : #include "access/heaptoast.h"
19 : #include "access/reloptions.h"
20 : #include "common/int.h"
21 : #include "lib/qunique.h"
22 : #include "port/pg_bitutils.h"
23 : #include "tsearch/ts_utils.h"
24 : #include "utils/fmgrprotos.h"
25 : #include "utils/pg_crc.h"
26 :
27 :
28 : /* tsvector_ops opclass options */
29 : typedef struct
30 : {
31 : int32 vl_len_; /* varlena header (do not touch directly!) */
32 : int siglen; /* signature length */
33 : } GistTsVectorOptions;
34 :
35 : #define SIGLEN_DEFAULT (31 * 4)
36 : #define SIGLEN_MAX GISTMaxIndexKeySize
37 : #define GET_SIGLEN() (PG_HAS_OPCLASS_OPTIONS() ? \
38 : ((GistTsVectorOptions *) PG_GET_OPCLASS_OPTIONS())->siglen : \
39 : SIGLEN_DEFAULT)
40 :
41 : #define SIGLENBIT(siglen) ((siglen) * BITS_PER_BYTE)
42 :
43 : typedef char *BITVECP;
44 :
45 : #define LOOPBYTE(siglen) \
46 : for (i = 0; i < siglen; i++)
47 :
48 : #define GETBYTE(x,i) ( *( (BITVECP)(x) + (int)( (i) / BITS_PER_BYTE ) ) )
49 : #define GETBITBYTE(x,i) ( ((char)(x)) >> (i) & 0x01 )
50 : #define CLRBIT(x,i) GETBYTE(x,i) &= ~( 0x01 << ( (i) % BITS_PER_BYTE ) )
51 : #define SETBIT(x,i) GETBYTE(x,i) |= ( 0x01 << ( (i) % BITS_PER_BYTE ) )
52 : #define GETBIT(x,i) ( (GETBYTE(x,i) >> ( (i) % BITS_PER_BYTE )) & 0x01 )
53 :
54 : #define HASHVAL(val, siglen) (((unsigned int)(val)) % SIGLENBIT(siglen))
55 : #define HASH(sign, val, siglen) SETBIT((sign), HASHVAL(val, siglen))
56 :
57 : #define GETENTRY(vec,pos) ((SignTSVector *) DatumGetPointer((vec)->vector[(pos)].key))
58 :
59 : /*
60 : * type of GiST index key
61 : */
62 :
63 : typedef struct
64 : {
65 : int32 vl_len_; /* varlena header (do not touch directly!) */
66 : int32 flag;
67 : char data[FLEXIBLE_ARRAY_MEMBER];
68 : } SignTSVector;
69 :
70 : #define ARRKEY 0x01
71 : #define SIGNKEY 0x02
72 : #define ALLISTRUE 0x04
73 :
74 : #define ISARRKEY(x) ( ((SignTSVector*)(x))->flag & ARRKEY )
75 : #define ISSIGNKEY(x) ( ((SignTSVector*)(x))->flag & SIGNKEY )
76 : #define ISALLTRUE(x) ( ((SignTSVector*)(x))->flag & ALLISTRUE )
77 :
78 : #define GTHDRSIZE ( VARHDRSZ + sizeof(int32) )
79 : #define CALCGTSIZE(flag, len) ( GTHDRSIZE + ( ( (flag) & ARRKEY ) ? ((len)*sizeof(int32)) : (((flag) & ALLISTRUE) ? 0 : (len)) ) )
80 :
81 : #define GETSIGN(x) ( (BITVECP)( (char*)(x)+GTHDRSIZE ) )
82 : #define GETSIGLEN(x)( VARSIZE(x) - GTHDRSIZE )
83 : #define GETARR(x) ( (int32*)( (char*)(x)+GTHDRSIZE ) )
84 : #define ARRNELEM(x) ( ( VARSIZE(x) - GTHDRSIZE )/sizeof(int32) )
85 :
86 : static int32 sizebitvec(BITVECP sign, int siglen);
87 :
88 : Datum
89 0 : gtsvectorin(PG_FUNCTION_ARGS)
90 : {
91 : /* There's no need to support input of gtsvectors */
92 0 : ereport(ERROR,
93 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
94 : errmsg("cannot accept a value of type %s", "gtsvector")));
95 :
96 : PG_RETURN_VOID(); /* keep compiler quiet */
97 : }
98 :
99 : Datum
100 0 : gtsvectorout(PG_FUNCTION_ARGS)
101 : {
102 0 : SignTSVector *key = (SignTSVector *) PG_DETOAST_DATUM(PG_GETARG_DATUM(0));
103 : char *outbuf;
104 :
105 0 : if (ISARRKEY(key))
106 0 : outbuf = psprintf("%d unique words", (int) ARRNELEM(key));
107 : else
108 : {
109 0 : if (ISALLTRUE(key))
110 0 : outbuf = pstrdup("all true bits");
111 : else
112 : {
113 0 : int siglen = GETSIGLEN(key);
114 0 : int cnttrue = sizebitvec(GETSIGN(key), siglen);
115 :
116 0 : outbuf = psprintf("%d true bits, %d false bits",
117 0 : cnttrue, (int) SIGLENBIT(siglen) - cnttrue);
118 : }
119 : }
120 :
121 0 : PG_FREE_IF_COPY(key, 0);
122 0 : PG_RETURN_POINTER(outbuf);
123 : }
124 :
125 : static int
126 3626136 : compareint(const void *va, const void *vb)
127 : {
128 3626136 : int32 a = *((const int32 *) va);
129 3626136 : int32 b = *((const int32 *) vb);
130 :
131 3626136 : return pg_cmp_s32(a, b);
132 : }
133 :
134 : static void
135 74850 : makesign(BITVECP sign, SignTSVector *a, int siglen)
136 : {
137 : int32 k,
138 74850 : len = ARRNELEM(a);
139 74850 : int32 *ptr = GETARR(a);
140 :
141 74850 : MemSet(sign, 0, siglen);
142 4183252 : for (k = 0; k < len; k++)
143 4108402 : HASH(sign, ptr[k], siglen);
144 74850 : }
145 :
146 : static SignTSVector *
147 19424 : gtsvector_alloc(int flag, int len, BITVECP sign)
148 : {
149 19424 : int size = CALCGTSIZE(flag, len);
150 19424 : SignTSVector *res = palloc(size);
151 :
152 19424 : SET_VARSIZE(res, size);
153 19424 : res->flag = flag;
154 :
155 19424 : if ((flag & (SIGNKEY | ALLISTRUE)) == SIGNKEY && sign)
156 812 : memcpy(GETSIGN(res), sign, len);
157 :
158 19424 : return res;
159 : }
160 :
161 :
162 : Datum
163 15676 : gtsvector_compress(PG_FUNCTION_ARGS)
164 : {
165 15676 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
166 15676 : int siglen = GET_SIGLEN();
167 15676 : GISTENTRY *retval = entry;
168 :
169 15676 : if (entry->leafkey)
170 : { /* tsvector */
171 9144 : TSVector val = DatumGetTSVector(entry->key);
172 9144 : SignTSVector *res = gtsvector_alloc(ARRKEY, val->size, NULL);
173 : int32 len;
174 : int32 *arr;
175 9144 : WordEntry *ptr = ARRPTR(val);
176 9144 : char *words = STRPTR(val);
177 :
178 9144 : arr = GETARR(res);
179 9144 : len = val->size;
180 527544 : while (len--)
181 : {
182 : pg_crc32 c;
183 :
184 518400 : INIT_LEGACY_CRC32(c);
185 1555200 : COMP_LEGACY_CRC32(c, words + ptr->pos, ptr->len);
186 518400 : FIN_LEGACY_CRC32(c);
187 :
188 518400 : *arr = *(int32 *) &c;
189 518400 : arr++;
190 518400 : ptr++;
191 : }
192 :
193 9144 : qsort(GETARR(res), val->size, sizeof(int), compareint);
194 9144 : len = qunique(GETARR(res), val->size, sizeof(int), compareint);
195 9144 : if (len != val->size)
196 : {
197 : /*
198 : * there is a collision of hash-function; len is always less than
199 : * val->size
200 : */
201 0 : len = CALCGTSIZE(ARRKEY, len);
202 0 : res = (SignTSVector *) repalloc(res, len);
203 0 : SET_VARSIZE(res, len);
204 : }
205 :
206 : /* make signature, if array is too long */
207 9144 : if (VARSIZE(res) > TOAST_INDEX_TARGET)
208 : {
209 0 : SignTSVector *ressign = gtsvector_alloc(SIGNKEY, siglen, NULL);
210 :
211 0 : makesign(GETSIGN(ressign), res, siglen);
212 0 : res = ressign;
213 : }
214 :
215 9144 : retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
216 9144 : gistentryinit(*retval, PointerGetDatum(res),
217 : entry->rel, entry->page,
218 : entry->offset, false);
219 : }
220 6532 : else if (ISSIGNKEY(DatumGetPointer(entry->key)) &&
221 6532 : !ISALLTRUE(DatumGetPointer(entry->key)))
222 : {
223 : int32 i;
224 : SignTSVector *res;
225 6532 : BITVECP sign = GETSIGN(DatumGetPointer(entry->key));
226 :
227 6904 : LOOPBYTE(siglen)
228 : {
229 6532 : if ((sign[i] & 0xff) != 0xff)
230 6160 : PG_RETURN_POINTER(retval);
231 : }
232 :
233 372 : res = gtsvector_alloc(SIGNKEY | ALLISTRUE, siglen, sign);
234 372 : retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
235 372 : gistentryinit(*retval, PointerGetDatum(res),
236 : entry->rel, entry->page,
237 : entry->offset, false);
238 : }
239 9516 : PG_RETURN_POINTER(retval);
240 : }
241 :
242 : Datum
243 403670 : gtsvector_decompress(PG_FUNCTION_ARGS)
244 : {
245 : /*
246 : * We need to detoast the stored value, because the other gtsvector
247 : * support functions don't cope with toasted values.
248 : */
249 403670 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
250 403670 : SignTSVector *key = (SignTSVector *) PG_DETOAST_DATUM(entry->key);
251 :
252 403670 : if (key != (SignTSVector *) DatumGetPointer(entry->key))
253 : {
254 0 : GISTENTRY *retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
255 :
256 0 : gistentryinit(*retval, PointerGetDatum(key),
257 : entry->rel, entry->page,
258 : entry->offset, false);
259 :
260 0 : PG_RETURN_POINTER(retval);
261 : }
262 :
263 403670 : PG_RETURN_POINTER(entry);
264 : }
265 :
266 : typedef struct
267 : {
268 : int32 *arrb;
269 : int32 *arre;
270 : } CHKVAL;
271 :
272 : /*
273 : * TS_execute callback for matching a tsquery operand to GIST leaf-page data
274 : */
275 : static TSTernaryValue
276 415518 : checkcondition_arr(void *checkval, QueryOperand *val, ExecPhraseData *data)
277 : {
278 415518 : int32 *StopLow = ((CHKVAL *) checkval)->arrb;
279 415518 : int32 *StopHigh = ((CHKVAL *) checkval)->arre;
280 : int32 *StopMiddle;
281 :
282 : /* Loop invariant: StopLow <= val < StopHigh */
283 :
284 : /*
285 : * we are not able to find a prefix by hash value
286 : */
287 415518 : if (val->prefix)
288 24384 : return TS_MAYBE;
289 :
290 2514456 : while (StopLow < StopHigh)
291 : {
292 2171654 : StopMiddle = StopLow + (StopHigh - StopLow) / 2;
293 2171654 : if (*StopMiddle == val->valcrc)
294 48332 : return TS_MAYBE;
295 2123322 : else if (*StopMiddle < val->valcrc)
296 902440 : StopLow = StopMiddle + 1;
297 : else
298 1220882 : StopHigh = StopMiddle;
299 : }
300 :
301 342802 : return TS_NO;
302 : }
303 :
304 : /*
305 : * TS_execute callback for matching a tsquery operand to GIST non-leaf data
306 : */
307 : static TSTernaryValue
308 15692 : checkcondition_bit(void *checkval, QueryOperand *val, ExecPhraseData *data)
309 : {
310 15692 : void *key = (SignTSVector *) checkval;
311 :
312 : /*
313 : * we are not able to find a prefix in signature tree
314 : */
315 15692 : if (val->prefix)
316 700 : return TS_MAYBE;
317 :
318 14992 : if (GETBIT(GETSIGN(key), HASHVAL(val->valcrc, GETSIGLEN(key))))
319 14060 : return TS_MAYBE;
320 : else
321 932 : return TS_NO;
322 : }
323 :
324 : Datum
325 301442 : gtsvector_consistent(PG_FUNCTION_ARGS)
326 : {
327 301442 : GISTENTRY *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
328 301442 : TSQuery query = PG_GETARG_TSQUERY(1);
329 :
330 : /* StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2); */
331 : /* Oid subtype = PG_GETARG_OID(3); */
332 301442 : bool *recheck = (bool *) PG_GETARG_POINTER(4);
333 301442 : SignTSVector *key = (SignTSVector *) DatumGetPointer(entry->key);
334 :
335 : /* All cases served by this function are inexact */
336 301442 : *recheck = true;
337 :
338 301442 : if (!query->size)
339 0 : PG_RETURN_BOOL(false);
340 :
341 301442 : if (ISSIGNKEY(key))
342 : {
343 13326 : if (ISALLTRUE(key))
344 4800 : PG_RETURN_BOOL(true);
345 :
346 8526 : PG_RETURN_BOOL(TS_execute(GETQUERY(query),
347 : key,
348 : TS_EXEC_PHRASE_NO_POS,
349 : checkcondition_bit));
350 : }
351 : else
352 : { /* only leaf pages */
353 : CHKVAL chkval;
354 :
355 288116 : chkval.arrb = GETARR(key);
356 288116 : chkval.arre = chkval.arrb + ARRNELEM(key);
357 288116 : PG_RETURN_BOOL(TS_execute(GETQUERY(query),
358 : &chkval,
359 : TS_EXEC_PHRASE_NO_POS,
360 : checkcondition_arr));
361 : }
362 : }
363 :
364 : static int32
365 15372 : unionkey(BITVECP sbase, SignTSVector *add, int siglen)
366 : {
367 : int32 i;
368 :
369 15372 : if (ISSIGNKEY(add))
370 : {
371 9096 : BITVECP sadd = GETSIGN(add);
372 :
373 9096 : if (ISALLTRUE(add))
374 2820 : return 1;
375 :
376 : Assert(GETSIGLEN(add) == siglen);
377 :
378 2028660 : LOOPBYTE(siglen)
379 2022384 : sbase[i] |= sadd[i];
380 : }
381 : else
382 : {
383 6276 : int32 *ptr = GETARR(add);
384 :
385 369230 : for (i = 0; i < ARRNELEM(add); i++)
386 362954 : HASH(sbase, ptr[i], siglen);
387 : }
388 12552 : return 0;
389 : }
390 :
391 :
392 : Datum
393 9096 : gtsvector_union(PG_FUNCTION_ARGS)
394 : {
395 9096 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
396 9096 : int *size = (int *) PG_GETARG_POINTER(1);
397 9096 : int siglen = GET_SIGLEN();
398 9096 : SignTSVector *result = gtsvector_alloc(SIGNKEY, siglen, NULL);
399 9096 : BITVECP base = GETSIGN(result);
400 : int32 i;
401 :
402 9096 : memset(base, 0, siglen);
403 :
404 21648 : for (i = 0; i < entryvec->n; i++)
405 : {
406 15372 : if (unionkey(base, GETENTRY(entryvec, i), siglen))
407 : {
408 2820 : result->flag |= ALLISTRUE;
409 2820 : SET_VARSIZE(result, CALCGTSIZE(result->flag, siglen));
410 2820 : break;
411 : }
412 : }
413 :
414 9096 : *size = VARSIZE(result);
415 :
416 9096 : PG_RETURN_POINTER(result);
417 : }
418 :
419 : Datum
420 9096 : gtsvector_same(PG_FUNCTION_ARGS)
421 : {
422 9096 : SignTSVector *a = (SignTSVector *) PG_GETARG_POINTER(0);
423 9096 : SignTSVector *b = (SignTSVector *) PG_GETARG_POINTER(1);
424 9096 : bool *result = (bool *) PG_GETARG_POINTER(2);
425 9096 : int siglen = GET_SIGLEN();
426 :
427 9096 : if (ISSIGNKEY(a))
428 : { /* then b also ISSIGNKEY */
429 9096 : if (ISALLTRUE(a) && ISALLTRUE(b))
430 2820 : *result = true;
431 6276 : else if (ISALLTRUE(a))
432 0 : *result = false;
433 6276 : else if (ISALLTRUE(b))
434 0 : *result = false;
435 : else
436 : {
437 : int32 i;
438 6276 : BITVECP sa = GETSIGN(a),
439 6276 : sb = GETSIGN(b);
440 :
441 : Assert(GETSIGLEN(a) == siglen && GETSIGLEN(b) == siglen);
442 :
443 6276 : *result = true;
444 496798 : LOOPBYTE(siglen)
445 : {
446 496242 : if (sa[i] != sb[i])
447 : {
448 5720 : *result = false;
449 5720 : break;
450 : }
451 : }
452 : }
453 : }
454 : else
455 : { /* a and b ISARRKEY */
456 0 : int32 lena = ARRNELEM(a),
457 0 : lenb = ARRNELEM(b);
458 :
459 0 : if (lena != lenb)
460 0 : *result = false;
461 : else
462 : {
463 0 : int32 *ptra = GETARR(a),
464 0 : *ptrb = GETARR(b);
465 : int32 i;
466 :
467 0 : *result = true;
468 0 : for (i = 0; i < lena; i++)
469 0 : if (ptra[i] != ptrb[i])
470 : {
471 0 : *result = false;
472 0 : break;
473 : }
474 : }
475 : }
476 :
477 9096 : PG_RETURN_POINTER(result);
478 : }
479 :
480 : static int32
481 10566 : sizebitvec(BITVECP sign, int siglen)
482 : {
483 10566 : return pg_popcount(sign, siglen);
484 : }
485 :
486 : static int
487 279254 : hemdistsign(BITVECP a, BITVECP b, int siglen)
488 : {
489 : int i,
490 : diff,
491 279254 : dist = 0;
492 :
493 51641746 : LOOPBYTE(siglen)
494 : {
495 51362492 : diff = (unsigned char) (a[i] ^ b[i]);
496 : /* Using the popcount functions here isn't likely to win */
497 51362492 : dist += pg_number_of_ones[diff];
498 : }
499 279254 : return dist;
500 : }
501 :
502 : static int
503 0 : hemdist(SignTSVector *a, SignTSVector *b)
504 : {
505 0 : int siglena = GETSIGLEN(a);
506 0 : int siglenb = GETSIGLEN(b);
507 :
508 0 : if (ISALLTRUE(a))
509 : {
510 0 : if (ISALLTRUE(b))
511 0 : return 0;
512 : else
513 0 : return SIGLENBIT(siglenb) - sizebitvec(GETSIGN(b), siglenb);
514 : }
515 0 : else if (ISALLTRUE(b))
516 0 : return SIGLENBIT(siglena) - sizebitvec(GETSIGN(a), siglena);
517 :
518 : Assert(siglena == siglenb);
519 :
520 0 : return hemdistsign(GETSIGN(a), GETSIGN(b), siglena);
521 : }
522 :
523 : Datum
524 62374 : gtsvector_penalty(PG_FUNCTION_ARGS)
525 : {
526 62374 : GISTENTRY *origentry = (GISTENTRY *) PG_GETARG_POINTER(0); /* always ISSIGNKEY */
527 62374 : GISTENTRY *newentry = (GISTENTRY *) PG_GETARG_POINTER(1);
528 62374 : float *penalty = (float *) PG_GETARG_POINTER(2);
529 62374 : int siglen = GET_SIGLEN();
530 62374 : SignTSVector *origval = (SignTSVector *) DatumGetPointer(origentry->key);
531 62374 : SignTSVector *newval = (SignTSVector *) DatumGetPointer(newentry->key);
532 62374 : BITVECP orig = GETSIGN(origval);
533 :
534 62374 : *penalty = 0.0;
535 :
536 62374 : if (ISARRKEY(newval))
537 : {
538 62374 : BITVECP sign = palloc(siglen);
539 :
540 62374 : makesign(sign, newval, siglen);
541 :
542 62374 : if (ISALLTRUE(origval))
543 : {
544 10566 : int siglenbit = SIGLENBIT(siglen);
545 :
546 10566 : *penalty =
547 10566 : (float) (siglenbit - sizebitvec(sign, siglen)) /
548 10566 : (float) (siglenbit + 1);
549 : }
550 : else
551 51808 : *penalty = hemdistsign(sign, orig, siglen);
552 :
553 62374 : pfree(sign);
554 : }
555 : else
556 0 : *penalty = hemdist(origval, newval);
557 62374 : PG_RETURN_POINTER(penalty);
558 : }
559 :
560 : typedef struct
561 : {
562 : bool allistrue;
563 : BITVECP sign;
564 : } CACHESIGN;
565 :
566 : static void
567 12566 : fillcache(CACHESIGN *item, SignTSVector *key, int siglen)
568 : {
569 12566 : item->allistrue = false;
570 12566 : if (ISARRKEY(key))
571 12476 : makesign(item->sign, key, siglen);
572 90 : else if (ISALLTRUE(key))
573 0 : item->allistrue = true;
574 : else
575 90 : memcpy(item->sign, GETSIGN(key), siglen);
576 12566 : }
577 :
578 : #define WISH_F(a,b,c) (double)( -(double)(((a)-(b))*((a)-(b))*((a)-(b)))*(c) )
579 : typedef struct
580 : {
581 : OffsetNumber pos;
582 : int32 cost;
583 : } SPLITCOST;
584 :
585 : static int
586 31254 : comparecost(const void *va, const void *vb)
587 : {
588 31254 : const SPLITCOST *a = (const SPLITCOST *) va;
589 31254 : const SPLITCOST *b = (const SPLITCOST *) vb;
590 :
591 31254 : return pg_cmp_s32(a->cost, b->cost);
592 : }
593 :
594 :
595 : static int
596 203938 : hemdistcache(CACHESIGN *a, CACHESIGN *b, int siglen)
597 : {
598 203938 : if (a->allistrue)
599 : {
600 0 : if (b->allistrue)
601 0 : return 0;
602 : else
603 0 : return SIGLENBIT(siglen) - sizebitvec(b->sign, siglen);
604 : }
605 203938 : else if (b->allistrue)
606 0 : return SIGLENBIT(siglen) - sizebitvec(a->sign, siglen);
607 :
608 203938 : return hemdistsign(a->sign, b->sign, siglen);
609 : }
610 :
611 : Datum
612 406 : gtsvector_picksplit(PG_FUNCTION_ARGS)
613 : {
614 406 : GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
615 406 : GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
616 406 : int siglen = GET_SIGLEN();
617 : OffsetNumber k,
618 : j;
619 : SignTSVector *datum_l,
620 : *datum_r;
621 : BITVECP union_l,
622 : union_r;
623 : int32 size_alpha,
624 : size_beta;
625 : int32 size_waste,
626 406 : waste = -1;
627 : int32 nbytes;
628 406 : OffsetNumber seed_1 = 0,
629 406 : seed_2 = 0;
630 : OffsetNumber *left,
631 : *right;
632 : OffsetNumber maxoff;
633 : BITVECP ptr;
634 : int i;
635 : CACHESIGN *cache;
636 : char *cache_sign;
637 : SPLITCOST *costvector;
638 :
639 406 : maxoff = entryvec->n - 2;
640 406 : nbytes = (maxoff + 2) * sizeof(OffsetNumber);
641 406 : v->spl_left = (OffsetNumber *) palloc(nbytes);
642 406 : v->spl_right = (OffsetNumber *) palloc(nbytes);
643 :
644 406 : cache = (CACHESIGN *) palloc(sizeof(CACHESIGN) * (maxoff + 2));
645 406 : cache_sign = palloc(siglen * (maxoff + 2));
646 :
647 13378 : for (j = 0; j < maxoff + 2; j++)
648 12972 : cache[j].sign = &cache_sign[siglen * j];
649 :
650 406 : fillcache(&cache[FirstOffsetNumber], GETENTRY(entryvec, FirstOffsetNumber),
651 : siglen);
652 :
653 12160 : for (k = FirstOffsetNumber; k < maxoff; k = OffsetNumberNext(k))
654 : {
655 190560 : for (j = OffsetNumberNext(k); j <= maxoff; j = OffsetNumberNext(j))
656 : {
657 178806 : if (k == FirstOffsetNumber)
658 11754 : fillcache(&cache[j], GETENTRY(entryvec, j), siglen);
659 :
660 178806 : size_waste = hemdistcache(&(cache[j]), &(cache[k]), siglen);
661 178806 : if (size_waste > waste)
662 : {
663 2472 : waste = size_waste;
664 2472 : seed_1 = k;
665 2472 : seed_2 = j;
666 : }
667 : }
668 : }
669 :
670 406 : left = v->spl_left;
671 406 : v->spl_nleft = 0;
672 406 : right = v->spl_right;
673 406 : v->spl_nright = 0;
674 :
675 406 : if (seed_1 == 0 || seed_2 == 0)
676 : {
677 0 : seed_1 = 1;
678 0 : seed_2 = 2;
679 : }
680 :
681 : /* form initial .. */
682 406 : datum_l = gtsvector_alloc(SIGNKEY | (cache[seed_1].allistrue ? ALLISTRUE : 0),
683 406 : siglen, cache[seed_1].sign);
684 406 : datum_r = gtsvector_alloc(SIGNKEY | (cache[seed_2].allistrue ? ALLISTRUE : 0),
685 406 : siglen, cache[seed_2].sign);
686 406 : union_l = GETSIGN(datum_l);
687 406 : union_r = GETSIGN(datum_r);
688 406 : maxoff = OffsetNumberNext(maxoff);
689 406 : fillcache(&cache[maxoff], GETENTRY(entryvec, maxoff), siglen);
690 : /* sort before ... */
691 406 : costvector = (SPLITCOST *) palloc(sizeof(SPLITCOST) * maxoff);
692 12972 : for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
693 : {
694 12566 : costvector[j - 1].pos = j;
695 12566 : size_alpha = hemdistcache(&(cache[seed_1]), &(cache[j]), siglen);
696 12566 : size_beta = hemdistcache(&(cache[seed_2]), &(cache[j]), siglen);
697 12566 : costvector[j - 1].cost = abs(size_alpha - size_beta);
698 : }
699 406 : qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
700 :
701 12972 : for (k = 0; k < maxoff; k++)
702 : {
703 12566 : j = costvector[k].pos;
704 12566 : if (j == seed_1)
705 : {
706 406 : *left++ = j;
707 406 : v->spl_nleft++;
708 406 : continue;
709 : }
710 12160 : else if (j == seed_2)
711 : {
712 406 : *right++ = j;
713 406 : v->spl_nright++;
714 406 : continue;
715 : }
716 :
717 11754 : if (ISALLTRUE(datum_l) || cache[j].allistrue)
718 : {
719 0 : if (ISALLTRUE(datum_l) && cache[j].allistrue)
720 0 : size_alpha = 0;
721 : else
722 0 : size_alpha = SIGLENBIT(siglen) -
723 0 : sizebitvec((cache[j].allistrue) ?
724 : GETSIGN(datum_l) :
725 0 : cache[j].sign,
726 : siglen);
727 : }
728 : else
729 11754 : size_alpha = hemdistsign(cache[j].sign, GETSIGN(datum_l), siglen);
730 :
731 11754 : if (ISALLTRUE(datum_r) || cache[j].allistrue)
732 : {
733 0 : if (ISALLTRUE(datum_r) && cache[j].allistrue)
734 0 : size_beta = 0;
735 : else
736 0 : size_beta = SIGLENBIT(siglen) -
737 0 : sizebitvec((cache[j].allistrue) ?
738 : GETSIGN(datum_r) :
739 0 : cache[j].sign,
740 : siglen);
741 : }
742 : else
743 11754 : size_beta = hemdistsign(cache[j].sign, GETSIGN(datum_r), siglen);
744 :
745 11754 : if (size_alpha < size_beta + WISH_F(v->spl_nleft, v->spl_nright, 0.1))
746 : {
747 5792 : if (ISALLTRUE(datum_l) || cache[j].allistrue)
748 : {
749 0 : if (!ISALLTRUE(datum_l))
750 0 : memset(GETSIGN(datum_l), 0xff, siglen);
751 : }
752 : else
753 : {
754 5792 : ptr = cache[j].sign;
755 942754 : LOOPBYTE(siglen)
756 936962 : union_l[i] |= ptr[i];
757 : }
758 5792 : *left++ = j;
759 5792 : v->spl_nleft++;
760 : }
761 : else
762 : {
763 5962 : if (ISALLTRUE(datum_r) || cache[j].allistrue)
764 : {
765 0 : if (!ISALLTRUE(datum_r))
766 0 : memset(GETSIGN(datum_r), 0xff, siglen);
767 : }
768 : else
769 : {
770 5962 : ptr = cache[j].sign;
771 987968 : LOOPBYTE(siglen)
772 982006 : union_r[i] |= ptr[i];
773 : }
774 5962 : *right++ = j;
775 5962 : v->spl_nright++;
776 : }
777 : }
778 :
779 406 : *right = *left = FirstOffsetNumber;
780 406 : v->spl_ldatum = PointerGetDatum(datum_l);
781 406 : v->spl_rdatum = PointerGetDatum(datum_r);
782 :
783 406 : PG_RETURN_POINTER(v);
784 : }
785 :
786 : /*
787 : * Formerly, gtsvector_consistent was declared in pg_proc.h with arguments
788 : * that did not match the documented conventions for GiST support functions.
789 : * We fixed that, but we still need a pg_proc entry with the old signature
790 : * to support reloading pre-9.6 contrib/tsearch2 opclass declarations.
791 : * This compatibility function should go away eventually.
792 : */
793 : Datum
794 0 : gtsvector_consistent_oldsig(PG_FUNCTION_ARGS)
795 : {
796 0 : return gtsvector_consistent(fcinfo);
797 : }
798 :
799 : Datum
800 354 : gtsvector_options(PG_FUNCTION_ARGS)
801 : {
802 354 : local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
803 :
804 354 : init_local_reloptions(relopts, sizeof(GistTsVectorOptions));
805 354 : add_local_int_reloption(relopts, "siglen", "signature length",
806 : SIGLEN_DEFAULT, 1, SIGLEN_MAX,
807 : offsetof(GistTsVectorOptions, siglen));
808 :
809 354 : PG_RETURN_VOID();
810 : }
|