Age Owner Branch data TLA 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-2024, 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
3401 heikki.linnakangas@i 30 :CBC 183714 : pairingheap_GISTSearchItem_cmp(const pairingheap_node *a, const pairingheap_node *b, void *arg)
31 : : {
32 : 183714 : const GISTSearchItem *sa = (const GISTSearchItem *) a;
33 : 183714 : const GISTSearchItem *sb = (const GISTSearchItem *) b;
4881 tgl@sss.pgh.pa.us 34 : 183714 : IndexScanDesc scan = (IndexScanDesc) arg;
35 : : int i;
36 : :
37 : : /* Order according to distance comparison */
38 [ + + ]: 185660 : for (i = 0; i < scan->numberOfOrderBys; i++)
39 : : {
1669 akorotkov@postgresql 40 [ + + ]: 34291 : if (sa->distances[i].isnull)
41 : : {
42 [ + - ]: 54 : if (!sb->distances[i].isnull)
1680 43 : 54 : return -1;
44 : : }
1669 45 [ + + ]: 34237 : else if (sb->distances[i].isnull)
46 : : {
1680 47 : 7 : return 1;
48 : : }
49 : : else
50 : : {
1669 51 : 68460 : int cmp = -float8_cmp_internal(sa->distances[i].value,
52 : 34230 : sb->distances[i].value);
53 : :
1680 54 [ + + ]: 34230 : if (cmp != 0)
55 : 32284 : return cmp;
56 : : }
57 : : }
58 : :
59 : : /* Heap items go before inner pages, to ensure a depth-first search */
3401 heikki.linnakangas@i 60 [ + + - + ]: 151369 : if (GISTSearchItemIsHeap(*sa) && !GISTSearchItemIsHeap(*sb))
3401 heikki.linnakangas@i 61 :UBC 0 : return 1;
3344 heikki.linnakangas@i 62 [ + + + + ]:CBC 151369 : if (!GISTSearchItemIsHeap(*sa) && GISTSearchItemIsHeap(*sb))
63 : 2 : return -1;
64 : :
3401 65 : 151367 : return 0;
66 : : }
67 : :
68 : :
69 : : /*
70 : : * Index AM API functions for scanning GiST indexes
71 : : */
72 : :
73 : : IndexScanDesc
3010 tgl@sss.pgh.pa.us 74 : 2310 : gistbeginscan(Relation r, int nkeys, int norderbys)
75 : : {
76 : : IndexScanDesc scan;
77 : : GISTSTATE *giststate;
78 : : GISTScanOpaque so;
79 : : MemoryContext oldCxt;
80 : :
4882 81 : 2310 : scan = RelationGetIndexScan(r, nkeys, norderbys);
82 : :
83 : : /* First, set up a GISTSTATE with a scan-lifespan memory context */
4580 84 : 2310 : 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 : 2310 : oldCxt = MemoryContextSwitchTo(giststate->scanCxt);
91 : :
92 : : /* initialize opaque data */
4881 93 : 2310 : so = (GISTScanOpaque) palloc0(sizeof(GISTScanOpaqueData));
4580 94 : 2310 : so->giststate = giststate;
95 : 2310 : giststate->tempCxt = createTempGistContext();
96 : 2310 : so->queue = NULL;
4326 bruce@momjian.us 97 : 2310 : so->queueCxt = giststate->scanCxt; /* see gistrescan */
98 : :
99 : : /* workspaces with size dependent on numberOfOrderBys: */
1669 akorotkov@postgresql 100 : 2310 : so->distances = palloc(sizeof(so->distances[0]) * scan->numberOfOrderBys);
4881 tgl@sss.pgh.pa.us 101 : 2310 : so->qual_ok = true; /* in case there are zero keys */
3257 heikki.linnakangas@i 102 [ + + ]: 2310 : if (scan->numberOfOrderBys > 0)
103 : : {
3249 tgl@sss.pgh.pa.us 104 : 63 : scan->xs_orderbyvals = palloc0(sizeof(Datum) * scan->numberOfOrderBys);
3257 heikki.linnakangas@i 105 : 63 : scan->xs_orderbynulls = palloc(sizeof(bool) * scan->numberOfOrderBys);
3249 tgl@sss.pgh.pa.us 106 : 63 : memset(scan->xs_orderbynulls, true, sizeof(bool) * scan->numberOfOrderBys);
107 : : }
108 : :
3140 teodor@sigaev.ru 109 : 2310 : so->killedItems = NULL; /* until needed */
110 : 2310 : so->numKilled = 0;
111 : 2310 : so->curBlkno = InvalidBlockNumber;
112 : 2310 : so->curPageLSN = InvalidXLogRecPtr;
113 : :
4882 tgl@sss.pgh.pa.us 114 : 2310 : 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 : :
4580 121 : 2310 : MemoryContextSwitchTo(oldCxt);
122 : :
3010 123 : 2310 : return scan;
124 : : }
125 : :
126 : : void
127 : 2355 : gistrescan(IndexScanDesc scan, ScanKey key, int nkeys,
128 : : ScanKey orderbys, int norderbys)
129 : : {
130 : : /* nkeys and norderbys arguments are ignored */
4882 131 : 2355 : 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 : : */
4580 148 [ + + ]: 2355 : if (so->queue == NULL)
149 : : {
150 : : /* first time through */
151 [ - + ]: 2310 : Assert(so->queueCxt == so->giststate->scanCxt);
152 : 2310 : first_time = true;
153 : : }
154 [ + + ]: 45 : else if (so->queueCxt == so->giststate->scanCxt)
155 : : {
156 : : /* second time through */
157 : 15 : so->queueCxt = AllocSetContextCreate(so->giststate->scanCxt,
158 : : "GiST queue context",
159 : : ALLOCSET_DEFAULT_SIZES);
160 : 15 : first_time = false;
161 : : }
162 : : else
163 : : {
164 : : /* third or later time through */
165 : 30 : MemoryContextReset(so->queueCxt);
166 : 30 : 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 : : */
2603 174 [ + + + + ]: 2355 : 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 : : */
3249 bruce@momjian.us 186 : 334 : natts = RelationGetNumberOfAttributes(scan->indexRelation);
1862 akorotkov@postgresql 187 : 334 : nkeyatts = IndexRelationGetNumberOfKeyAttributes(scan->indexRelation);
1972 andres@anarazel.de 188 : 334 : so->giststate->fetchTupdesc = CreateTemplateTupleDesc(natts);
1862 akorotkov@postgresql 189 [ + + ]: 681 : for (attno = 1; attno <= nkeyatts; attno++)
190 : : {
3307 heikki.linnakangas@i 191 : 347 : TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
192 : 347 : scan->indexRelation->rd_opcintype[attno - 1],
193 : : -1, 0);
194 : : }
195 : :
1862 akorotkov@postgresql 196 [ - + ]: 334 : for (; attno <= natts; attno++)
197 : : {
198 : : /* taking opcintype from giststate->tupdesc */
1862 akorotkov@postgresql 199 :UBC 0 : TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
200 : 0 : TupleDescAttr(so->giststate->leafTupdesc,
201 : : attno - 1)->atttypid,
202 : : -1, 0);
203 : : }
2603 tgl@sss.pgh.pa.us 204 :CBC 334 : scan->xs_hitupdesc = so->giststate->fetchTupdesc;
205 : :
206 : : /* Also create a memory context that will hold the returned tuples */
3307 heikki.linnakangas@i 207 : 334 : so->pageDataCxt = AllocSetContextCreate(so->giststate->scanCxt,
208 : : "GiST page data context",
209 : : ALLOCSET_DEFAULT_SIZES);
210 : : }
211 : :
212 : : /* create new, empty pairing heap for search queue */
4881 tgl@sss.pgh.pa.us 213 : 2355 : oldCxt = MemoryContextSwitchTo(so->queueCxt);
3401 heikki.linnakangas@i 214 : 2355 : so->queue = pairingheap_allocate(pairingheap_GISTSearchItem_cmp, scan);
4881 tgl@sss.pgh.pa.us 215 : 2355 : MemoryContextSwitchTo(oldCxt);
216 : :
217 : 2355 : so->firstCall = true;
218 : :
219 : : /* Update scan key, if a new one is given */
6907 neilc@samurai.com 220 [ + - + + ]: 2355 : if (key && scan->numberOfKeys > 0)
221 : : {
3993 tgl@sss.pgh.pa.us 222 : 2280 : void **fn_extras = NULL;
223 : :
224 : : /*
225 : : * If this isn't the first time through, preserve the fn_extra
226 : : * pointers, so that if the consistentFns are using them to cache
227 : : * data, that data is not leaked across a rescan.
228 : : */
4580 229 [ + + ]: 2280 : if (!first_time)
230 : : {
3993 231 : 15 : fn_extras = (void **) palloc(scan->numberOfKeys * sizeof(void *));
4580 232 [ + + ]: 30 : for (i = 0; i < scan->numberOfKeys; i++)
3993 233 : 15 : fn_extras[i] = scan->keyData[i].sk_func.fn_extra;
234 : : }
235 : :
6907 neilc@samurai.com 236 : 2280 : memmove(scan->keyData, key,
237 : 2280 : 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 : : */
4881 tgl@sss.pgh.pa.us 250 : 2280 : so->qual_ok = true;
251 : :
5421 bruce@momjian.us 252 [ + + ]: 5401 : for (i = 0; i < scan->numberOfKeys; i++)
253 : : {
4881 tgl@sss.pgh.pa.us 254 : 3121 : ScanKey skey = scan->keyData + i;
255 : :
256 : : /*
257 : : * Copy consistent support function to ScanKey structure instead
258 : : * of function implementing filtering operator.
259 : : */
3993 260 : 3121 : fmgr_info_copy(&(skey->sk_func),
261 : 3121 : &(so->giststate->consistentFn[skey->sk_attno - 1]),
262 : 3121 : so->giststate->scanCxt);
263 : :
264 : : /* Restore prior fn_extra pointers, if not first time */
265 [ + + ]: 3121 : if (!first_time)
266 : 15 : skey->sk_func.fn_extra = fn_extras[i];
267 : :
5217 268 [ + + ]: 3121 : if (skey->sk_flags & SK_ISNULL)
269 : : {
270 [ - + ]: 13 : if (!(skey->sk_flags & (SK_SEARCHNULL | SK_SEARCHNOTNULL)))
5658 teodor@sigaev.ru 271 :UBC 0 : so->qual_ok = false;
272 : : }
273 : : }
274 : :
3993 tgl@sss.pgh.pa.us 275 [ + + ]:CBC 2280 : if (!first_time)
276 : 15 : pfree(fn_extras);
277 : : }
278 : :
279 : : /* Update order-by key, if a new one is given */
4881 280 [ + + + + ]: 2355 : if (orderbys && scan->numberOfOrderBys > 0)
281 : : {
3993 282 : 99 : void **fn_extras = NULL;
283 : :
284 : : /* As above, preserve fn_extra if not first time through */
4580 285 [ + + ]: 99 : if (!first_time)
286 : : {
3993 287 : 36 : fn_extras = (void **) palloc(scan->numberOfOrderBys * sizeof(void *));
4580 288 [ + + ]: 72 : for (i = 0; i < scan->numberOfOrderBys; i++)
3993 289 : 36 : fn_extras[i] = scan->orderByData[i].sk_func.fn_extra;
290 : : }
291 : :
4881 292 : 99 : memmove(scan->orderByData, orderbys,
293 : 99 : scan->numberOfOrderBys * sizeof(ScanKeyData));
294 : :
3257 heikki.linnakangas@i 295 : 99 : so->orderByTypes = (Oid *) palloc(scan->numberOfOrderBys * sizeof(Oid));
296 : :
297 : : /*
298 : : * Modify the order-by key so that the Distance method is called for
299 : : * all comparisons. The original operator is passed to the Distance
300 : : * function in the form of its strategy number, which is available
301 : : * from the sk_strategy field, and its subtype from the sk_subtype
302 : : * field.
303 : : */
4881 tgl@sss.pgh.pa.us 304 [ + + ]: 198 : for (i = 0; i < scan->numberOfOrderBys; i++)
305 : : {
306 : 99 : ScanKey skey = scan->orderByData + i;
3993 307 : 99 : FmgrInfo *finfo = &(so->giststate->distanceFn[skey->sk_attno - 1]);
308 : :
309 : : /* Check we actually have a distance function ... */
310 [ - + ]: 99 : if (!OidIsValid(finfo->fn_oid))
4881 tgl@sss.pgh.pa.us 311 [ # # ]:UBC 0 : elog(ERROR, "missing support function %d for attribute %d of index \"%s\"",
312 : : GIST_DISTANCE_PROC, skey->sk_attno,
313 : : RelationGetRelationName(scan->indexRelation));
314 : :
315 : : /*
316 : : * Look up the datatype returned by the original ordering
317 : : * operator. GiST always uses a float8 for the distance function,
318 : : * but the ordering operator could be anything else.
319 : : *
320 : : * XXX: The distance function is only allowed to be lossy if the
321 : : * ordering operator's result type is float4 or float8. Otherwise
322 : : * we don't know how to return the distance to the executor. But
323 : : * we cannot check that here, as we won't know if the distance
324 : : * function is lossy until it returns *recheck = true for the
325 : : * first time.
326 : : */
3257 heikki.linnakangas@i 327 :CBC 99 : so->orderByTypes[i] = get_func_rettype(skey->sk_func.fn_oid);
328 : :
329 : : /*
330 : : * Copy distance support function to ScanKey structure instead of
331 : : * function implementing ordering operator.
332 : : */
2994 teodor@sigaev.ru 333 : 99 : fmgr_info_copy(&(skey->sk_func), finfo, so->giststate->scanCxt);
334 : :
335 : : /* Restore prior fn_extra pointers, if not first time */
3993 tgl@sss.pgh.pa.us 336 [ + + ]: 99 : if (!first_time)
337 : 36 : skey->sk_func.fn_extra = fn_extras[i];
338 : : }
339 : :
340 [ + + ]: 99 : if (!first_time)
341 : 36 : pfree(fn_extras);
342 : : }
343 : :
344 : : /* any previous xs_hitup will have been pfree'd in context resets above */
2537 345 : 2355 : scan->xs_hitup = NULL;
10093 scrappy@hub.org 346 : 2355 : }
347 : :
348 : : void
3010 tgl@sss.pgh.pa.us 349 : 2208 : gistendscan(IndexScanDesc scan)
350 : : {
4882 351 : 2208 : GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
352 : :
353 : : /*
354 : : * freeGISTstate is enough to clean up everything made by gistbeginscan,
355 : : * as well as the queueCxt if there is a separate context for it.
356 : : */
4881 357 : 2208 : freeGISTstate(so->giststate);
10093 scrappy@hub.org 358 : 2208 : }
|