Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * gistscan.c
4 : * routines to manage scans on GiST index relations
5 : *
6 : *
7 : * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
8 : * Portions Copyright (c) 1994, Regents of the University of California
9 : *
10 : * IDENTIFICATION
11 : * src/backend/access/gist/gistscan.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 : #include "postgres.h"
16 :
17 : #include "access/gist_private.h"
18 : #include "access/gistscan.h"
19 : #include "access/relscan.h"
20 : #include "utils/float.h"
21 : #include "utils/lsyscache.h"
22 : #include "utils/memutils.h"
23 : #include "utils/rel.h"
24 :
25 :
26 : /*
27 : * Pairing heap comparison function for the GISTSearchItem queue
28 : */
29 : static int
30 197654 : pairingheap_GISTSearchItem_cmp(const pairingheap_node *a, const pairingheap_node *b, void *arg)
31 : {
32 197654 : const GISTSearchItem *sa = (const GISTSearchItem *) a;
33 197654 : const GISTSearchItem *sb = (const GISTSearchItem *) b;
34 197654 : IndexScanDesc scan = (IndexScanDesc) arg;
35 : int i;
36 :
37 : /* Order according to distance comparison */
38 199789 : for (i = 0; i < scan->numberOfOrderBys; i++)
39 : {
40 37991 : if (sa->distances[i].isnull)
41 : {
42 68 : if (!sb->distances[i].isnull)
43 68 : return -1;
44 : }
45 37923 : else if (sb->distances[i].isnull)
46 : {
47 16 : return 1;
48 : }
49 : else
50 : {
51 75814 : int cmp = -float8_cmp_internal(sa->distances[i].value,
52 37907 : sb->distances[i].value);
53 :
54 37907 : if (cmp != 0)
55 35772 : return cmp;
56 : }
57 : }
58 :
59 : /* Heap items go before inner pages, to ensure a depth-first search */
60 161798 : if (GISTSearchItemIsHeap(*sa) && !GISTSearchItemIsHeap(*sb))
61 0 : return 1;
62 161798 : if (!GISTSearchItemIsHeap(*sa) && GISTSearchItemIsHeap(*sb))
63 2 : return -1;
64 :
65 161796 : return 0;
66 : }
67 :
68 :
69 : /*
70 : * Index AM API functions for scanning GiST indexes
71 : */
72 :
73 : IndexScanDesc
74 3748 : gistbeginscan(Relation r, int nkeys, int norderbys)
75 : {
76 : IndexScanDesc scan;
77 : GISTSTATE *giststate;
78 : GISTScanOpaque so;
79 : MemoryContext oldCxt;
80 :
81 3748 : scan = RelationGetIndexScan(r, nkeys, norderbys);
82 :
83 : /* First, set up a GISTSTATE with a scan-lifespan memory context */
84 3748 : giststate = initGISTstate(scan->indexRelation);
85 :
86 : /*
87 : * Everything made below is in the scanCxt, or is a child of the scanCxt,
88 : * so it'll all go away automatically in gistendscan.
89 : */
90 3748 : oldCxt = MemoryContextSwitchTo(giststate->scanCxt);
91 :
92 : /* initialize opaque data */
93 3748 : so = palloc0_object(GISTScanOpaqueData);
94 3748 : so->giststate = giststate;
95 3748 : giststate->tempCxt = createTempGistContext();
96 3748 : so->queue = NULL;
97 3748 : so->queueCxt = giststate->scanCxt; /* see gistrescan */
98 :
99 : /* workspaces with size dependent on numberOfOrderBys: */
100 3748 : so->distances = palloc(sizeof(so->distances[0]) * scan->numberOfOrderBys);
101 3748 : so->qual_ok = true; /* in case there are zero keys */
102 3748 : if (scan->numberOfOrderBys > 0)
103 : {
104 75 : scan->xs_orderbyvals = palloc0_array(Datum, scan->numberOfOrderBys);
105 75 : scan->xs_orderbynulls = palloc_array(bool, scan->numberOfOrderBys);
106 75 : memset(scan->xs_orderbynulls, true, sizeof(bool) * scan->numberOfOrderBys);
107 : }
108 :
109 3748 : so->killedItems = NULL; /* until needed */
110 3748 : so->numKilled = 0;
111 3748 : so->curBlkno = InvalidBlockNumber;
112 3748 : so->curPageLSN = InvalidXLogRecPtr;
113 :
114 3748 : scan->opaque = so;
115 :
116 : /*
117 : * All fields required for index-only scans are initialized in gistrescan,
118 : * as we don't know yet if we're doing an index-only scan or not.
119 : */
120 :
121 3748 : MemoryContextSwitchTo(oldCxt);
122 :
123 3748 : return scan;
124 : }
125 :
126 : void
127 3808 : gistrescan(IndexScanDesc scan, ScanKey key, int nkeys,
128 : ScanKey orderbys, int norderbys)
129 : {
130 : /* nkeys and norderbys arguments are ignored */
131 3808 : GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
132 : bool first_time;
133 : int i;
134 : MemoryContext oldCxt;
135 :
136 : /* rescan an existing indexscan --- reset state */
137 :
138 : /*
139 : * The first time through, we create the search queue in the scanCxt.
140 : * Subsequent times through, we create the queue in a separate queueCxt,
141 : * which is created on the second call and reset on later calls. Thus, in
142 : * the common case where a scan is only rescan'd once, we just put the
143 : * queue in scanCxt and don't pay the overhead of making a second memory
144 : * context. If we do rescan more than once, the first queue is just left
145 : * for dead until end of scan; this small wastage seems worth the savings
146 : * in the common case.
147 : */
148 3808 : if (so->queue == NULL)
149 : {
150 : /* first time through */
151 : Assert(so->queueCxt == so->giststate->scanCxt);
152 3748 : first_time = true;
153 : }
154 60 : else if (so->queueCxt == so->giststate->scanCxt)
155 : {
156 : /* second time through */
157 20 : so->queueCxt = AllocSetContextCreate(so->giststate->scanCxt,
158 : "GiST queue context",
159 : ALLOCSET_DEFAULT_SIZES);
160 20 : first_time = false;
161 : }
162 : else
163 : {
164 : /* third or later time through */
165 40 : MemoryContextReset(so->queueCxt);
166 40 : first_time = false;
167 : }
168 :
169 : /*
170 : * If we're doing an index-only scan, on the first call, also initialize a
171 : * tuple descriptor to represent the returned index tuples and create a
172 : * memory context to hold them during the scan.
173 : */
174 3808 : if (scan->xs_want_itup && !scan->xs_hitupdesc)
175 : {
176 : int natts;
177 : int nkeyatts;
178 : int attno;
179 :
180 : /*
181 : * The storage type of the index can be different from the original
182 : * datatype being indexed, so we cannot just grab the index's tuple
183 : * descriptor. Instead, construct a descriptor with the original data
184 : * types.
185 : */
186 406 : natts = RelationGetNumberOfAttributes(scan->indexRelation);
187 406 : nkeyatts = IndexRelationGetNumberOfKeyAttributes(scan->indexRelation);
188 406 : so->giststate->fetchTupdesc = CreateTemplateTupleDesc(natts);
189 829 : for (attno = 1; attno <= nkeyatts; attno++)
190 : {
191 423 : TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
192 423 : scan->indexRelation->rd_opcintype[attno - 1],
193 : -1, 0);
194 : }
195 :
196 406 : for (; attno <= natts; attno++)
197 : {
198 : /* taking opcintype from giststate->tupdesc */
199 0 : TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
200 0 : TupleDescAttr(so->giststate->leafTupdesc,
201 : attno - 1)->atttypid,
202 : -1, 0);
203 : }
204 406 : TupleDescFinalize(so->giststate->fetchTupdesc);
205 406 : scan->xs_hitupdesc = so->giststate->fetchTupdesc;
206 :
207 : /* Also create a memory context that will hold the returned tuples */
208 406 : so->pageDataCxt = AllocSetContextCreate(so->giststate->scanCxt,
209 : "GiST page data context",
210 : ALLOCSET_DEFAULT_SIZES);
211 : }
212 :
213 : /* create new, empty pairing heap for search queue */
214 3808 : oldCxt = MemoryContextSwitchTo(so->queueCxt);
215 3808 : so->queue = pairingheap_allocate(pairingheap_GISTSearchItem_cmp, scan);
216 3808 : MemoryContextSwitchTo(oldCxt);
217 :
218 3808 : so->firstCall = true;
219 :
220 : /* Update scan key, if a new one is given */
221 3808 : if (key && scan->numberOfKeys > 0)
222 : {
223 3717 : void **fn_extras = NULL;
224 :
225 : /*
226 : * If this isn't the first time through, preserve the fn_extra
227 : * pointers, so that if the consistentFns are using them to cache
228 : * data, that data is not leaked across a rescan.
229 : */
230 3717 : if (!first_time)
231 : {
232 20 : fn_extras = (void **) palloc(scan->numberOfKeys * sizeof(void *));
233 40 : for (i = 0; i < scan->numberOfKeys; i++)
234 20 : fn_extras[i] = scan->keyData[i].sk_func.fn_extra;
235 : }
236 :
237 3717 : memcpy(scan->keyData, key, scan->numberOfKeys * sizeof(ScanKeyData));
238 :
239 : /*
240 : * Modify the scan key so that the Consistent method is called for all
241 : * comparisons. The original operator is passed to the Consistent
242 : * function in the form of its strategy number, which is available
243 : * from the sk_strategy field, and its subtype from the sk_subtype
244 : * field.
245 : *
246 : * Next, if any of keys is a NULL and that key is not marked with
247 : * SK_SEARCHNULL/SK_SEARCHNOTNULL then nothing can be found (ie, we
248 : * assume all indexable operators are strict).
249 : */
250 3717 : so->qual_ok = true;
251 :
252 9340 : for (i = 0; i < scan->numberOfKeys; i++)
253 : {
254 5623 : ScanKey skey = scan->keyData + i;
255 :
256 : /*
257 : * Copy consistent support function to ScanKey structure instead
258 : * of function implementing filtering operator.
259 : */
260 5623 : fmgr_info_copy(&(skey->sk_func),
261 5623 : &(so->giststate->consistentFn[skey->sk_attno - 1]),
262 5623 : so->giststate->scanCxt);
263 :
264 : /* Restore prior fn_extra pointers, if not first time */
265 5623 : if (!first_time)
266 20 : skey->sk_func.fn_extra = fn_extras[i];
267 :
268 5623 : if (skey->sk_flags & SK_ISNULL)
269 : {
270 17 : if (!(skey->sk_flags & (SK_SEARCHNULL | SK_SEARCHNOTNULL)))
271 0 : so->qual_ok = false;
272 : }
273 : }
274 :
275 3717 : if (!first_time)
276 20 : pfree(fn_extras);
277 : }
278 :
279 : /* Update order-by key, if a new one is given */
280 3808 : if (orderbys && scan->numberOfOrderBys > 0)
281 : {
282 123 : void **fn_extras = NULL;
283 :
284 : /* As above, preserve fn_extra if not first time through */
285 123 : if (!first_time)
286 : {
287 48 : fn_extras = (void **) palloc(scan->numberOfOrderBys * sizeof(void *));
288 96 : for (i = 0; i < scan->numberOfOrderBys; i++)
289 48 : fn_extras[i] = scan->orderByData[i].sk_func.fn_extra;
290 : }
291 :
292 123 : memcpy(scan->orderByData, orderbys, scan->numberOfOrderBys * sizeof(ScanKeyData));
293 :
294 123 : so->orderByTypes = (Oid *) palloc(scan->numberOfOrderBys * sizeof(Oid));
295 :
296 : /*
297 : * Modify the order-by key so that the Distance method is called for
298 : * all comparisons. The original operator is passed to the Distance
299 : * function in the form of its strategy number, which is available
300 : * from the sk_strategy field, and its subtype from the sk_subtype
301 : * field.
302 : */
303 246 : for (i = 0; i < scan->numberOfOrderBys; i++)
304 : {
305 123 : ScanKey skey = scan->orderByData + i;
306 123 : FmgrInfo *finfo = &(so->giststate->distanceFn[skey->sk_attno - 1]);
307 :
308 : /* Check we actually have a distance function ... */
309 123 : if (!OidIsValid(finfo->fn_oid))
310 0 : elog(ERROR, "missing support function %d for attribute %d of index \"%s\"",
311 : GIST_DISTANCE_PROC, skey->sk_attno,
312 : RelationGetRelationName(scan->indexRelation));
313 :
314 : /*
315 : * Look up the datatype returned by the original ordering
316 : * operator. GiST always uses a float8 for the distance function,
317 : * but the ordering operator could be anything else.
318 : *
319 : * XXX: The distance function is only allowed to be lossy if the
320 : * ordering operator's result type is float4 or float8. Otherwise
321 : * we don't know how to return the distance to the executor. But
322 : * we cannot check that here, as we won't know if the distance
323 : * function is lossy until it returns *recheck = true for the
324 : * first time.
325 : */
326 123 : so->orderByTypes[i] = get_func_rettype(skey->sk_func.fn_oid);
327 :
328 : /*
329 : * Copy distance support function to ScanKey structure instead of
330 : * function implementing ordering operator.
331 : */
332 123 : fmgr_info_copy(&(skey->sk_func), finfo, so->giststate->scanCxt);
333 :
334 : /* Restore prior fn_extra pointers, if not first time */
335 123 : if (!first_time)
336 48 : skey->sk_func.fn_extra = fn_extras[i];
337 : }
338 :
339 123 : if (!first_time)
340 48 : pfree(fn_extras);
341 : }
342 :
343 : /* any previous xs_hitup will have been pfree'd in context resets above */
344 3808 : scan->xs_hitup = NULL;
345 3808 : }
346 :
347 : void
348 3561 : gistendscan(IndexScanDesc scan)
349 : {
350 3561 : GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
351 :
352 : /*
353 : * freeGISTstate is enough to clean up everything made by gistbeginscan,
354 : * as well as the queueCxt if there is a separate context for it.
355 : */
356 3561 : freeGISTstate(so->giststate);
357 3561 : }
|