Commit 6980f817 authored by Tom Lane's avatar Tom Lane

Stop btree indexscans upon reaching nulls in either direction.

The existing scan-direction-sensitive tests were overly complex, and
failed to stop the scan in cases where it's perfectly legitimate to do so.
Per bug #6278 from Maksym Boguk.

Back-patch to 8.3, which is as far back as the patch applies easily.
Doesn't seem worth sweating over a relatively minor performance issue in
8.2 at this late date.  (But note that this was a performance regression
from 8.1 and before, so 8.2 is being left as an outlier.)
parent 6743a878
...@@ -608,11 +608,11 @@ _bt_advance_array_keys(IndexScanDesc scan, ScanDirection dir) ...@@ -608,11 +608,11 @@ _bt_advance_array_keys(IndexScanDesc scan, ScanDirection dir)
* Also, for a DESC column, we commute (flip) all the sk_strategy numbers * Also, for a DESC column, we commute (flip) all the sk_strategy numbers
* so that the index sorts in the desired direction. * so that the index sorts in the desired direction.
* *
* One key purpose of this routine is to discover how many scan keys * One key purpose of this routine is to discover which scan keys must be
* must be satisfied to continue the scan. It also attempts to eliminate * satisfied to continue the scan. It also attempts to eliminate redundant
* redundant keys and detect contradictory keys. (If the index opfamily * keys and detect contradictory keys. (If the index opfamily provides
* provides incomplete sets of cross-type operators, we may fail to detect * incomplete sets of cross-type operators, we may fail to detect redundant
* redundant or contradictory keys, but we can survive that.) * or contradictory keys, but we can survive that.)
* *
* The output keys must be sorted by index attribute. Presently we expect * The output keys must be sorted by index attribute. Presently we expect
* (but verify) that the input keys are already so sorted --- this is done * (but verify) that the input keys are already so sorted --- this is done
...@@ -647,6 +647,16 @@ _bt_advance_array_keys(IndexScanDesc scan, ScanDirection dir) ...@@ -647,6 +647,16 @@ _bt_advance_array_keys(IndexScanDesc scan, ScanDirection dir)
* </<= keys if we can't compare them. The logic about required keys still * </<= keys if we can't compare them. The logic about required keys still
* works if we don't eliminate redundant keys. * works if we don't eliminate redundant keys.
* *
* Note that the reason we need direction-sensitive required-key flags is
* precisely that we may not be able to eliminate redundant keys. Suppose
* we have "x > 4::int AND x > 10::bigint", and we are unable to determine
* which key is more restrictive for lack of a suitable cross-type operator.
* _bt_first will arbitrarily pick one of the keys to do the initial
* positioning with. If it picks x > 4, then the x > 10 condition will fail
* until we reach index entries > 10; but we can't stop the scan just because
* x > 10 is failing. On the other hand, if we are scanning backwards, then
* failure of either key is indeed enough to stop the scan.
*
* As a byproduct of this work, we can detect contradictory quals such * As a byproduct of this work, we can detect contradictory quals such
* as "x = 1 AND x > 2". If we see that, we return so->qual_ok = FALSE, * as "x = 1 AND x > 2". If we see that, we return so->qual_ok = FALSE,
* indicating the scan need not be run at all since no tuples can match. * indicating the scan need not be run at all since no tuples can match.
...@@ -1384,15 +1394,16 @@ _bt_checkkeys(IndexScanDesc scan, ...@@ -1384,15 +1394,16 @@ _bt_checkkeys(IndexScanDesc scan,
} }
/* /*
* Tuple fails this qual. If it's a required qual for the current * Tuple fails this qual. If it's a required qual, then we can
* scan direction, then we can conclude no further tuples will * conclude no further tuples will pass, either. We can stop
* pass, either. * regardless of the scan direction, because we know that NULLs
*/ * sort to one end or the other of the range of values. If this
if ((key->sk_flags & SK_BT_REQFWD) && * tuple doesn't pass, then no future ones will either, until we
ScanDirectionIsForward(dir)) * reach the next set of values of the higher-order index attrs
*continuescan = false; * (if any) ... and those attrs must have equality quals, else
else if ((key->sk_flags & SK_BT_REQBKWD) && * this one wouldn't be marked required.
ScanDirectionIsBackward(dir)) */
if (key->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD))
*continuescan = false; *continuescan = false;
/* /*
...@@ -1402,33 +1413,16 @@ _bt_checkkeys(IndexScanDesc scan, ...@@ -1402,33 +1413,16 @@ _bt_checkkeys(IndexScanDesc scan,
} }
if (isNull) if (isNull)
{
if (key->sk_flags & SK_BT_NULLS_FIRST)
{
/*
* Since NULLs are sorted before non-NULLs, we know we have
* reached the lower limit of the range of values for this
* index attr. On a backward scan, we can stop if this qual
* is one of the "must match" subset. On a forward scan,
* however, we should keep going.
*/
if ((key->sk_flags & SK_BT_REQBKWD) &&
ScanDirectionIsBackward(dir))
*continuescan = false;
}
else
{ {
/* /*
* Since NULLs are sorted after non-NULLs, we know we have * The index entry is NULL, so it must fail this qual (we assume
* reached the upper limit of the range of values for this * all btree operators are strict). Furthermore, we know that
* index attr. On a forward scan, we can stop if this qual is * all remaining entries with the same higher-order index attr
* one of the "must match" subset. On a backward scan, * values must be NULLs too. So, just as above, we can stop the
* however, we should keep going. * scan regardless of direction, if the qual is required.
*/ */
if ((key->sk_flags & SK_BT_REQFWD) && if (key->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD))
ScanDirectionIsForward(dir))
*continuescan = false; *continuescan = false;
}
/* /*
* In any case, this indextuple doesn't match the qual. * In any case, this indextuple doesn't match the qual.
...@@ -1507,33 +1501,16 @@ _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, TupleDesc tupdesc, ...@@ -1507,33 +1501,16 @@ _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, TupleDesc tupdesc,
&isNull); &isNull);
if (isNull) if (isNull)
{
if (subkey->sk_flags & SK_BT_NULLS_FIRST)
{
/*
* Since NULLs are sorted before non-NULLs, we know we have
* reached the lower limit of the range of values for this
* index attr. On a backward scan, we can stop if this qual is
* one of the "must match" subset. On a forward scan,
* however, we should keep going.
*/
if ((subkey->sk_flags & SK_BT_REQBKWD) &&
ScanDirectionIsBackward(dir))
*continuescan = false;
}
else
{ {
/* /*
* Since NULLs are sorted after non-NULLs, we know we have * The index entry is NULL, so it must fail this qual (we assume
* reached the upper limit of the range of values for this * all btree operators are strict). Furthermore, we know that
* index attr. On a forward scan, we can stop if this qual is * all remaining entries with the same higher-order index attr
* one of the "must match" subset. On a backward scan, * values must be NULLs too. So, just as above, we can stop the
* however, we should keep going. * scan regardless of direction, if the qual is required.
*/ */
if ((subkey->sk_flags & SK_BT_REQFWD) && if (subkey->sk_flags & (SK_BT_REQFWD | SK_BT_REQBKWD))
ScanDirectionIsForward(dir))
*continuescan = false; *continuescan = false;
}
/* /*
* In any case, this indextuple doesn't match the qual. * In any case, this indextuple doesn't match the qual.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment