From 099f34335abbd95904fa23d85b8d4513a1975792 Mon Sep 17 00:00:00 2001 From: Peter Geoghegan Date: Sat, 16 Nov 2024 15:58:41 -0500 Subject: [PATCH v18 3/3] Add "skipskip" nbtree skip scan optimization. Fix regressions in cases that are nominally eligible to use skip scan but can never actually benefit from skipping. These are cases where the leading skipped prefix column contains many distinct values -- often as many distinct values are there are total index tuples. For example, the test case posted here is fixed by the work from this commit: https://postgr.es/m/51d00219180323d121572e1f83ccde2a@oss.nttdata.com Note that this commit doesn't actually change anything about when or how skip scan decides when or how to skip. It just avoids wasting CPU cycles on uselessly maintaining a skip array at the tuple granularity, preferring to maintain the skip arrays at something closer to the page granularity when that makes sense. See: https://www.postgresql.org/message-id/flat/CAH2-Wz%3DE7XrkvscBN0U6V81NK3Q-dQOmivvbEsjG-zwEfDdFpg%40mail.gmail.com#7d34e8aa875d7a718043834c5ef4c167 Doing well on cases like this is important because we can't expect the optimizer to never choose an affected plan -- we prefer to solve these problems in the executor, which has access to the most reliable and current information about the index. The optimizer can afford to be very optimistic about skipping if actual runtime scan behavior is very similar to a traditional full index scan in the worst case. See "optimizer" section from the original intro mail for more information: https://postgr.es/m/CAH2-Wzmn1YsLzOGgjAQZdn1STSG_y8qP__vggTaPAYXJP%2BG4bw%40mail.gmail.com Author: Peter Geoghegan Reviewed-By: Masahiro Ikeda Discussion: https://postgr.es/m/CAH2-Wz=Y93jf5WjoOsN=xvqpMjRy-bxCE037bVFi-EasrpeUJA@mail.gmail.com --- src/include/access/nbtree.h | 8 + src/backend/access/nbtree/nbtree.c | 1 + src/backend/access/nbtree/nbtsearch.c | 34 +++- src/backend/access/nbtree/nbtutils.c | 222 +++++++++++++++++++++----- 4 files changed, 222 insertions(+), 43 deletions(-) diff --git a/src/include/access/nbtree.h b/src/include/access/nbtree.h index d841e85bc..fd5de8181 100644 --- a/src/include/access/nbtree.h +++ b/src/include/access/nbtree.h @@ -1051,6 +1051,7 @@ typedef struct BTScanOpaqueData /* workspace for SK_SEARCHARRAY support */ int numArrayKeys; /* number of equality-type array keys */ + bool skipScan; /* At least one skip array in arrayKeys[]? */ bool needPrimScan; /* New prim scan to continue in current dir? */ bool scanBehind; /* Last array advancement matched -inf attr? */ bool oppositeDirCheck; /* explicit scanBehind recheck needed? */ @@ -1111,6 +1112,13 @@ typedef struct BTReadPageState bool prechecked; /* precheck set continuescan to 'true'? */ bool firstmatch; /* at least one match so far? */ + /* + * Input and output parameters, set and unset by both _bt_readpage and + * _bt_checkkeys to manage "skipskip" optimization during skip scans + */ + bool skipskip; + bool noskipskip; + /* * Private _bt_checkkeys state used to manage "look ahead" optimization * (only used during scans with array keys) diff --git a/src/backend/access/nbtree/nbtree.c b/src/backend/access/nbtree/nbtree.c index 923629d5e..bf239f811 100644 --- a/src/backend/access/nbtree/nbtree.c +++ b/src/backend/access/nbtree/nbtree.c @@ -423,6 +423,7 @@ btrescan(IndexScanDesc scan, ScanKey scankey, int nscankeys, memcpy(scan->keyData, scankey, scan->numberOfKeys * sizeof(ScanKeyData)); so->numberOfKeys = 0; /* until _bt_preprocess_keys sets it */ so->numArrayKeys = 0; /* ditto */ + so->skipScan = false; /* ditto */ } /* diff --git a/src/backend/access/nbtree/nbtsearch.c b/src/backend/access/nbtree/nbtsearch.c index 43e321896..db000b8f6 100644 --- a/src/backend/access/nbtree/nbtsearch.c +++ b/src/backend/access/nbtree/nbtsearch.c @@ -1644,6 +1644,26 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum, pstate.continuescan = true; /* default assumption */ pstate.prechecked = false; pstate.firstmatch = false; + + /* + * Initialize "skipskip" optimization state (used only during scans with + * skip arrays). + * + * Skip scans use this to manage the overhead of maintaining skip arrays + * on columns with many distinct values. It also works as a substitute + * for the pstate.prechecked optimization, which skip scan never uses. + * + * We never do this for the first page read by each primitive scan. This + * avoids slowing down queries with skip arrays that have relatively few + * distinct values -- the "look ahead" optimization is preferred there. + */ + pstate.skipskip = false; + pstate.noskipskip = firstPage; + + /* + * Initialize "look ahead" optimization state (used only during scans with + * arrays, including those that just use skip arrays) + */ pstate.rechecks = 0; pstate.targetdistance = 0; @@ -1660,10 +1680,10 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum, * corresponding value from the last item on the page. So checking with * the last item on the page would give a more precise answer. * - * We skip this for the first page read by each (primitive) scan, to avoid - * slowing down point queries. They typically don't stand to gain much - * when the optimization can be applied, and are more likely to notice the - * overhead of the precheck. + * We don't do this for the first page read by each (primitive) scan, to + * avoid slowing down point queries. They typically don't stand to gain + * much when the optimization can be applied, and are more likely to + * notice the overhead of the precheck. Also avoid it during skip scans. * * The optimization is unsafe and must be avoided whenever _bt_checkkeys * just set a low-order required array's key to the best available match @@ -1687,7 +1707,7 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum, * required < or <= strategy scan keys) during the precheck, we can safely * assume that this must also be true of all earlier tuples from the page. */ - if (!firstPage && !so->scanBehind && minoff < maxoff) + if (!firstPage && !so->skipScan && !so->scanBehind && minoff < maxoff) { ItemId iid; IndexTuple itup; @@ -1837,6 +1857,8 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum, truncatt = BTreeTupleGetNAtts(itup, rel); pstate.prechecked = false; /* precheck didn't cover HIKEY */ + if (pstate.skipskip) + pstate.skipskip = false; /* reset for finaltup */ _bt_checkkeys(scan, &pstate, arrayKeys, itup, truncatt); } @@ -1897,6 +1919,8 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum, Assert(!BTreeTupleIsPivot(itup)); pstate.offnum = offnum; + if (offnum == minoff && pstate.skipskip) + pstate.skipskip = false; /* reset for finaltup */ passes_quals = _bt_checkkeys(scan, &pstate, arrayKeys, itup, indnatts); diff --git a/src/backend/access/nbtree/nbtutils.c b/src/backend/access/nbtree/nbtutils.c index 12905e799..1f7ae5f7a 100644 --- a/src/backend/access/nbtree/nbtutils.c +++ b/src/backend/access/nbtree/nbtutils.c @@ -151,11 +151,14 @@ static bool _bt_fix_scankey_strategy(ScanKey skey, int16 *indoption); static void _bt_mark_scankey_required(ScanKey skey); static bool _bt_check_compare(IndexScanDesc scan, ScanDirection dir, IndexTuple tuple, int tupnatts, TupleDesc tupdesc, - bool advancenonrequired, bool prechecked, bool firstmatch, + bool advancenonrequired, bool skipskip, + bool prechecked, bool firstmatch, bool *continuescan, int *ikey); static bool _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, int tupnatts, TupleDesc tupdesc, ScanDirection dir, bool *continuescan); +static bool _bt_checkkeys_skipskip(IndexScanDesc scan, IndexTuple tuple, + IndexTuple finaltup, TupleDesc tupdesc); static void _bt_checkkeys_look_ahead(IndexScanDesc scan, BTReadPageState *pstate, int tupnatts, TupleDesc tupdesc); static int _bt_keep_natts(Relation rel, IndexTuple lastleft, @@ -364,6 +367,7 @@ _bt_preprocess_array_keys(IndexScanDesc scan, int *new_numberOfKeys) * is the size of the original scan->keyData[] input array, plus space for * any additional skip array scan keys described within skipatts[] */ + so->skipScan = (numSkipArrayKeys > 0); numArrayKeyData = scan->numberOfKeys + numSkipArrayKeys; /* @@ -2883,7 +2887,7 @@ _bt_tuple_before_array_skeys(IndexScanDesc scan, ScanDirection dir, tupdatum, tupnull, cur->sk_argument, cur); - if (result == 0 && (cur->sk_flags & SK_BT_NEXT)) + if (unlikely(result == 0 && (cur->sk_flags & SK_BT_NEXT))) { /* * tupdatum is == sk_argument, but true current array element @@ -2903,7 +2907,7 @@ _bt_tuple_before_array_skeys(IndexScanDesc scan, ScanDirection dir, */ return false; } - else if (result == 0 && (cur->sk_flags & SK_BT_PRIOR)) + else if (unlikely(result == 0 && (cur->sk_flags & SK_BT_PRIOR))) { /* * tupdatum is == sk_argument, but true current array element @@ -3211,8 +3215,6 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, if (cur->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD)) { - Assert(sktrig_required); - required = true; if (cur->sk_attno > tupnatts) @@ -3359,7 +3361,7 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, } else { - Assert(sktrig_required && required); + Assert(required); /* * This is a required non-array equality strategy scan key, which @@ -3401,7 +3403,7 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, * be eliminated by _bt_preprocess_keys. It won't matter if some of * our "true" array scan keys (or even all of them) are non-required. */ - if (required && + if (sktrig_required && required && ((ScanDirectionIsForward(dir) && result > 0) || (ScanDirectionIsBackward(dir) && result < 0))) beyond_end_advance = true; @@ -3416,7 +3418,7 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, * array scan keys are considered interesting.) */ all_satisfied = false; - if (required) + if (sktrig_required && required) all_required_satisfied = false; else { @@ -3528,7 +3530,7 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, /* Recheck _bt_check_compare on behalf of caller */ if (_bt_check_compare(scan, dir, tuple, tupnatts, tupdesc, - false, false, false, + false, !sktrig_required, false, false, &continuescan, &nsktrig) && !so->scanBehind) { @@ -3597,6 +3599,9 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, return false; } + /* _bt_readpage must have unset skipskip flag (for finaltup call) */ + Assert(!pstate->skipskip); + /* * Postcondition array state assertion (for still-unsatisfied tuples). * @@ -3766,6 +3771,25 @@ _bt_advance_array_keys(IndexScanDesc scan, BTReadPageState *pstate, pstate->skip = pstate->maxoff + 1; } + /* + * Optimization: if a scan with a skip array doesn't satisfy every + * required key (in practice this is almost always all the scan's keys), + * we assume that this page isn't likely to skip "within" a page using + * _bt_checkkeys_look_ahead. We'll apply the 'skipskip' optimization. + * + * The 'skipskip' optimization allows _bt_checkkeys/_bt_check_compare to + * stop maintaining the scan's skip arrays until we've reached finaltup. + */ + else if (so->skipScan && !pstate->noskipskip && !all_required_satisfied && + tuple != pstate->finaltup) + { + /* Don't consider 'skipskip' optimization more than once per page */ + if (_bt_checkkeys_skipskip(scan, tuple, pstate->finaltup, tupdesc)) + pstate->skipskip = true; + else + pstate->noskipskip = true; + } + /* Caller's tuple doesn't match the new qual */ return false; @@ -3867,7 +3891,8 @@ end_toplevel_scan: * make the scan much more efficient: if there are few distinct values in "x", * we'll be able to skip over many irrelevant leaf pages. (If on the other * hand there are many distinct values in "x" then the scan will degenerate - * into a full index scan at run time.) + * into a full index scan at run time, but we'll be no worse off overall. + * _bt_checkkeys's 'skipskip' optimization keeps the runtime overhead low.) * * If possible, redundant keys are eliminated: we keep only the tightest * >/>= bound and the tightest indexRelation) == tupnatts); res = _bt_check_compare(scan, dir, tuple, tupnatts, tupdesc, - arrayKeys, pstate->prechecked, pstate->firstmatch, + arrayKeys, pstate->skipskip, + pstate->prechecked, pstate->firstmatch, &pstate->continuescan, &ikey); #ifdef USE_ASSERT_CHECKING @@ -4920,7 +4946,8 @@ _bt_checkkeys(IndexScanDesc scan, BTReadPageState *pstate, bool arrayKeys, * Assert that the scan isn't in danger of becoming confused. */ Assert(!so->scanBehind && !so->oppositeDirCheck); - Assert(!pstate->prechecked && !pstate->firstmatch); + Assert(!pstate->skipskip && !pstate->prechecked && + !pstate->firstmatch); Assert(!_bt_tuple_before_array_skeys(scan, dir, tuple, tupdesc, tupnatts, false, 0, NULL)); } @@ -4934,7 +4961,7 @@ _bt_checkkeys(IndexScanDesc scan, BTReadPageState *pstate, bool arrayKeys, * get the same answer without those optimizations */ Assert(res == _bt_check_compare(scan, dir, tuple, tupnatts, tupdesc, - false, false, false, + false, pstate->skipskip, false, false, &dcontinuescan, &dikey)); Assert(pstate->continuescan == dcontinuescan); } @@ -5065,7 +5092,7 @@ _bt_oppodir_checkkeys(IndexScanDesc scan, ScanDirection dir, Assert(so->numArrayKeys); _bt_check_compare(scan, flipped, finaltup, nfinaltupatts, tupdesc, - false, false, false, &continuescan, &ikey); + false, false, false, false, &continuescan, &ikey); if (!continuescan && so->keyData[ikey].sk_strategy != BTEqualStrategyNumber) return false; @@ -5104,17 +5131,24 @@ _bt_oppodir_checkkeys(IndexScanDesc scan, ScanDirection dir, * * Though we advance non-required array keys on our own, that shouldn't have * any lasting consequences for the scan. By definition, non-required arrays - * have no fixed relationship with the scan's progress. (There are delicate - * considerations for non-required arrays when the arrays need to be advanced - * following our setting continuescan to false, but that doesn't concern us.) + * have no fixed relationship with the scan's progress. (skipskip=true makes + * us treat required scan keys as non-required, though. That's safe because + * _bt_readpage will account for our failure to fully maintain the scan's + * arrays later on, right before its finaltup call to _bt_checkkeys.) * * Pass advancenonrequired=false to avoid all array related side effects. * This allows _bt_advance_array_keys caller to avoid infinite recursion. + * + * Pass skipskip=true to instruct us to skip all of the scan's skip arrays. + * This provides the scan with a way of keeping the cost of maintaining its + * skip arrays under control, given skip arrays on high cardinality columns + * (i.e. given a skip array on a column which isn't a good fit for skip scan). */ static bool _bt_check_compare(IndexScanDesc scan, ScanDirection dir, IndexTuple tuple, int tupnatts, TupleDesc tupdesc, - bool advancenonrequired, bool prechecked, bool firstmatch, + bool advancenonrequired, bool skipskip, + bool prechecked, bool firstmatch, bool *continuescan, int *ikey) { BTScanOpaque so = (BTScanOpaque) scan->opaque; @@ -5124,6 +5158,10 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, for (; *ikey < so->numberOfKeys; (*ikey)++) { ScanKey key = so->keyData + *ikey; + int sk_flags = key->sk_flags; + Oid sk_collation = key->sk_collation; + FmgrInfo sk_func = key->sk_func; + Datum sk_argument = key->sk_argument; Datum datum; bool isNull; bool requiredSameDir = false, @@ -5131,13 +5169,21 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, /* * Check if the key is required in the current scan direction, in the - * opposite scan direction _only_, or in neither direction + * opposite scan direction _only_, or in neither direction (except + * when reading a page that's now using the "skipskip" optimization) */ - if (((key->sk_flags & SK_BT_REQFWD) && ScanDirectionIsForward(dir)) || - ((key->sk_flags & SK_BT_REQBKWD) && ScanDirectionIsBackward(dir))) + if (unlikely(skipskip)) + { + Assert(!prechecked); + + if (sk_flags & SK_BT_SKIP) + continue; + } + else if (((sk_flags & SK_BT_REQFWD) && ScanDirectionIsForward(dir)) || + ((sk_flags & SK_BT_REQBKWD) && ScanDirectionIsBackward(dir))) requiredSameDir = true; - else if (((key->sk_flags & SK_BT_REQFWD) && ScanDirectionIsBackward(dir)) || - ((key->sk_flags & SK_BT_REQBKWD) && ScanDirectionIsForward(dir))) + else if (((sk_flags & SK_BT_REQFWD) && ScanDirectionIsBackward(dir)) || + ((sk_flags & SK_BT_REQBKWD) && ScanDirectionIsForward(dir))) requiredOppositeDirOnly = true; /* @@ -5158,7 +5204,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, */ if (prechecked && (requiredSameDir || (requiredOppositeDirOnly && firstmatch)) && - !(key->sk_flags & SK_ROW_HEADER)) + !(sk_flags & SK_ROW_HEADER)) continue; if (key->sk_attno > tupnatts) @@ -5174,7 +5220,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, } /* row-comparison keys need special processing */ - if (key->sk_flags & SK_ROW_HEADER) + if (sk_flags & SK_ROW_HEADER) { if (_bt_check_rowcompare(key, tuple, tupnatts, tupdesc, dir, continuescan)) @@ -5191,28 +5237,27 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, * A skip array scan key might be negative/positive infinity. Might * also be next key/prior key sentinel, which we don't deal with. */ - if (key->sk_flags & (SK_BT_NEGPOSINF | SK_BT_NEXT | SK_BT_PRIOR)) + if (unlikely(sk_flags & (SK_BT_NEGPOSINF | SK_BT_NEXT | SK_BT_PRIOR))) { - Assert(key->sk_flags & SK_SEARCHARRAY); - Assert(key->sk_flags & SK_BT_SKIP); + Assert(sk_flags & SK_SEARCHARRAY); + Assert(sk_flags & SK_BT_SKIP); Assert(requiredSameDir); *continuescan = false; return false; } - - if (key->sk_flags & SK_ISNULL) + if (sk_flags & SK_ISNULL) { /* Handle IS NULL/NOT NULL tests */ - if (key->sk_flags & SK_SEARCHNULL) + if (sk_flags & SK_SEARCHNULL) { if (isNull) continue; /* tuple satisfies this qual */ } else { - Assert(key->sk_flags & SK_SEARCHNOTNULL); + Assert(sk_flags & SK_SEARCHNOTNULL); if (!isNull) continue; /* tuple satisfies this qual */ } @@ -5233,7 +5278,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, if (isNull) { - if (key->sk_flags & SK_BT_NULLS_FIRST) + if (sk_flags & SK_BT_NULLS_FIRST) { /* * Since NULLs are sorted before non-NULLs, we know we have @@ -5247,7 +5292,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, * (_bt_advance_array_keys also relies on this behavior during * forward scans.) */ - if ((key->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD)) && + if ((requiredSameDir || requiredOppositeDirOnly) && ScanDirectionIsBackward(dir)) *continuescan = false; } @@ -5265,7 +5310,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, * (_bt_advance_array_keys also relies on this behavior during * backward scans.) */ - if ((key->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD)) && + if ((requiredSameDir || requiredOppositeDirOnly) && ScanDirectionIsForward(dir)) *continuescan = false; } @@ -5284,8 +5329,8 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, * an earlier tuple from this same page satisfied it earlier on. */ if (!(requiredOppositeDirOnly && firstmatch) && - !DatumGetBool(FunctionCall2Coll(&key->sk_func, key->sk_collation, - datum, key->sk_argument))) + !DatumGetBool(FunctionCall2Coll(&sk_func, sk_collation, + datum, sk_argument))) { /* * Tuple fails this qual. If it's a required qual for the current @@ -5308,7 +5353,7 @@ _bt_check_compare(IndexScanDesc scan, ScanDirection dir, */ else if (advancenonrequired && key->sk_strategy == BTEqualStrategyNumber && - (key->sk_flags & SK_SEARCHARRAY)) + (sk_flags & SK_SEARCHARRAY)) return _bt_advance_array_keys(scan, NULL, tuple, tupnatts, tupdesc, *ikey, false); @@ -5500,6 +5545,100 @@ _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, int tupnatts, return result; } +/* + * Can _bt_checkkeys/_bt_check_compare apply the 'skipskip' optimization? + * + * Called when _bt_advance_array_keys finds that no required scan key is + * satisfied during a scan with at least one skip array. + * + * Return value indicates if the optimization is safe for the tuples on the + * page after caller's tuple, but before its page's finaltup. + */ +static bool +_bt_checkkeys_skipskip(IndexScanDesc scan, IndexTuple tuple, + IndexTuple finaltup, TupleDesc tupdesc) +{ + BTScanOpaque so = (BTScanOpaque) scan->opaque; + Relation rel = scan->indexRelation; + ScanDirection dir = so->currPos.dir; + int arrayidx = 0, + nfinaltupatts = 0; + bool rangearrayseen = false; + + Assert(!BTreeTupleIsPivot(tuple)); + + if (finaltup) + nfinaltupatts = BTreeTupleGetNAtts(finaltup, rel); + for (int ikey = 0; ikey < so->numberOfKeys; ikey++) + { + ScanKey cur = so->keyData + ikey; + BTArrayKeyInfo *array = NULL; + Datum tupdatum; + bool tupnull; + int32 result; + + /* + * Only need to check range skip arrays within this loop. + * + * A SAOP array can always be treated as a non-required array within + * _bt_check_compare. A skip array without a lower or upper bound is + * always safe to skip within _bt_check_compare, since it is satisfied + * by every possible value. + */ + if (cur->sk_strategy != BTEqualStrategyNumber) + continue; + if (!(cur->sk_flags & SK_SEARCHARRAY)) + continue; + array = &so->arrayKeys[arrayidx++]; + Assert(array->scan_key == ikey); + if (array->num_elems != -1 || array->null_elem) + continue; + + /* + * Found a range skip array to test. + * + * Scans with more than one range skip array are not eligible to use + * the optimization. Note that we support the skipskip optimization + * for a qual like "WHERE a BETWEEN 1 AND 10 AND b BETWEEN 1 AND 3", + * since there the qual actually requires only a single skip array. + * However, if such a qual ended with "... AND C > 42", then it will + * prevent use of the skipskip optimization. + */ + if (rangearrayseen) + return false; + + /* + * Don't attempt the optimization when we have a skip array and are + * reading the rightmost leaf page (or the leftmost leaf page, when + * scanning backwards) + */ + if (!finaltup) + return false; + rangearrayseen = true; + + /* test the tuple that just advanced arrays within our caller */ + Assert(cur->sk_flags & SK_BT_SKIP); + Assert(cur->sk_flags & SK_BT_REQFWD); + tupdatum = index_getattr(tuple, cur->sk_attno, tupdesc, &tupnull); + _bt_binsrch_skiparray_skey(&so->orderProcs[ikey], false, dir, + tupdatum, tupnull, array, cur, &result); + if (result != 0) + return false; + + /* test the page's finaltup iff relevant attribute isn't truncated */ + if (cur->sk_attno > nfinaltupatts) + continue; + + tupdatum = index_getattr(finaltup, cur->sk_attno, tupdesc, &tupnull); + _bt_binsrch_skiparray_skey(&so->orderProcs[ikey], false, dir, + tupdatum, tupnull, array, cur, &result); + if (result != 0) + return false; + } + + return true; +} + /* * Determine if a scan with array keys should skip over uninteresting tuples. * @@ -5525,6 +5664,13 @@ _bt_checkkeys_look_ahead(IndexScanDesc scan, BTReadPageState *pstate, OffsetNumber aheadoffnum; IndexTuple ahead; + /* + * The "look ahead" skipping mechanism cannot be used at the same time as + * skip scan's similar "skipskip" mechanism (though it can be used during + * skip scans when it's not in use) + */ + Assert(!pstate->skipskip); + /* Avoid looking ahead when comparing the page high key */ if (pstate->offnum < pstate->minoff) return; -- 2.45.2