Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * nodeIndexscan.c
4 : : * Routines to support indexed scans of relations
5 : : *
6 : : * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
7 : : * Portions Copyright (c) 1994, Regents of the University of California
8 : : *
9 : : *
10 : : * IDENTIFICATION
11 : : * src/backend/executor/nodeIndexscan.c
12 : : *
13 : : *-------------------------------------------------------------------------
14 : : */
15 : : /*
16 : : * INTERFACE ROUTINES
17 : : * ExecIndexScan scans a relation using an index
18 : : * IndexNext retrieve next tuple using index
19 : : * IndexNextWithReorder same, but recheck ORDER BY expressions
20 : : * ExecInitIndexScan creates and initializes state info.
21 : : * ExecReScanIndexScan rescans the indexed relation.
22 : : * ExecEndIndexScan releases all storage.
23 : : * ExecIndexMarkPos marks scan position.
24 : : * ExecIndexRestrPos restores scan position.
25 : : * ExecIndexScanEstimate estimates DSM space needed for parallel index scan
26 : : * ExecIndexScanInitializeDSM initialize DSM for parallel indexscan
27 : : * ExecIndexScanReInitializeDSM reinitialize DSM for fresh scan
28 : : * ExecIndexScanInitializeWorker attach to DSM info in parallel worker
29 : : */
30 : : #include "postgres.h"
31 : :
32 : : #include "access/nbtree.h"
33 : : #include "access/relscan.h"
34 : : #include "access/tableam.h"
35 : : #include "catalog/pg_am.h"
36 : : #include "executor/executor.h"
37 : : #include "executor/nodeIndexscan.h"
38 : : #include "lib/pairingheap.h"
39 : : #include "miscadmin.h"
40 : : #include "nodes/nodeFuncs.h"
41 : : #include "utils/array.h"
42 : : #include "utils/datum.h"
43 : : #include "utils/lsyscache.h"
44 : : #include "utils/rel.h"
45 : :
46 : : /*
47 : : * When an ordering operator is used, tuples fetched from the index that
48 : : * need to be reordered are queued in a pairing heap, as ReorderTuples.
49 : : */
50 : : typedef struct
51 : : {
52 : : pairingheap_node ph_node;
53 : : HeapTuple htup;
54 : : Datum *orderbyvals;
55 : : bool *orderbynulls;
56 : : } ReorderTuple;
57 : :
58 : : static TupleTableSlot *IndexNext(IndexScanState *node);
59 : : static TupleTableSlot *IndexNextWithReorder(IndexScanState *node);
60 : : static void EvalOrderByExpressions(IndexScanState *node, ExprContext *econtext);
61 : : static bool IndexRecheck(IndexScanState *node, TupleTableSlot *slot);
62 : : static int cmp_orderbyvals(const Datum *adist, const bool *anulls,
63 : : const Datum *bdist, const bool *bnulls,
64 : : IndexScanState *node);
65 : : static int reorderqueue_cmp(const pairingheap_node *a,
66 : : const pairingheap_node *b, void *arg);
67 : : static void reorderqueue_push(IndexScanState *node, TupleTableSlot *slot,
68 : : Datum *orderbyvals, bool *orderbynulls);
69 : : static HeapTuple reorderqueue_pop(IndexScanState *node);
70 : :
71 : :
72 : : /* ----------------------------------------------------------------
73 : : * IndexNext
74 : : *
75 : : * Retrieve a tuple from the IndexScan node's currentRelation
76 : : * using the index specified in the IndexScanState information.
77 : : * ----------------------------------------------------------------
78 : : */
79 : : static TupleTableSlot *
7801 tgl@sss.pgh.pa.us 80 :CBC 955672 : IndexNext(IndexScanState *node)
81 : : {
82 : : EState *estate;
83 : : ExprContext *econtext;
84 : : ScanDirection direction;
85 : : IndexScanDesc scandesc;
86 : : TupleTableSlot *slot;
87 : :
88 : : /*
89 : : * extract necessary information from index scan node
90 : : */
91 : 955672 : estate = node->ss.ps.state;
92 : :
93 : : /*
94 : : * Determine which direction to scan the index in based on the plan's scan
95 : : * direction and the current direction of execution.
96 : : */
438 drowley@postgresql.o 97 : 955672 : direction = ScanDirectionCombine(estate->es_direction,
98 : : ((IndexScan *) node->ss.ps.plan)->indexorderdir);
6929 tgl@sss.pgh.pa.us 99 : 955672 : scandesc = node->iss_ScanDesc;
7801 100 : 955672 : econtext = node->ss.ps.ps_ExprContext;
101 : 955672 : slot = node->ss.ss_ScanTupleSlot;
102 : :
2594 rhaas@postgresql.org 103 [ + + ]: 955672 : if (scandesc == NULL)
104 : : {
105 : : /*
106 : : * We reach here if the index scan is not parallel, or if we're
107 : : * serially executing an index scan that was planned to be parallel.
108 : : */
109 : 60510 : scandesc = index_beginscan(node->ss.ss_currentRelation,
110 : : node->iss_RelationDesc,
111 : : estate->es_snapshot,
112 : : node->iss_NumScanKeys,
113 : : node->iss_NumOrderByKeys);
114 : :
115 : 60510 : node->iss_ScanDesc = scandesc;
116 : :
117 : : /*
118 : : * If no run-time keys to calculate or they are ready, go ahead and
119 : : * pass the scankeys to the index AM.
120 : : */
121 [ + + + - ]: 60510 : if (node->iss_NumRuntimeKeys == 0 || node->iss_RuntimeKeysReady)
122 : 60510 : index_rescan(scandesc,
123 : 60510 : node->iss_ScanKeys, node->iss_NumScanKeys,
124 : 60510 : node->iss_OrderByKeys, node->iss_NumOrderByKeys);
125 : : }
126 : :
127 : : /*
128 : : * ok, now that we have what we need, fetch the next tuple.
129 : : */
1861 andres@anarazel.de 130 [ + + ]: 957063 : while (index_getnext_slot(scandesc, direction, slot))
131 : : {
2455 132 [ + + ]: 755529 : CHECK_FOR_INTERRUPTS();
133 : :
134 : : /*
135 : : * If the index was lossy, we have to recheck the index quals using
136 : : * the fetched tuple.
137 : : */
5845 tgl@sss.pgh.pa.us 138 [ + + ]: 755529 : if (scandesc->xs_recheck)
139 : : {
140 : 162920 : econtext->ecxt_scantuple = slot;
2267 andres@anarazel.de 141 [ + + ]: 162920 : if (!ExecQualAndReset(node->indexqualorig, econtext))
142 : : {
143 : : /* Fails recheck, so drop it and loop back for another */
4588 tgl@sss.pgh.pa.us 144 [ - + ]: 1391 : InstrCountFiltered2(node, 1);
145 : 1391 : continue;
146 : : }
147 : : }
148 : :
6929 149 : 754138 : return slot;
150 : : }
151 : :
152 : : /*
153 : : * if we get here it means the index scan failed so we are at the end of
154 : : * the scan..
155 : : */
3257 heikki.linnakangas@i 156 : 201532 : node->iss_ReachedEnd = true;
157 : 201532 : return ExecClearTuple(slot);
158 : : }
159 : :
160 : : /* ----------------------------------------------------------------
161 : : * IndexNextWithReorder
162 : : *
163 : : * Like IndexNext, but this version can also re-check ORDER BY
164 : : * expressions, and reorder the tuples as necessary.
165 : : * ----------------------------------------------------------------
166 : : */
167 : : static TupleTableSlot *
168 : 41331 : IndexNextWithReorder(IndexScanState *node)
169 : : {
170 : : EState *estate;
171 : : ExprContext *econtext;
172 : : IndexScanDesc scandesc;
173 : : TupleTableSlot *slot;
174 : 41331 : ReorderTuple *topmost = NULL;
175 : : bool was_exact;
176 : : Datum *lastfetched_vals;
177 : : bool *lastfetched_nulls;
178 : : int cmp;
179 : :
2594 rhaas@postgresql.org 180 : 41331 : estate = node->ss.ps.state;
181 : :
182 : : /*
183 : : * Only forward scan is supported with reordering. Note: we can get away
184 : : * with just Asserting here because the system will not try to run the
185 : : * plan backwards if ExecSupportsBackwardScan() says it won't work.
186 : : * Currently, that is guaranteed because no index AMs support both
187 : : * amcanorderbyop and amcanbackward; if any ever do,
188 : : * ExecSupportsBackwardScan() will need to consider indexorderbys
189 : : * explicitly.
190 : : */
3257 heikki.linnakangas@i 191 [ - + ]: 41331 : Assert(!ScanDirectionIsBackward(((IndexScan *) node->ss.ps.plan)->indexorderdir));
2594 rhaas@postgresql.org 192 [ - + ]: 41331 : Assert(ScanDirectionIsForward(estate->es_direction));
193 : :
3257 heikki.linnakangas@i 194 : 41331 : scandesc = node->iss_ScanDesc;
195 : 41331 : econtext = node->ss.ps.ps_ExprContext;
1822 andres@anarazel.de 196 : 41331 : slot = node->ss.ss_ScanTupleSlot;
197 : :
2594 rhaas@postgresql.org 198 [ + + ]: 41331 : if (scandesc == NULL)
199 : : {
200 : : /*
201 : : * We reach here if the index scan is not parallel, or if we're
202 : : * serially executing an index scan that was planned to be parallel.
203 : : */
204 : 23 : scandesc = index_beginscan(node->ss.ss_currentRelation,
205 : : node->iss_RelationDesc,
206 : : estate->es_snapshot,
207 : : node->iss_NumScanKeys,
208 : : node->iss_NumOrderByKeys);
209 : :
210 : 23 : node->iss_ScanDesc = scandesc;
211 : :
212 : : /*
213 : : * If no run-time keys to calculate or they are ready, go ahead and
214 : : * pass the scankeys to the index AM.
215 : : */
216 [ + + + - ]: 23 : if (node->iss_NumRuntimeKeys == 0 || node->iss_RuntimeKeysReady)
217 : 23 : index_rescan(scandesc,
218 : 23 : node->iss_ScanKeys, node->iss_NumScanKeys,
219 : 23 : node->iss_OrderByKeys, node->iss_NumOrderByKeys);
220 : : }
221 : :
222 : : for (;;)
223 : : {
2455 andres@anarazel.de 224 [ - + ]: 43928 : CHECK_FOR_INTERRUPTS();
225 : :
226 : : /*
227 : : * Check the reorder queue first. If the topmost tuple in the queue
228 : : * has an ORDER BY value smaller than (or equal to) the value last
229 : : * returned by the index, we can return it now.
230 : : */
3257 heikki.linnakangas@i 231 [ + + ]: 43928 : if (!pairingheap_is_empty(node->iss_ReorderQueue))
232 : : {
233 : 5126 : topmost = (ReorderTuple *) pairingheap_first(node->iss_ReorderQueue);
234 : :
235 [ + + + + ]: 10249 : if (node->iss_ReachedEnd ||
236 : 5123 : cmp_orderbyvals(topmost->orderbyvals,
237 : 5123 : topmost->orderbynulls,
238 : 5123 : scandesc->xs_orderbyvals,
239 : 5123 : scandesc->xs_orderbynulls,
240 : : node) <= 0)
241 : : {
242 : : HeapTuple tuple;
243 : :
244 : 2538 : tuple = reorderqueue_pop(node);
245 : :
246 : : /* Pass 'true', as the tuple in the queue is a palloc'd copy */
1822 andres@anarazel.de 247 : 2538 : ExecForceStoreHeapTuple(tuple, slot, true);
3257 heikki.linnakangas@i 248 : 2538 : return slot;
249 : : }
250 : : }
251 [ + + ]: 38802 : else if (node->iss_ReachedEnd)
252 : : {
253 : : /* Queue is empty, and no more tuples from index. We're done. */
1822 andres@anarazel.de 254 : 9 : return ExecClearTuple(slot);
255 : : }
256 : :
257 : : /*
258 : : * Fetch next tuple from the index.
259 : : */
3257 heikki.linnakangas@i 260 : 41381 : next_indextuple:
1861 andres@anarazel.de 261 [ + + ]: 43451 : if (!index_getnext_slot(scandesc, ForwardScanDirection, slot))
262 : : {
263 : : /*
264 : : * No more tuples from the index. But we still need to drain any
265 : : * remaining tuples from the queue before we're done.
266 : : */
3257 heikki.linnakangas@i 267 : 9 : node->iss_ReachedEnd = true;
268 : 9 : continue;
269 : : }
270 : :
271 : : /*
272 : : * If the index was lossy, we have to recheck the index quals and
273 : : * ORDER BY expressions using the fetched tuple.
274 : : */
275 [ + + ]: 43442 : if (scandesc->xs_recheck)
276 : : {
277 : 4563 : econtext->ecxt_scantuple = slot;
2267 andres@anarazel.de 278 [ + + ]: 4563 : if (!ExecQualAndReset(node->indexqualorig, econtext))
279 : : {
280 : : /* Fails recheck, so drop it and loop back for another */
3257 heikki.linnakangas@i 281 [ - + ]: 2070 : InstrCountFiltered2(node, 1);
282 : : /* allow this loop to be cancellable */
2455 andres@anarazel.de 283 [ - + ]: 2070 : CHECK_FOR_INTERRUPTS();
3257 heikki.linnakangas@i 284 : 2070 : goto next_indextuple;
285 : : }
286 : : }
287 : :
288 [ + + ]: 41372 : if (scandesc->xs_recheckorderby)
289 : : {
290 : 2650 : econtext->ecxt_scantuple = slot;
291 : 2650 : ResetExprContext(econtext);
292 : 2650 : EvalOrderByExpressions(node, econtext);
293 : :
294 : : /*
295 : : * Was the ORDER BY value returned by the index accurate? The
296 : : * recheck flag means that the index can return inaccurate values,
297 : : * but then again, the value returned for any particular tuple
298 : : * could also be exactly correct. Compare the value returned by
299 : : * the index with the recalculated value. (If the value returned
300 : : * by the index happened to be exact right, we can often avoid
301 : : * pushing the tuple to the queue, just to pop it back out again.)
302 : : */
303 : 2650 : cmp = cmp_orderbyvals(node->iss_OrderByValues,
304 : 2650 : node->iss_OrderByNulls,
305 : 2650 : scandesc->xs_orderbyvals,
306 : 2650 : scandesc->xs_orderbynulls,
307 : : node);
308 [ - + ]: 2650 : if (cmp < 0)
3257 heikki.linnakangas@i 309 [ # # ]:UBC 0 : elog(ERROR, "index returned tuples in wrong order");
3257 heikki.linnakangas@i 310 [ + + ]:CBC 2650 : else if (cmp == 0)
311 : 66 : was_exact = true;
312 : : else
313 : 2584 : was_exact = false;
314 : 2650 : lastfetched_vals = node->iss_OrderByValues;
315 : 2650 : lastfetched_nulls = node->iss_OrderByNulls;
316 : : }
317 : : else
318 : : {
319 : 38722 : was_exact = true;
320 : 38722 : lastfetched_vals = scandesc->xs_orderbyvals;
321 : 38722 : lastfetched_nulls = scandesc->xs_orderbynulls;
322 : : }
323 : :
324 : : /*
325 : : * Can we return this tuple immediately, or does it need to be pushed
326 : : * to the reorder queue? If the ORDER BY expression values returned
327 : : * by the index were inaccurate, we can't return it yet, because the
328 : : * next tuple from the index might need to come before this one. Also,
329 : : * we can't return it yet if there are any smaller tuples in the queue
330 : : * already.
331 : : */
332 [ + + + + : 41417 : if (!was_exact || (topmost && cmp_orderbyvals(lastfetched_vals,
+ + ]
333 : : lastfetched_nulls,
334 : 45 : topmost->orderbyvals,
335 : 45 : topmost->orderbynulls,
336 : : node) > 0))
337 : : {
338 : : /* Put this tuple to the queue */
1861 andres@anarazel.de 339 : 2588 : reorderqueue_push(node, slot, lastfetched_vals, lastfetched_nulls);
3257 heikki.linnakangas@i 340 : 2588 : continue;
341 : : }
342 : : else
343 : : {
344 : : /* Can return this tuple immediately. */
345 : 38784 : return slot;
346 : : }
347 : : }
348 : :
349 : : /*
350 : : * if we get here it means the index scan failed so we are at the end of
351 : : * the scan..
352 : : */
353 : : return ExecClearTuple(slot);
354 : : }
355 : :
356 : : /*
357 : : * Calculate the expressions in the ORDER BY clause, based on the heap tuple.
358 : : */
359 : : static void
360 : 2650 : EvalOrderByExpressions(IndexScanState *node, ExprContext *econtext)
361 : : {
362 : : int i;
363 : : ListCell *l;
364 : : MemoryContext oldContext;
365 : :
366 : 2650 : oldContext = MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
367 : :
368 : 2650 : i = 0;
369 [ + - + + : 5300 : foreach(l, node->indexorderbyorig)
+ + ]
370 : : {
371 : 2650 : ExprState *orderby = (ExprState *) lfirst(l);
372 : :
373 : 5300 : node->iss_OrderByValues[i] = ExecEvalExpr(orderby,
374 : : econtext,
2642 andres@anarazel.de 375 : 2650 : &node->iss_OrderByNulls[i]);
3257 heikki.linnakangas@i 376 : 2650 : i++;
377 : : }
378 : :
379 : 2650 : MemoryContextSwitchTo(oldContext);
380 : 2650 : }
381 : :
382 : : /*
383 : : * IndexRecheck -- access method routine to recheck a tuple in EvalPlanQual
384 : : */
385 : : static bool
5284 tgl@sss.pgh.pa.us 386 : 50 : IndexRecheck(IndexScanState *node, TupleTableSlot *slot)
387 : : {
388 : : ExprContext *econtext;
389 : :
390 : : /*
391 : : * extract necessary information from index scan node
392 : : */
393 : 50 : econtext = node->ss.ps.ps_ExprContext;
394 : :
395 : : /* Does the tuple meet the indexqual condition? */
396 : 50 : econtext->ecxt_scantuple = slot;
2267 andres@anarazel.de 397 : 50 : return ExecQualAndReset(node->indexqualorig, econtext);
398 : : }
399 : :
400 : :
401 : : /*
402 : : * Compare ORDER BY expression values.
403 : : */
404 : : static int
3257 heikki.linnakangas@i 405 : 14615 : cmp_orderbyvals(const Datum *adist, const bool *anulls,
406 : : const Datum *bdist, const bool *bnulls,
407 : : IndexScanState *node)
408 : : {
409 : : int i;
410 : : int result;
411 : :
412 [ + + ]: 14710 : for (i = 0; i < node->iss_NumOrderByKeys; i++)
413 : : {
414 : 14615 : SortSupport ssup = &node->iss_SortSupport[i];
415 : :
416 : : /*
417 : : * Handle nulls. We only need to support NULLS LAST ordering, because
418 : : * match_pathkeys_to_index() doesn't consider indexorderby
419 : : * implementation otherwise.
420 : : */
421 [ - + - - ]: 14615 : if (anulls[i] && !bnulls[i])
3257 heikki.linnakangas@i 422 :UBC 0 : return 1;
3257 heikki.linnakangas@i 423 [ + - - + ]:CBC 14615 : else if (!anulls[i] && bnulls[i])
3257 heikki.linnakangas@i 424 :UBC 0 : return -1;
3257 heikki.linnakangas@i 425 [ - + - - ]:CBC 14615 : else if (anulls[i] && bnulls[i])
3257 heikki.linnakangas@i 426 :UBC 0 : return 0;
427 : :
3257 heikki.linnakangas@i 428 :CBC 14615 : result = ssup->comparator(adist[i], bdist[i], ssup);
429 [ + + ]: 14615 : if (result != 0)
430 : 14520 : return result;
431 : : }
432 : :
433 : 95 : return 0;
434 : : }
435 : :
436 : : /*
437 : : * Pairing heap provides getting topmost (greatest) element while KNN provides
438 : : * ascending sort. That's why we invert the sort order.
439 : : */
440 : : static int
441 : 6797 : reorderqueue_cmp(const pairingheap_node *a, const pairingheap_node *b,
442 : : void *arg)
443 : : {
444 : 6797 : ReorderTuple *rta = (ReorderTuple *) a;
445 : 6797 : ReorderTuple *rtb = (ReorderTuple *) b;
446 : 6797 : IndexScanState *node = (IndexScanState *) arg;
447 : :
448 : : /* exchange argument order to invert the sort order */
2018 tgl@sss.pgh.pa.us 449 : 13594 : return cmp_orderbyvals(rtb->orderbyvals, rtb->orderbynulls,
450 : 6797 : rta->orderbyvals, rta->orderbynulls,
451 : : node);
452 : : }
453 : :
454 : : /*
455 : : * Helper function to push a tuple to the reorder queue.
456 : : */
457 : : static void
1861 andres@anarazel.de 458 : 2588 : reorderqueue_push(IndexScanState *node, TupleTableSlot *slot,
459 : : Datum *orderbyvals, bool *orderbynulls)
460 : : {
3257 heikki.linnakangas@i 461 : 2588 : IndexScanDesc scandesc = node->iss_ScanDesc;
462 : 2588 : EState *estate = node->ss.ps.state;
463 : 2588 : MemoryContext oldContext = MemoryContextSwitchTo(estate->es_query_cxt);
464 : : ReorderTuple *rt;
465 : : int i;
466 : :
467 : 2588 : rt = (ReorderTuple *) palloc(sizeof(ReorderTuple));
1861 andres@anarazel.de 468 : 2588 : rt->htup = ExecCopySlotHeapTuple(slot);
3257 heikki.linnakangas@i 469 : 2588 : rt->orderbyvals =
470 : 2588 : (Datum *) palloc(sizeof(Datum) * scandesc->numberOfOrderBys);
471 : 2588 : rt->orderbynulls =
472 : 2588 : (bool *) palloc(sizeof(bool) * scandesc->numberOfOrderBys);
473 [ + + ]: 5176 : for (i = 0; i < node->iss_NumOrderByKeys; i++)
474 : : {
475 [ + - ]: 2588 : if (!orderbynulls[i])
476 : 2588 : rt->orderbyvals[i] = datumCopy(orderbyvals[i],
477 : 2588 : node->iss_OrderByTypByVals[i],
478 : 2588 : node->iss_OrderByTypLens[i]);
479 : : else
3257 heikki.linnakangas@i 480 :UBC 0 : rt->orderbyvals[i] = (Datum) 0;
3257 heikki.linnakangas@i 481 :CBC 2588 : rt->orderbynulls[i] = orderbynulls[i];
482 : : }
483 : 2588 : pairingheap_add(node->iss_ReorderQueue, &rt->ph_node);
484 : :
485 : 2588 : MemoryContextSwitchTo(oldContext);
486 : 2588 : }
487 : :
488 : : /*
489 : : * Helper function to pop the next tuple from the reorder queue.
490 : : */
491 : : static HeapTuple
492 : 2568 : reorderqueue_pop(IndexScanState *node)
493 : : {
494 : : HeapTuple result;
495 : : ReorderTuple *topmost;
496 : : int i;
497 : :
498 : 2568 : topmost = (ReorderTuple *) pairingheap_remove_first(node->iss_ReorderQueue);
499 : :
500 : 2568 : result = topmost->htup;
3249 tgl@sss.pgh.pa.us 501 [ + + ]: 5136 : for (i = 0; i < node->iss_NumOrderByKeys; i++)
502 : : {
503 [ - + - - ]: 2568 : if (!node->iss_OrderByTypByVals[i] && !topmost->orderbynulls[i])
3249 tgl@sss.pgh.pa.us 504 :UBC 0 : pfree(DatumGetPointer(topmost->orderbyvals[i]));
505 : : }
3257 heikki.linnakangas@i 506 :CBC 2568 : pfree(topmost->orderbyvals);
507 : 2568 : pfree(topmost->orderbynulls);
508 : 2568 : pfree(topmost);
509 : :
510 : 2568 : return result;
511 : : }
512 : :
513 : :
514 : : /* ----------------------------------------------------------------
515 : : * ExecIndexScan(node)
516 : : * ----------------------------------------------------------------
517 : : */
518 : : static TupleTableSlot *
2463 andres@anarazel.de 519 : 902786 : ExecIndexScan(PlanState *pstate)
520 : : {
521 : 902786 : IndexScanState *node = castNode(IndexScanState, pstate);
522 : :
523 : : /*
524 : : * If we have runtime keys and they've not already been set up, do it now.
525 : : */
6715 tgl@sss.pgh.pa.us 526 [ + + + + ]: 902786 : if (node->iss_NumRuntimeKeys != 0 && !node->iss_RuntimeKeysReady)
5025 527 : 8970 : ExecReScan((PlanState *) node);
528 : :
3257 heikki.linnakangas@i 529 [ + + ]: 902783 : if (node->iss_NumOrderByKeys > 0)
530 : 41331 : return ExecScan(&node->ss,
531 : : (ExecScanAccessMtd) IndexNextWithReorder,
532 : : (ExecScanRecheckMtd) IndexRecheck);
533 : : else
534 : 861452 : return ExecScan(&node->ss,
535 : : (ExecScanAccessMtd) IndexNext,
536 : : (ExecScanRecheckMtd) IndexRecheck);
537 : : }
538 : :
539 : : /* ----------------------------------------------------------------
540 : : * ExecReScanIndexScan(node)
541 : : *
542 : : * Recalculates the values of any scan keys whose value depends on
543 : : * information known at runtime, then rescans the indexed relation.
544 : : *
545 : : * Updating the scan key was formerly done separately in
546 : : * ExecUpdateIndexScanKeys. Integrating it into ReScan makes
547 : : * rescans of indices and relations/general streams more uniform.
548 : : * ----------------------------------------------------------------
549 : : */
550 : : void
5025 tgl@sss.pgh.pa.us 551 : 202154 : ExecReScanIndexScan(IndexScanState *node)
552 : : {
553 : : /*
554 : : * If we are doing runtime key calculations (ie, any of the index key
555 : : * values weren't simple Consts), compute the new key values. But first,
556 : : * reset the context so we don't leak memory as each outer tuple is
557 : : * scanned. Note this assumes that we will recalculate *all* runtime keys
558 : : * on each call.
559 : : */
6715 560 [ + + ]: 202154 : if (node->iss_NumRuntimeKeys != 0)
561 : : {
5025 562 : 196548 : ExprContext *econtext = node->iss_RuntimeContext;
563 : :
564 : 196548 : ResetExprContext(econtext);
6929 565 : 196548 : ExecIndexEvalRuntimeKeys(econtext,
566 : : node->iss_RuntimeKeys,
567 : : node->iss_NumRuntimeKeys);
568 : : }
6715 569 : 202151 : node->iss_RuntimeKeysReady = true;
570 : :
571 : : /* flush the reorder queue */
3247 heikki.linnakangas@i 572 [ + + ]: 202151 : if (node->iss_ReorderQueue)
573 : : {
574 : : HeapTuple tuple;
575 : :
576 [ + + ]: 63 : while (!pairingheap_is_empty(node->iss_ReorderQueue))
577 : : {
790 akorotkov@postgresql 578 : 30 : tuple = reorderqueue_pop(node);
579 : 30 : heap_freetuple(tuple);
580 : : }
581 : : }
582 : :
583 : : /* reset index scan */
2615 rhaas@postgresql.org 584 [ + + ]: 202151 : if (node->iss_ScanDesc)
585 : 176097 : index_rescan(node->iss_ScanDesc,
586 : 176097 : node->iss_ScanKeys, node->iss_NumScanKeys,
587 : 176097 : node->iss_OrderByKeys, node->iss_NumOrderByKeys);
3247 heikki.linnakangas@i 588 : 202151 : node->iss_ReachedEnd = false;
589 : :
5284 tgl@sss.pgh.pa.us 590 : 202151 : ExecScanReScan(&node->ss);
6929 591 : 202151 : }
592 : :
593 : :
594 : : /*
595 : : * ExecIndexEvalRuntimeKeys
596 : : * Evaluate any runtime key values, and update the scankeys.
597 : : */
598 : : void
599 : 239744 : ExecIndexEvalRuntimeKeys(ExprContext *econtext,
600 : : IndexRuntimeKeyInfo *runtimeKeys, int numRuntimeKeys)
601 : : {
602 : : int j;
603 : : MemoryContext oldContext;
604 : :
605 : : /* We want to keep the key values in per-tuple memory */
5348 606 : 239744 : oldContext = MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
607 : :
6715 608 [ + + ]: 489968 : for (j = 0; j < numRuntimeKeys; j++)
609 : : {
610 : 250227 : ScanKey scan_key = runtimeKeys[j].scan_key;
611 : 250227 : ExprState *key_expr = runtimeKeys[j].key_expr;
612 : : Datum scanvalue;
613 : : bool isNull;
614 : :
615 : : /*
616 : : * For each run-time key, extract the run-time expression and evaluate
617 : : * it with respect to the current context. We then stick the result
618 : : * into the proper scan key.
619 : : *
620 : : * Note: the result of the eval could be a pass-by-ref value that's
621 : : * stored in some outer scan's tuple, not in
622 : : * econtext->ecxt_per_tuple_memory. We assume that the outer tuple
623 : : * will stay put throughout our scan. If this is wrong, we could copy
624 : : * the result into our context explicitly, but I think that's not
625 : : * necessary.
626 : : *
627 : : * It's also entirely possible that the result of the eval is a
628 : : * toasted value. In this case we should forcibly detoast it, to
629 : : * avoid repeat detoastings each time the value is examined by an
630 : : * index support function.
631 : : */
5348 632 : 250227 : scanvalue = ExecEvalExpr(key_expr,
633 : : econtext,
634 : : &isNull);
6715 635 [ + + ]: 250224 : if (isNull)
636 : : {
5348 637 : 1366 : scan_key->sk_argument = scanvalue;
6715 638 : 1366 : scan_key->sk_flags |= SK_ISNULL;
639 : : }
640 : : else
641 : : {
5348 642 [ + + ]: 248858 : if (runtimeKeys[j].key_toastable)
643 : 1799 : scanvalue = PointerGetDatum(PG_DETOAST_DATUM(scanvalue));
644 : 248858 : scan_key->sk_argument = scanvalue;
6715 645 : 248858 : scan_key->sk_flags &= ~SK_ISNULL;
646 : : }
647 : : }
648 : :
5348 649 : 239741 : MemoryContextSwitchTo(oldContext);
6715 650 : 239741 : }
651 : :
652 : : /*
653 : : * ExecIndexEvalArrayKeys
654 : : * Evaluate any array key values, and set up to iterate through arrays.
655 : : *
656 : : * Returns true if there are array elements to consider; false means there
657 : : * is at least one null or empty array, so no match is possible. On true
658 : : * result, the scankeys are initialized with the first elements of the arrays.
659 : : */
660 : : bool
661 : 13 : ExecIndexEvalArrayKeys(ExprContext *econtext,
662 : : IndexArrayKeyInfo *arrayKeys, int numArrayKeys)
663 : : {
664 : 13 : bool result = true;
665 : : int j;
666 : : MemoryContext oldContext;
667 : :
668 : : /* We want to keep the arrays in per-tuple memory */
669 : 13 : oldContext = MemoryContextSwitchTo(econtext->ecxt_per_tuple_memory);
670 : :
671 [ + + ]: 26 : for (j = 0; j < numArrayKeys; j++)
672 : : {
673 : 13 : ScanKey scan_key = arrayKeys[j].scan_key;
674 : 13 : ExprState *array_expr = arrayKeys[j].array_expr;
675 : : Datum arraydatum;
676 : : bool isNull;
677 : : ArrayType *arrayval;
678 : : int16 elmlen;
679 : : bool elmbyval;
680 : : char elmalign;
681 : : int num_elems;
682 : : Datum *elem_values;
683 : : bool *elem_nulls;
684 : :
685 : : /*
686 : : * Compute and deconstruct the array expression. (Notes in
687 : : * ExecIndexEvalRuntimeKeys() apply here too.)
688 : : */
689 : 13 : arraydatum = ExecEvalExpr(array_expr,
690 : : econtext,
691 : : &isNull);
692 [ - + ]: 13 : if (isNull)
693 : : {
6715 tgl@sss.pgh.pa.us 694 :UBC 0 : result = false;
695 : 0 : break; /* no point in evaluating more */
696 : : }
6715 tgl@sss.pgh.pa.us 697 :CBC 13 : arrayval = DatumGetArrayTypeP(arraydatum);
698 : : /* We could cache this data, but not clear it's worth it */
699 : 13 : get_typlenbyvalalign(ARR_ELEMTYPE(arrayval),
700 : : &elmlen, &elmbyval, &elmalign);
701 : 13 : deconstruct_array(arrayval,
702 : : ARR_ELEMTYPE(arrayval),
703 : : elmlen, elmbyval, elmalign,
704 : : &elem_values, &elem_nulls, &num_elems);
705 [ - + ]: 13 : if (num_elems <= 0)
706 : : {
6715 tgl@sss.pgh.pa.us 707 :UBC 0 : result = false;
708 : 0 : break; /* no point in evaluating more */
709 : : }
710 : :
711 : : /*
712 : : * Note: we expect the previous array data, if any, to be
713 : : * automatically freed by resetting the per-tuple context; hence no
714 : : * pfree's here.
715 : : */
6715 tgl@sss.pgh.pa.us 716 :CBC 13 : arrayKeys[j].elem_values = elem_values;
717 : 13 : arrayKeys[j].elem_nulls = elem_nulls;
718 : 13 : arrayKeys[j].num_elems = num_elems;
719 : 13 : scan_key->sk_argument = elem_values[0];
720 [ - + ]: 13 : if (elem_nulls[0])
6715 tgl@sss.pgh.pa.us 721 :UBC 0 : scan_key->sk_flags |= SK_ISNULL;
722 : : else
6715 tgl@sss.pgh.pa.us 723 :CBC 13 : scan_key->sk_flags &= ~SK_ISNULL;
724 : 13 : arrayKeys[j].next_elem = 1;
725 : : }
726 : :
727 : 13 : MemoryContextSwitchTo(oldContext);
728 : :
729 : 13 : return result;
730 : : }
731 : :
732 : : /*
733 : : * ExecIndexAdvanceArrayKeys
734 : : * Advance to the next set of array key values, if any.
735 : : *
736 : : * Returns true if there is another set of values to consider, false if not.
737 : : * On true result, the scankeys are initialized with the next set of values.
738 : : */
739 : : bool
740 : 9365 : ExecIndexAdvanceArrayKeys(IndexArrayKeyInfo *arrayKeys, int numArrayKeys)
741 : : {
742 : 9365 : bool found = false;
743 : : int j;
744 : :
745 : : /*
746 : : * Note we advance the rightmost array key most quickly, since it will
747 : : * correspond to the lowest-order index column among the available
748 : : * qualifications. This is hypothesized to result in better locality of
749 : : * access in the index.
750 : : */
5871 751 [ + + ]: 9378 : for (j = numArrayKeys - 1; j >= 0; j--)
752 : : {
6715 753 : 26 : ScanKey scan_key = arrayKeys[j].scan_key;
754 : 26 : int next_elem = arrayKeys[j].next_elem;
755 : 26 : int num_elems = arrayKeys[j].num_elems;
756 : 26 : Datum *elem_values = arrayKeys[j].elem_values;
757 : 26 : bool *elem_nulls = arrayKeys[j].elem_nulls;
758 : :
759 [ + + ]: 26 : if (next_elem >= num_elems)
760 : : {
761 : 13 : next_elem = 0;
762 : 13 : found = false; /* need to advance next array key */
763 : : }
764 : : else
765 : 13 : found = true;
766 : 26 : scan_key->sk_argument = elem_values[next_elem];
767 [ - + ]: 26 : if (elem_nulls[next_elem])
6715 tgl@sss.pgh.pa.us 768 :UBC 0 : scan_key->sk_flags |= SK_ISNULL;
769 : : else
6715 tgl@sss.pgh.pa.us 770 :CBC 26 : scan_key->sk_flags &= ~SK_ISNULL;
771 : 26 : arrayKeys[j].next_elem = next_elem + 1;
772 [ + + ]: 26 : if (found)
773 : 13 : break;
774 : : }
775 : :
776 : 9365 : return found;
777 : : }
778 : :
779 : :
780 : : /* ----------------------------------------------------------------
781 : : * ExecEndIndexScan
782 : : * ----------------------------------------------------------------
783 : : */
784 : : void
7801 785 : 72767 : ExecEndIndexScan(IndexScanState *node)
786 : : {
787 : : Relation indexRelationDesc;
788 : : IndexScanDesc indexScanDesc;
789 : :
790 : : /*
791 : : * extract information from the node
792 : : */
6929 793 : 72767 : indexRelationDesc = node->iss_RelationDesc;
794 : 72767 : indexScanDesc = node->iss_ScanDesc;
795 : :
796 : : /*
797 : : * close the index relation (no-op if we didn't open it)
798 : : */
6169 799 [ + + ]: 72767 : if (indexScanDesc)
800 : 60219 : index_endscan(indexScanDesc);
801 [ + + ]: 72767 : if (indexRelationDesc)
802 : 71279 : index_close(indexRelationDesc, NoLock);
10141 scrappy@hub.org 803 : 72767 : }
804 : :
805 : : /* ----------------------------------------------------------------
806 : : * ExecIndexMarkPos
807 : : *
808 : : * Note: we assume that no caller attempts to set a mark before having read
809 : : * at least one tuple. Otherwise, iss_ScanDesc might still be NULL.
810 : : * ----------------------------------------------------------------
811 : : */
812 : : void
7801 tgl@sss.pgh.pa.us 813 : 3038 : ExecIndexMarkPos(IndexScanState *node)
814 : : {
2269 815 : 3038 : EState *estate = node->ss.ps.state;
1683 andres@anarazel.de 816 : 3038 : EPQState *epqstate = estate->es_epq_active;
817 : :
818 [ + + ]: 3038 : if (epqstate != NULL)
819 : : {
820 : : /*
821 : : * We are inside an EvalPlanQual recheck. If a test tuple exists for
822 : : * this relation, then we shouldn't access the index at all. We would
823 : : * instead need to save, and later restore, the state of the
824 : : * relsubs_done flag, so that re-fetching the test tuple is possible.
825 : : * However, given the assumption that no caller sets a mark at the
826 : : * start of the scan, we can only get here with relsubs_done[i]
827 : : * already set, and so no state need be saved.
828 : : */
2269 tgl@sss.pgh.pa.us 829 : 1 : Index scanrelid = ((Scan *) node->ss.ps.plan)->scanrelid;
830 : :
831 [ - + ]: 1 : Assert(scanrelid > 0);
1683 andres@anarazel.de 832 [ - + ]: 1 : if (epqstate->relsubs_slot[scanrelid - 1] != NULL ||
1683 andres@anarazel.de 833 [ # # ]:UBC 0 : epqstate->relsubs_rowmark[scanrelid - 1] != NULL)
834 : : {
835 : : /* Verify the claim above */
1683 andres@anarazel.de 836 [ - + ]:CBC 1 : if (!epqstate->relsubs_done[scanrelid - 1])
2269 tgl@sss.pgh.pa.us 837 [ # # ]:UBC 0 : elog(ERROR, "unexpected ExecIndexMarkPos call in EPQ recheck");
2269 tgl@sss.pgh.pa.us 838 :CBC 1 : return;
839 : : }
840 : : }
841 : :
6929 842 : 3037 : index_markpos(node->iss_ScanDesc);
843 : : }
844 : :
845 : : /* ----------------------------------------------------------------
846 : : * ExecIndexRestrPos
847 : : * ----------------------------------------------------------------
848 : : */
849 : : void
7801 850 : 27015 : ExecIndexRestrPos(IndexScanState *node)
851 : : {
2269 852 : 27015 : EState *estate = node->ss.ps.state;
1683 andres@anarazel.de 853 : 27015 : EPQState *epqstate = estate->es_epq_active;
854 : :
855 [ - + ]: 27015 : if (estate->es_epq_active != NULL)
856 : : {
857 : : /* See comments in ExecIndexMarkPos */
2269 tgl@sss.pgh.pa.us 858 :UBC 0 : Index scanrelid = ((Scan *) node->ss.ps.plan)->scanrelid;
859 : :
860 [ # # ]: 0 : Assert(scanrelid > 0);
1683 andres@anarazel.de 861 [ # # ]: 0 : if (epqstate->relsubs_slot[scanrelid - 1] != NULL ||
862 [ # # ]: 0 : epqstate->relsubs_rowmark[scanrelid - 1] != NULL)
863 : : {
864 : : /* Verify the claim above */
865 [ # # ]: 0 : if (!epqstate->relsubs_done[scanrelid - 1])
2269 tgl@sss.pgh.pa.us 866 [ # # ]: 0 : elog(ERROR, "unexpected ExecIndexRestrPos call in EPQ recheck");
867 : 0 : return;
868 : : }
869 : : }
870 : :
6929 tgl@sss.pgh.pa.us 871 :CBC 27015 : index_restrpos(node->iss_ScanDesc);
872 : : }
873 : :
874 : : /* ----------------------------------------------------------------
875 : : * ExecInitIndexScan
876 : : *
877 : : * Initializes the index scan's state information, creates
878 : : * scan keys, and opens the base and index relations.
879 : : *
880 : : * Note: index scans have 2 sets of state information because
881 : : * we have to keep track of the base relation and the
882 : : * index relation.
883 : : * ----------------------------------------------------------------
884 : : */
885 : : IndexScanState *
6620 886 : 73170 : ExecInitIndexScan(IndexScan *node, EState *estate, int eflags)
887 : : {
888 : : IndexScanState *indexstate;
889 : : Relation currentRelation;
890 : : LOCKMODE lockmode;
891 : :
892 : : /*
893 : : * create state structure
894 : : */
7801 895 : 73170 : indexstate = makeNode(IndexScanState);
896 : 73170 : indexstate->ss.ps.plan = (Plan *) node;
897 : 73170 : indexstate->ss.ps.state = estate;
2463 andres@anarazel.de 898 : 73170 : indexstate->ss.ps.ExecProcNode = ExecIndexScan;
899 : :
900 : : /*
901 : : * Miscellaneous initialization
902 : : *
903 : : * create expression context for node
904 : : */
7801 tgl@sss.pgh.pa.us 905 : 73170 : ExecAssignExprContext(estate, &indexstate->ss.ps);
906 : :
907 : : /*
908 : : * open the scan relation
909 : : */
2249 andres@anarazel.de 910 : 73170 : currentRelation = ExecOpenScanRelation(estate, node->scan.scanrelid, eflags);
911 : :
912 : 73170 : indexstate->ss.ss_currentRelation = currentRelation;
913 : 73170 : indexstate->ss.ss_currentScanDesc = NULL; /* no heap scan here */
914 : :
915 : : /*
916 : : * get the scan type from the relation descriptor.
917 : : */
918 : 73170 : ExecInitScanTupleSlot(estate, &indexstate->ss,
919 : : RelationGetDescr(currentRelation),
920 : : table_slot_callbacks(currentRelation));
921 : :
922 : : /*
923 : : * Initialize result type and projection.
924 : : */
1983 925 : 73170 : ExecInitResultTypeTL(&indexstate->ss.ps);
2249 926 : 73170 : ExecAssignScanProjectionInfo(&indexstate->ss);
927 : :
928 : : /*
929 : : * initialize child expressions
930 : : *
931 : : * Note: we don't initialize all of the indexqual expression, only the
932 : : * sub-parts corresponding to runtime keys (see below). Likewise for
933 : : * indexorderby, if any. But the indexqualorig expression is always
934 : : * initialized even though it will only be used in some uncommon cases ---
935 : : * would be nice to improve that. (Problem is that any SubPlans present
936 : : * in the expression must be found now...)
937 : : */
2588 938 : 73170 : indexstate->ss.ps.qual =
939 : 73170 : ExecInitQual(node->scan.plan.qual, (PlanState *) indexstate);
940 : 73170 : indexstate->indexqualorig =
941 : 73170 : ExecInitQual(node->indexqualorig, (PlanState *) indexstate);
942 : 73170 : indexstate->indexorderbyorig =
943 : 73170 : ExecInitExprList(node->indexorderbyorig, (PlanState *) indexstate);
944 : :
945 : : /*
946 : : * If we are just doing EXPLAIN (ie, aren't going to run the plan), stop
947 : : * here. This allows an index-advisor plugin to EXPLAIN a plan containing
948 : : * references to nonexistent indexes.
949 : : */
6169 tgl@sss.pgh.pa.us 950 [ + + ]: 73170 : if (eflags & EXEC_FLAG_EXPLAIN_ONLY)
951 : 1488 : return indexstate;
952 : :
953 : : /* Open the index relation. */
1837 954 : 71682 : lockmode = exec_rt_fetch(node->scan.scanrelid, estate)->rellockmode;
955 : 71682 : indexstate->iss_RelationDesc = index_open(node->indexid, lockmode);
956 : :
957 : : /*
958 : : * Initialize index-specific scan state
959 : : */
8645 960 : 71682 : indexstate->iss_RuntimeKeysReady = false;
4882 961 : 71682 : indexstate->iss_RuntimeKeys = NULL;
962 : 71682 : indexstate->iss_NumRuntimeKeys = 0;
963 : :
964 : : /*
965 : : * build the index scan keys from the index qualification
966 : : */
6715 967 : 71682 : ExecIndexBuildScanKeys((PlanState *) indexstate,
968 : : indexstate->iss_RelationDesc,
969 : : node->indexqual,
970 : : false,
971 : 71682 : &indexstate->iss_ScanKeys,
972 : : &indexstate->iss_NumScanKeys,
973 : : &indexstate->iss_RuntimeKeys,
974 : : &indexstate->iss_NumRuntimeKeys,
975 : : NULL, /* no ArrayKeys */
976 : : NULL);
977 : :
978 : : /*
979 : : * any ORDER BY exprs have to be turned into scankeys in the same way
980 : : */
4882 981 : 71682 : ExecIndexBuildScanKeys((PlanState *) indexstate,
982 : : indexstate->iss_RelationDesc,
983 : : node->indexorderby,
984 : : true,
985 : 71682 : &indexstate->iss_OrderByKeys,
986 : : &indexstate->iss_NumOrderByKeys,
987 : : &indexstate->iss_RuntimeKeys,
988 : : &indexstate->iss_NumRuntimeKeys,
989 : : NULL, /* no ArrayKeys */
990 : : NULL);
991 : :
992 : : /* Initialize sort support, if we need to re-check ORDER BY exprs */
3257 heikki.linnakangas@i 993 [ + + ]: 71682 : if (indexstate->iss_NumOrderByKeys > 0)
994 : : {
995 : 23 : int numOrderByKeys = indexstate->iss_NumOrderByKeys;
996 : : int i;
997 : : ListCell *lco;
998 : : ListCell *lcx;
999 : :
1000 : : /*
1001 : : * Prepare sort support, and look up the data type for each ORDER BY
1002 : : * expression.
1003 : : */
3255 tgl@sss.pgh.pa.us 1004 [ - + ]: 23 : Assert(numOrderByKeys == list_length(node->indexorderbyops));
3251 1005 [ - + ]: 23 : Assert(numOrderByKeys == list_length(node->indexorderbyorig));
1006 : 23 : indexstate->iss_SortSupport = (SortSupportData *)
3257 heikki.linnakangas@i 1007 : 23 : palloc0(numOrderByKeys * sizeof(SortSupportData));
3251 tgl@sss.pgh.pa.us 1008 : 23 : indexstate->iss_OrderByTypByVals = (bool *)
3257 heikki.linnakangas@i 1009 : 23 : palloc(numOrderByKeys * sizeof(bool));
3251 tgl@sss.pgh.pa.us 1010 : 23 : indexstate->iss_OrderByTypLens = (int16 *)
3257 heikki.linnakangas@i 1011 : 23 : palloc(numOrderByKeys * sizeof(int16));
3255 tgl@sss.pgh.pa.us 1012 : 23 : i = 0;
3251 1013 [ + - + + : 46 : forboth(lco, node->indexorderbyops, lcx, node->indexorderbyorig)
+ - + + +
+ + - +
+ ]
1014 : : {
1015 : 23 : Oid orderbyop = lfirst_oid(lco);
1016 : 23 : Node *orderbyexpr = (Node *) lfirst(lcx);
1017 : 23 : Oid orderbyType = exprType(orderbyexpr);
2870 1018 : 23 : Oid orderbyColl = exprCollation(orderbyexpr);
1019 : 23 : SortSupport orderbysort = &indexstate->iss_SortSupport[i];
1020 : :
1021 : : /* Initialize sort support */
1022 : 23 : orderbysort->ssup_cxt = CurrentMemoryContext;
1023 : 23 : orderbysort->ssup_collation = orderbyColl;
1024 : : /* See cmp_orderbyvals() comments on NULLS LAST */
1025 : 23 : orderbysort->ssup_nulls_first = false;
1026 : : /* ssup_attno is unused here and elsewhere */
1027 : 23 : orderbysort->ssup_attno = 0;
1028 : : /* No abbreviation */
1029 : 23 : orderbysort->abbreviate = false;
1030 : 23 : PrepareSortSupportFromOrderingOp(orderbyop, orderbysort);
1031 : :
3257 heikki.linnakangas@i 1032 : 23 : get_typlenbyval(orderbyType,
1033 : 23 : &indexstate->iss_OrderByTypLens[i],
1034 : 23 : &indexstate->iss_OrderByTypByVals[i]);
3255 tgl@sss.pgh.pa.us 1035 : 23 : i++;
1036 : : }
1037 : :
1038 : : /* allocate arrays to hold the re-calculated distances */
3251 1039 : 23 : indexstate->iss_OrderByValues = (Datum *)
1040 : 23 : palloc(numOrderByKeys * sizeof(Datum));
1041 : 23 : indexstate->iss_OrderByNulls = (bool *)
1042 : 23 : palloc(numOrderByKeys * sizeof(bool));
1043 : :
1044 : : /* and initialize the reorder queue */
3257 heikki.linnakangas@i 1045 : 23 : indexstate->iss_ReorderQueue = pairingheap_allocate(reorderqueue_cmp,
1046 : : indexstate);
1047 : : }
1048 : :
1049 : : /*
1050 : : * If we have runtime keys, we need an ExprContext to evaluate them. The
1051 : : * node's standard context won't do because we want to reset that context
1052 : : * for every tuple. So, build another context just like the other one...
1053 : : * -tgl 7/11/00
1054 : : */
6715 tgl@sss.pgh.pa.us 1055 [ + + ]: 71682 : if (indexstate->iss_NumRuntimeKeys != 0)
1056 : : {
7801 1057 : 31464 : ExprContext *stdecontext = indexstate->ss.ps.ps_ExprContext;
1058 : :
1059 : 31464 : ExecAssignExprContext(estate, &indexstate->ss.ps);
1060 : 31464 : indexstate->iss_RuntimeContext = indexstate->ss.ps.ps_ExprContext;
1061 : 31464 : indexstate->ss.ps.ps_ExprContext = stdecontext;
1062 : : }
1063 : : else
1064 : : {
8677 1065 : 40218 : indexstate->iss_RuntimeContext = NULL;
1066 : : }
1067 : :
1068 : : /*
1069 : : * all done.
1070 : : */
7801 1071 : 71682 : return indexstate;
1072 : : }
1073 : :
1074 : :
1075 : : /*
1076 : : * ExecIndexBuildScanKeys
1077 : : * Build the index scan keys from the index qualification expressions
1078 : : *
1079 : : * The index quals are passed to the index AM in the form of a ScanKey array.
1080 : : * This routine sets up the ScanKeys, fills in all constant fields of the
1081 : : * ScanKeys, and prepares information about the keys that have non-constant
1082 : : * comparison values. We divide index qual expressions into five types:
1083 : : *
1084 : : * 1. Simple operator with constant comparison value ("indexkey op constant").
1085 : : * For these, we just fill in a ScanKey containing the constant value.
1086 : : *
1087 : : * 2. Simple operator with non-constant value ("indexkey op expression").
1088 : : * For these, we create a ScanKey with everything filled in except the
1089 : : * expression value, and set up an IndexRuntimeKeyInfo struct to drive
1090 : : * evaluation of the expression at the right times.
1091 : : *
1092 : : * 3. RowCompareExpr ("(indexkey, indexkey, ...) op (expr, expr, ...)").
1093 : : * For these, we create a header ScanKey plus a subsidiary ScanKey array,
1094 : : * as specified in access/skey.h. The elements of the row comparison
1095 : : * can have either constant or non-constant comparison values.
1096 : : *
1097 : : * 4. ScalarArrayOpExpr ("indexkey op ANY (array-expression)"). If the index
1098 : : * supports amsearcharray, we handle these the same as simple operators,
1099 : : * setting the SK_SEARCHARRAY flag to tell the AM to handle them. Otherwise,
1100 : : * we create a ScanKey with everything filled in except the comparison value,
1101 : : * and set up an IndexArrayKeyInfo struct to drive processing of the qual.
1102 : : * (Note that if we use an IndexArrayKeyInfo struct, the array expression is
1103 : : * always treated as requiring runtime evaluation, even if it's a constant.)
1104 : : *
1105 : : * 5. NullTest ("indexkey IS NULL/IS NOT NULL"). We just fill in the
1106 : : * ScanKey properly.
1107 : : *
1108 : : * This code is also used to prepare ORDER BY expressions for amcanorderbyop
1109 : : * indexes. The behavior is exactly the same, except that we have to look up
1110 : : * the operator differently. Note that only cases 1 and 2 are currently
1111 : : * possible for ORDER BY.
1112 : : *
1113 : : * Input params are:
1114 : : *
1115 : : * planstate: executor state node we are working for
1116 : : * index: the index we are building scan keys for
1117 : : * quals: indexquals (or indexorderbys) expressions
1118 : : * isorderby: true if processing ORDER BY exprs, false if processing quals
1119 : : * *runtimeKeys: ptr to pre-existing IndexRuntimeKeyInfos, or NULL if none
1120 : : * *numRuntimeKeys: number of pre-existing runtime keys
1121 : : *
1122 : : * Output params are:
1123 : : *
1124 : : * *scanKeys: receives ptr to array of ScanKeys
1125 : : * *numScanKeys: receives number of scankeys
1126 : : * *runtimeKeys: receives ptr to array of IndexRuntimeKeyInfos, or NULL if none
1127 : : * *numRuntimeKeys: receives number of runtime keys
1128 : : * *arrayKeys: receives ptr to array of IndexArrayKeyInfos, or NULL if none
1129 : : * *numArrayKeys: receives number of array keys
1130 : : *
1131 : : * Caller may pass NULL for arrayKeys and numArrayKeys to indicate that
1132 : : * IndexArrayKeyInfos are not supported.
1133 : : */
1134 : : void
4569 1135 : 162935 : ExecIndexBuildScanKeys(PlanState *planstate, Relation index,
1136 : : List *quals, bool isorderby,
1137 : : ScanKey *scanKeys, int *numScanKeys,
1138 : : IndexRuntimeKeyInfo **runtimeKeys, int *numRuntimeKeys,
1139 : : IndexArrayKeyInfo **arrayKeys, int *numArrayKeys)
1140 : : {
1141 : : ListCell *qual_cell;
1142 : : ScanKey scan_keys;
1143 : : IndexRuntimeKeyInfo *runtime_keys;
1144 : : IndexArrayKeyInfo *array_keys;
1145 : : int n_scan_keys;
1146 : : int n_runtime_keys;
1147 : : int max_runtime_keys;
1148 : : int n_array_keys;
1149 : : int j;
1150 : :
1151 : : /* Allocate array for ScanKey structs: one per qual */
4882 1152 : 162935 : n_scan_keys = list_length(quals);
1153 : 162935 : scan_keys = (ScanKey) palloc(n_scan_keys * sizeof(ScanKeyData));
1154 : :
1155 : : /*
1156 : : * runtime_keys array is dynamically resized as needed. We handle it this
1157 : : * way so that the same runtime keys array can be shared between
1158 : : * indexquals and indexorderbys, which will be processed in separate calls
1159 : : * of this function. Caller must be sure to pass in NULL/0 for first
1160 : : * call.
1161 : : */
1162 : 162935 : runtime_keys = *runtimeKeys;
1163 : 162935 : n_runtime_keys = max_runtime_keys = *numRuntimeKeys;
1164 : :
1165 : : /* Allocate array_keys as large as it could possibly need to be */
1166 : : array_keys = (IndexArrayKeyInfo *)
6715 1167 : 162935 : palloc0(n_scan_keys * sizeof(IndexArrayKeyInfo));
1168 : 162935 : n_array_keys = 0;
1169 : :
1170 : : /*
1171 : : * for each opclause in the given qual, convert the opclause into a single
1172 : : * scan key
1173 : : */
5845 1174 : 162935 : j = 0;
1175 [ + + + + : 260168 : foreach(qual_cell, quals)
+ + ]
1176 : : {
1177 : 97233 : Expr *clause = (Expr *) lfirst(qual_cell);
1178 : 97233 : ScanKey this_scan_key = &scan_keys[j++];
1179 : : Oid opno; /* operator's OID */
1180 : : RegProcedure opfuncid; /* operator proc id used in scan */
1181 : : Oid opfamily; /* opfamily of index column */
1182 : : int op_strategy; /* operator's strategy number */
1183 : : Oid op_lefttype; /* operator's declared input types */
1184 : : Oid op_righttype;
1185 : : Expr *leftop; /* expr on lhs of operator */
1186 : : Expr *rightop; /* expr on rhs ... */
1187 : : AttrNumber varattno; /* att number used in scan */
1188 : : int indnkeyatts;
1189 : :
2199 teodor@sigaev.ru 1190 : 97233 : indnkeyatts = IndexRelationGetNumberOfKeyAttributes(index);
6715 tgl@sss.pgh.pa.us 1191 [ + + ]: 97233 : if (IsA(clause, OpExpr))
1192 : : {
1193 : : /* indexkey op const or indexkey op expression */
1194 : 96424 : int flags = 0;
1195 : : Datum scanvalue;
1196 : :
5845 1197 : 96424 : opno = ((OpExpr *) clause)->opno;
6715 1198 : 96424 : opfuncid = ((OpExpr *) clause)->opfuncid;
1199 : :
1200 : : /*
1201 : : * leftop should be the index key Var, possibly relabeled
1202 : : */
1203 : 96424 : leftop = (Expr *) get_leftop(clause);
1204 : :
1205 [ + - - + ]: 96424 : if (leftop && IsA(leftop, RelabelType))
6715 tgl@sss.pgh.pa.us 1206 :UBC 0 : leftop = ((RelabelType *) leftop)->arg;
1207 : :
6715 tgl@sss.pgh.pa.us 1208 [ - + ]:CBC 96424 : Assert(leftop != NULL);
1209 : :
1210 [ + - ]: 96424 : if (!(IsA(leftop, Var) &&
4569 1211 [ - + ]: 96424 : ((Var *) leftop)->varno == INDEX_VAR))
6715 tgl@sss.pgh.pa.us 1212 [ # # ]:UBC 0 : elog(ERROR, "indexqual doesn't have key on left side");
1213 : :
6715 tgl@sss.pgh.pa.us 1214 :CBC 96424 : varattno = ((Var *) leftop)->varattno;
2199 teodor@sigaev.ru 1215 [ + - - + ]: 96424 : if (varattno < 1 || varattno > indnkeyatts)
5845 tgl@sss.pgh.pa.us 1216 [ # # ]:UBC 0 : elog(ERROR, "bogus index qualification");
1217 : :
1218 : : /*
1219 : : * We have to look up the operator's strategy number. This
1220 : : * provides a cross-check that the operator does match the index.
1221 : : */
5845 tgl@sss.pgh.pa.us 1222 :CBC 96424 : opfamily = index->rd_opfamily[varattno - 1];
1223 : :
4882 1224 : 96424 : get_op_opfamily_properties(opno, opfamily, isorderby,
1225 : : &op_strategy,
1226 : : &op_lefttype,
1227 : : &op_righttype);
1228 : :
1229 [ + + ]: 96424 : if (isorderby)
1230 : 99 : flags |= SK_ORDER_BY;
1231 : :
1232 : : /*
1233 : : * rightop is the constant or variable comparison value
1234 : : */
6715 1235 : 96424 : rightop = (Expr *) get_rightop(clause);
1236 : :
1237 [ + - + + ]: 96424 : if (rightop && IsA(rightop, RelabelType))
1238 : 407 : rightop = ((RelabelType *) rightop)->arg;
1239 : :
1240 [ - + ]: 96424 : Assert(rightop != NULL);
1241 : :
1242 [ + + ]: 96424 : if (IsA(rightop, Const))
1243 : : {
1244 : : /* OK, simple constant comparison value */
1245 : 58862 : scanvalue = ((Const *) rightop)->constvalue;
1246 [ - + ]: 58862 : if (((Const *) rightop)->constisnull)
6715 tgl@sss.pgh.pa.us 1247 :UBC 0 : flags |= SK_ISNULL;
1248 : : }
1249 : : else
1250 : : {
1251 : : /* Need to treat this one as a runtime key */
4882 tgl@sss.pgh.pa.us 1252 [ + + ]:CBC 37562 : if (n_runtime_keys >= max_runtime_keys)
1253 : : {
1254 [ + + ]: 33245 : if (max_runtime_keys == 0)
1255 : : {
1256 : 33242 : max_runtime_keys = 8;
1257 : : runtime_keys = (IndexRuntimeKeyInfo *)
1258 : 33242 : palloc(max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1259 : : }
1260 : : else
1261 : : {
1262 : 3 : max_runtime_keys *= 2;
1263 : : runtime_keys = (IndexRuntimeKeyInfo *)
1264 : 3 : repalloc(runtime_keys, max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1265 : : }
1266 : : }
6715 1267 : 37562 : runtime_keys[n_runtime_keys].scan_key = this_scan_key;
1268 : 75124 : runtime_keys[n_runtime_keys].key_expr =
1269 : 37562 : ExecInitExpr(rightop, planstate);
5348 1270 : 37562 : runtime_keys[n_runtime_keys].key_toastable =
1271 : 37562 : TypeIsToastable(op_righttype);
6715 1272 : 37562 : n_runtime_keys++;
1273 : 37562 : scanvalue = (Datum) 0;
1274 : : }
1275 : :
1276 : : /*
1277 : : * initialize the scan key's fields appropriately
1278 : : */
1279 : 96424 : ScanKeyEntryInitialize(this_scan_key,
1280 : : flags,
1281 : : varattno, /* attribute number to scan */
1282 : : op_strategy, /* op's strategy */
1283 : : op_righttype, /* strategy subtype */
1284 : : ((OpExpr *) clause)->inputcollid, /* collation */
1285 : : opfuncid, /* reg proc to use */
1286 : : scanvalue); /* constant */
1287 : : }
6654 1288 [ + + ]: 809 : else if (IsA(clause, RowCompareExpr))
1289 : : {
1290 : : /* (indexkey, indexkey, ...) op (expression, expression, ...) */
1291 : 18 : RowCompareExpr *rc = (RowCompareExpr *) clause;
1292 : : ScanKey first_sub_key;
1293 : : int n_sub_key;
1294 : : ListCell *largs_cell;
1295 : : ListCell *rargs_cell;
1296 : : ListCell *opnos_cell;
1297 : : ListCell *collids_cell;
1298 : :
4882 1299 [ - + ]: 18 : Assert(!isorderby);
1300 : :
1301 : : first_sub_key = (ScanKey)
1302 : 18 : palloc(list_length(rc->opnos) * sizeof(ScanKeyData));
1303 : 18 : n_sub_key = 0;
1304 : :
1305 : : /* Scan RowCompare columns and generate subsidiary ScanKey items */
1872 1306 [ + - + + : 54 : forfour(largs_cell, rc->largs, rargs_cell, rc->rargs,
+ - + + +
- + + + -
+ + + + +
- + - + -
+ + ]
1307 : : opnos_cell, rc->opnos, collids_cell, rc->inputcollids)
1308 : : {
4882 1309 : 36 : ScanKey this_sub_key = &first_sub_key[n_sub_key];
6654 1310 : 36 : int flags = SK_ROW_MEMBER;
1311 : : Datum scanvalue;
1312 : : Oid inputcollation;
1313 : :
1872 1314 : 36 : leftop = (Expr *) lfirst(largs_cell);
1315 : 36 : rightop = (Expr *) lfirst(rargs_cell);
1316 : 36 : opno = lfirst_oid(opnos_cell);
1317 : 36 : inputcollation = lfirst_oid(collids_cell);
1318 : :
1319 : : /*
1320 : : * leftop should be the index key Var, possibly relabeled
1321 : : */
6654 1322 [ + - - + ]: 36 : if (leftop && IsA(leftop, RelabelType))
6654 tgl@sss.pgh.pa.us 1323 :UBC 0 : leftop = ((RelabelType *) leftop)->arg;
1324 : :
6654 tgl@sss.pgh.pa.us 1325 [ - + ]:CBC 36 : Assert(leftop != NULL);
1326 : :
1327 [ + - ]: 36 : if (!(IsA(leftop, Var) &&
4569 1328 [ - + ]: 36 : ((Var *) leftop)->varno == INDEX_VAR))
6654 tgl@sss.pgh.pa.us 1329 [ # # ]:UBC 0 : elog(ERROR, "indexqual doesn't have key on left side");
1330 : :
6654 tgl@sss.pgh.pa.us 1331 :CBC 36 : varattno = ((Var *) leftop)->varattno;
1332 : :
1333 : : /*
1334 : : * We have to look up the operator's associated btree support
1335 : : * function
1336 : : */
5348 1337 [ + - + - ]: 36 : if (index->rd_rel->relam != BTREE_AM_OID ||
2199 teodor@sigaev.ru 1338 [ - + ]: 36 : varattno < 1 || varattno > indnkeyatts)
5348 tgl@sss.pgh.pa.us 1339 [ # # ]:UBC 0 : elog(ERROR, "bogus RowCompare index qualification");
5348 tgl@sss.pgh.pa.us 1340 :CBC 36 : opfamily = index->rd_opfamily[varattno - 1];
1341 : :
4882 1342 : 36 : get_op_opfamily_properties(opno, opfamily, isorderby,
1343 : : &op_strategy,
1344 : : &op_lefttype,
1345 : : &op_righttype);
1346 : :
5348 1347 [ - + ]: 36 : if (op_strategy != rc->rctype)
5348 tgl@sss.pgh.pa.us 1348 [ # # ]:UBC 0 : elog(ERROR, "RowCompare index qualification contains wrong operator");
1349 : :
5348 tgl@sss.pgh.pa.us 1350 :CBC 36 : opfuncid = get_opfamily_proc(opfamily,
1351 : : op_lefttype,
1352 : : op_righttype,
1353 : : BTORDER_PROC);
2456 1354 [ - + ]: 36 : if (!RegProcedureIsValid(opfuncid))
2456 tgl@sss.pgh.pa.us 1355 [ # # ]:UBC 0 : elog(ERROR, "missing support function %d(%u,%u) in opfamily %u",
1356 : : BTORDER_PROC, op_lefttype, op_righttype, opfamily);
1357 : :
1358 : : /*
1359 : : * rightop is the constant or variable comparison value
1360 : : */
6654 tgl@sss.pgh.pa.us 1361 [ + - - + ]:CBC 36 : if (rightop && IsA(rightop, RelabelType))
6654 tgl@sss.pgh.pa.us 1362 :UBC 0 : rightop = ((RelabelType *) rightop)->arg;
1363 : :
6654 tgl@sss.pgh.pa.us 1364 [ - + ]:CBC 36 : Assert(rightop != NULL);
1365 : :
1366 [ + - ]: 36 : if (IsA(rightop, Const))
1367 : : {
1368 : : /* OK, simple constant comparison value */
1369 : 36 : scanvalue = ((Const *) rightop)->constvalue;
1370 [ - + ]: 36 : if (((Const *) rightop)->constisnull)
6654 tgl@sss.pgh.pa.us 1371 :UBC 0 : flags |= SK_ISNULL;
1372 : : }
1373 : : else
1374 : : {
1375 : : /* Need to treat this one as a runtime key */
4882 1376 [ # # ]: 0 : if (n_runtime_keys >= max_runtime_keys)
1377 : : {
1378 [ # # ]: 0 : if (max_runtime_keys == 0)
1379 : : {
1380 : 0 : max_runtime_keys = 8;
1381 : : runtime_keys = (IndexRuntimeKeyInfo *)
1382 : 0 : palloc(max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1383 : : }
1384 : : else
1385 : : {
1386 : 0 : max_runtime_keys *= 2;
1387 : : runtime_keys = (IndexRuntimeKeyInfo *)
1388 : 0 : repalloc(runtime_keys, max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1389 : : }
1390 : : }
6654 1391 : 0 : runtime_keys[n_runtime_keys].scan_key = this_sub_key;
1392 : 0 : runtime_keys[n_runtime_keys].key_expr =
1393 : 0 : ExecInitExpr(rightop, planstate);
5348 1394 : 0 : runtime_keys[n_runtime_keys].key_toastable =
1395 : 0 : TypeIsToastable(op_righttype);
6654 1396 : 0 : n_runtime_keys++;
1397 : 0 : scanvalue = (Datum) 0;
1398 : : }
1399 : :
1400 : : /*
1401 : : * initialize the subsidiary scan key's fields appropriately
1402 : : */
6654 tgl@sss.pgh.pa.us 1403 :CBC 36 : ScanKeyEntryInitialize(this_sub_key,
1404 : : flags,
1405 : : varattno, /* attribute number */
1406 : : op_strategy, /* op's strategy */
1407 : : op_righttype, /* strategy subtype */
1408 : : inputcollation, /* collation */
1409 : : opfuncid, /* reg proc to use */
1410 : : scanvalue); /* constant */
4882 1411 : 36 : n_sub_key++;
1412 : : }
1413 : :
1414 : : /* Mark the last subsidiary scankey correctly */
1415 : 18 : first_sub_key[n_sub_key - 1].sk_flags |= SK_ROW_END;
1416 : :
1417 : : /*
1418 : : * We don't use ScanKeyEntryInitialize for the header because it
1419 : : * isn't going to contain a valid sk_func pointer.
1420 : : */
6654 1421 [ + - + - : 180 : MemSet(this_scan_key, 0, sizeof(ScanKeyData));
+ - + - +
+ ]
1422 : 18 : this_scan_key->sk_flags = SK_ROW_HEADER;
1423 : 18 : this_scan_key->sk_attno = first_sub_key->sk_attno;
1424 : 18 : this_scan_key->sk_strategy = rc->rctype;
1425 : : /* sk_subtype, sk_collation, sk_func not used in a header */
1426 : 18 : this_scan_key->sk_argument = PointerGetDatum(first_sub_key);
1427 : : }
6715 1428 [ + + ]: 791 : else if (IsA(clause, ScalarArrayOpExpr))
1429 : : {
1430 : : /* indexkey op ANY (array-expression) */
1431 : 527 : ScalarArrayOpExpr *saop = (ScalarArrayOpExpr *) clause;
4564 1432 : 527 : int flags = 0;
1433 : : Datum scanvalue;
1434 : :
4882 1435 [ - + ]: 527 : Assert(!isorderby);
1436 : :
6715 1437 [ - + ]: 527 : Assert(saop->useOr);
5845 1438 : 527 : opno = saop->opno;
6715 1439 : 527 : opfuncid = saop->opfuncid;
1440 : :
1441 : : /*
1442 : : * leftop should be the index key Var, possibly relabeled
1443 : : */
1444 : 527 : leftop = (Expr *) linitial(saop->args);
1445 : :
1446 [ + - - + ]: 527 : if (leftop && IsA(leftop, RelabelType))
6715 tgl@sss.pgh.pa.us 1447 :UBC 0 : leftop = ((RelabelType *) leftop)->arg;
1448 : :
6715 tgl@sss.pgh.pa.us 1449 [ - + ]:CBC 527 : Assert(leftop != NULL);
1450 : :
1451 [ + - ]: 527 : if (!(IsA(leftop, Var) &&
4569 1452 [ - + ]: 527 : ((Var *) leftop)->varno == INDEX_VAR))
6715 tgl@sss.pgh.pa.us 1453 [ # # ]:UBC 0 : elog(ERROR, "indexqual doesn't have key on left side");
1454 : :
6715 tgl@sss.pgh.pa.us 1455 :CBC 527 : varattno = ((Var *) leftop)->varattno;
2199 teodor@sigaev.ru 1456 [ + - - + ]: 527 : if (varattno < 1 || varattno > indnkeyatts)
5845 tgl@sss.pgh.pa.us 1457 [ # # ]:UBC 0 : elog(ERROR, "bogus index qualification");
1458 : :
1459 : : /*
1460 : : * We have to look up the operator's strategy number. This
1461 : : * provides a cross-check that the operator does match the index.
1462 : : */
5845 tgl@sss.pgh.pa.us 1463 :CBC 527 : opfamily = index->rd_opfamily[varattno - 1];
1464 : :
4882 1465 : 527 : get_op_opfamily_properties(opno, opfamily, isorderby,
1466 : : &op_strategy,
1467 : : &op_lefttype,
1468 : : &op_righttype);
1469 : :
1470 : : /*
1471 : : * rightop is the constant or variable array value
1472 : : */
6715 1473 : 527 : rightop = (Expr *) lsecond(saop->args);
1474 : :
1475 [ + - - + ]: 527 : if (rightop && IsA(rightop, RelabelType))
6715 tgl@sss.pgh.pa.us 1476 :UBC 0 : rightop = ((RelabelType *) rightop)->arg;
1477 : :
6715 tgl@sss.pgh.pa.us 1478 [ - + ]:CBC 527 : Assert(rightop != NULL);
1479 : :
1910 andres@anarazel.de 1480 [ + + ]: 527 : if (index->rd_indam->amsearcharray)
1481 : : {
1482 : : /* Index AM will handle this like a simple operator */
4564 tgl@sss.pgh.pa.us 1483 : 514 : flags |= SK_SEARCHARRAY;
1484 [ + + ]: 514 : if (IsA(rightop, Const))
1485 : : {
1486 : : /* OK, simple constant comparison value */
1487 : 484 : scanvalue = ((Const *) rightop)->constvalue;
1488 [ - + ]: 484 : if (((Const *) rightop)->constisnull)
4564 tgl@sss.pgh.pa.us 1489 :UBC 0 : flags |= SK_ISNULL;
1490 : : }
1491 : : else
1492 : : {
1493 : : /* Need to treat this one as a runtime key */
4564 tgl@sss.pgh.pa.us 1494 [ + - ]:CBC 30 : if (n_runtime_keys >= max_runtime_keys)
1495 : : {
1496 [ + - ]: 30 : if (max_runtime_keys == 0)
1497 : : {
1498 : 30 : max_runtime_keys = 8;
1499 : : runtime_keys = (IndexRuntimeKeyInfo *)
1500 : 30 : palloc(max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1501 : : }
1502 : : else
1503 : : {
4564 tgl@sss.pgh.pa.us 1504 :UBC 0 : max_runtime_keys *= 2;
1505 : : runtime_keys = (IndexRuntimeKeyInfo *)
1506 : 0 : repalloc(runtime_keys, max_runtime_keys * sizeof(IndexRuntimeKeyInfo));
1507 : : }
1508 : : }
4564 tgl@sss.pgh.pa.us 1509 :CBC 30 : runtime_keys[n_runtime_keys].scan_key = this_scan_key;
1510 : 60 : runtime_keys[n_runtime_keys].key_expr =
1511 : 30 : ExecInitExpr(rightop, planstate);
1512 : :
1513 : : /*
1514 : : * Careful here: the runtime expression is not of
1515 : : * op_righttype, but rather is an array of same; so
1516 : : * TypeIsToastable() isn't helpful. However, we can
1517 : : * assume that all array types are toastable.
1518 : : */
1519 : 30 : runtime_keys[n_runtime_keys].key_toastable = true;
1520 : 30 : n_runtime_keys++;
1521 : 30 : scanvalue = (Datum) 0;
1522 : : }
1523 : : }
1524 : : else
1525 : : {
1526 : : /* Executor has to expand the array value */
1527 : 13 : array_keys[n_array_keys].scan_key = this_scan_key;
1528 : 26 : array_keys[n_array_keys].array_expr =
1529 : 13 : ExecInitExpr(rightop, planstate);
1530 : : /* the remaining fields were zeroed by palloc0 */
1531 : 13 : n_array_keys++;
1532 : 13 : scanvalue = (Datum) 0;
1533 : : }
1534 : :
1535 : : /*
1536 : : * initialize the scan key's fields appropriately
1537 : : */
6715 1538 : 527 : ScanKeyEntryInitialize(this_scan_key,
1539 : : flags,
1540 : : varattno, /* attribute number to scan */
1541 : : op_strategy, /* op's strategy */
1542 : : op_righttype, /* strategy subtype */
1543 : : saop->inputcollid, /* collation */
1544 : : opfuncid, /* reg proc to use */
1545 : : scanvalue); /* constant */
1546 : : }
6218 1547 [ + - ]: 264 : else if (IsA(clause, NullTest))
1548 : : {
1549 : : /* indexkey IS NULL or indexkey IS NOT NULL */
5217 1550 : 264 : NullTest *ntest = (NullTest *) clause;
1551 : : int flags;
1552 : :
4882 1553 [ - + ]: 264 : Assert(!isorderby);
1554 : :
1555 : : /*
1556 : : * argument should be the index key Var, possibly relabeled
1557 : : */
5217 1558 : 264 : leftop = ntest->arg;
1559 : :
6218 1560 [ + - - + ]: 264 : if (leftop && IsA(leftop, RelabelType))
6218 tgl@sss.pgh.pa.us 1561 :UBC 0 : leftop = ((RelabelType *) leftop)->arg;
1562 : :
5995 bruce@momjian.us 1563 [ - + ]:CBC 264 : Assert(leftop != NULL);
1564 : :
6218 tgl@sss.pgh.pa.us 1565 [ + - ]: 264 : if (!(IsA(leftop, Var) &&
4569 1566 [ - + ]: 264 : ((Var *) leftop)->varno == INDEX_VAR))
6218 tgl@sss.pgh.pa.us 1567 [ # # ]:UBC 0 : elog(ERROR, "NullTest indexqual has wrong key");
1568 : :
6218 tgl@sss.pgh.pa.us 1569 :CBC 264 : varattno = ((Var *) leftop)->varattno;
1570 : :
1571 : : /*
1572 : : * initialize the scan key's fields appropriately
1573 : : */
5217 1574 [ + + - ]: 264 : switch (ntest->nulltesttype)
1575 : : {
1576 : 91 : case IS_NULL:
1577 : 91 : flags = SK_ISNULL | SK_SEARCHNULL;
1578 : 91 : break;
1579 : 173 : case IS_NOT_NULL:
1580 : 173 : flags = SK_ISNULL | SK_SEARCHNOTNULL;
1581 : 173 : break;
5217 tgl@sss.pgh.pa.us 1582 :UBC 0 : default:
1583 [ # # ]: 0 : elog(ERROR, "unrecognized nulltesttype: %d",
1584 : : (int) ntest->nulltesttype);
1585 : : flags = 0; /* keep compiler quiet */
1586 : : break;
1587 : : }
1588 : :
6218 tgl@sss.pgh.pa.us 1589 :CBC 264 : ScanKeyEntryInitialize(this_scan_key,
1590 : : flags,
1591 : : varattno, /* attribute number to scan */
1592 : : InvalidStrategy, /* no strategy */
1593 : : InvalidOid, /* no strategy subtype */
1594 : : InvalidOid, /* no collation */
1595 : : InvalidOid, /* no reg proc for this */
1596 : : (Datum) 0); /* constant */
1597 : : }
1598 : : else
6715 tgl@sss.pgh.pa.us 1599 [ # # ]:UBC 0 : elog(ERROR, "unsupported indexqual type: %d",
1600 : : (int) nodeTag(clause));
1601 : : }
1602 : :
4882 tgl@sss.pgh.pa.us 1603 [ - + ]:CBC 162935 : Assert(n_runtime_keys <= max_runtime_keys);
1604 : :
1605 : : /* Get rid of any unused arrays */
6715 1606 [ + + ]: 162935 : if (n_array_keys == 0)
1607 : : {
1608 : 162922 : pfree(array_keys);
1609 : 162922 : array_keys = NULL;
1610 : : }
1611 : :
1612 : : /*
1613 : : * Return info to our caller.
1614 : : */
6929 1615 : 162935 : *scanKeys = scan_keys;
6715 1616 : 162935 : *numScanKeys = n_scan_keys;
1617 : 162935 : *runtimeKeys = runtime_keys;
1618 : 162935 : *numRuntimeKeys = n_runtime_keys;
1619 [ + + ]: 162935 : if (arrayKeys)
1620 : : {
1621 : 7987 : *arrayKeys = array_keys;
1622 : 7987 : *numArrayKeys = n_array_keys;
1623 : : }
1624 [ - + ]: 154948 : else if (n_array_keys != 0)
6715 tgl@sss.pgh.pa.us 1625 [ # # ]:UBC 0 : elog(ERROR, "ScalarArrayOpExpr index qual found where not allowed");
7541 tgl@sss.pgh.pa.us 1626 :CBC 162935 : }
1627 : :
1628 : : /* ----------------------------------------------------------------
1629 : : * Parallel Scan Support
1630 : : * ----------------------------------------------------------------
1631 : : */
1632 : :
1633 : : /* ----------------------------------------------------------------
1634 : : * ExecIndexScanEstimate
1635 : : *
1636 : : * Compute the amount of space we'll need in the parallel
1637 : : * query DSM, and inform pcxt->estimator about our needs.
1638 : : * ----------------------------------------------------------------
1639 : : */
1640 : : void
2615 rhaas@postgresql.org 1641 : 9 : ExecIndexScanEstimate(IndexScanState *node,
1642 : : ParallelContext *pcxt)
1643 : : {
1644 : 9 : EState *estate = node->ss.ps.state;
1645 : :
1646 : 9 : node->iss_PscanLen = index_parallelscan_estimate(node->iss_RelationDesc,
1647 : : node->iss_NumScanKeys,
1648 : : node->iss_NumOrderByKeys,
1649 : : estate->es_snapshot);
1650 : 9 : shm_toc_estimate_chunk(&pcxt->estimator, node->iss_PscanLen);
1651 : 9 : shm_toc_estimate_keys(&pcxt->estimator, 1);
1652 : 9 : }
1653 : :
1654 : : /* ----------------------------------------------------------------
1655 : : * ExecIndexScanInitializeDSM
1656 : : *
1657 : : * Set up a parallel index scan descriptor.
1658 : : * ----------------------------------------------------------------
1659 : : */
1660 : : void
1661 : 9 : ExecIndexScanInitializeDSM(IndexScanState *node,
1662 : : ParallelContext *pcxt)
1663 : : {
1664 : 9 : EState *estate = node->ss.ps.state;
1665 : : ParallelIndexScanDesc piscan;
1666 : :
1667 : 9 : piscan = shm_toc_allocate(pcxt->toc, node->iss_PscanLen);
1668 : 9 : index_parallelscan_initialize(node->ss.ss_currentRelation,
1669 : : node->iss_RelationDesc,
1670 : : estate->es_snapshot,
1671 : : piscan);
1672 : 9 : shm_toc_insert(pcxt->toc, node->ss.ps.plan->plan_node_id, piscan);
1673 : 9 : node->iss_ScanDesc =
1674 : 9 : index_beginscan_parallel(node->ss.ss_currentRelation,
1675 : : node->iss_RelationDesc,
1676 : : node->iss_NumScanKeys,
1677 : : node->iss_NumOrderByKeys,
1678 : : piscan);
1679 : :
1680 : : /*
1681 : : * If no run-time keys to calculate or they are ready, go ahead and pass
1682 : : * the scankeys to the index AM.
1683 : : */
2594 1684 [ + + - + ]: 9 : if (node->iss_NumRuntimeKeys == 0 || node->iss_RuntimeKeysReady)
2615 1685 : 6 : index_rescan(node->iss_ScanDesc,
1686 : 6 : node->iss_ScanKeys, node->iss_NumScanKeys,
1687 : 6 : node->iss_OrderByKeys, node->iss_NumOrderByKeys);
1688 : 9 : }
1689 : :
1690 : : /* ----------------------------------------------------------------
1691 : : * ExecIndexScanReInitializeDSM
1692 : : *
1693 : : * Reset shared state before beginning a fresh scan.
1694 : : * ----------------------------------------------------------------
1695 : : */
1696 : : void
2419 tgl@sss.pgh.pa.us 1697 : 6 : ExecIndexScanReInitializeDSM(IndexScanState *node,
1698 : : ParallelContext *pcxt)
1699 : : {
1700 : 6 : index_parallelrescan(node->iss_ScanDesc);
1701 : 6 : }
1702 : :
1703 : : /* ----------------------------------------------------------------
1704 : : * ExecIndexScanInitializeWorker
1705 : : *
1706 : : * Copy relevant information from TOC into planstate.
1707 : : * ----------------------------------------------------------------
1708 : : */
1709 : : void
2341 andres@anarazel.de 1710 : 60 : ExecIndexScanInitializeWorker(IndexScanState *node,
1711 : : ParallelWorkerContext *pwcxt)
1712 : : {
1713 : : ParallelIndexScanDesc piscan;
1714 : :
1715 : 60 : piscan = shm_toc_lookup(pwcxt->toc, node->ss.ps.plan->plan_node_id, false);
2615 rhaas@postgresql.org 1716 : 60 : node->iss_ScanDesc =
1717 : 60 : index_beginscan_parallel(node->ss.ss_currentRelation,
1718 : : node->iss_RelationDesc,
1719 : : node->iss_NumScanKeys,
1720 : : node->iss_NumOrderByKeys,
1721 : : piscan);
1722 : :
1723 : : /*
1724 : : * If no run-time keys to calculate or they are ready, go ahead and pass
1725 : : * the scankeys to the index AM.
1726 : : */
2594 1727 [ + + - + ]: 60 : if (node->iss_NumRuntimeKeys == 0 || node->iss_RuntimeKeysReady)
2615 1728 : 48 : index_rescan(node->iss_ScanDesc,
1729 : 48 : node->iss_ScanKeys, node->iss_NumScanKeys,
1730 : 48 : node->iss_OrderByKeys, node->iss_NumOrderByKeys);
1731 : 60 : }
|