Commit 322e82b7 authored by Alexander Korotkov's avatar Alexander Korotkov

Fixes for multirange selectivity estimation

 * Fix enumeration of the multirange operators in calc_multirangesel() and
   calc_multirangesel() switches.
 * Add more regression tests for matching to empty ranges/multiranges.

Reported-by: Alexander Lakhin
Discussion: https://postgr.es/m/c5269c65-f967-77c5-ff7c-15e621c47f6a%40gmail.com
Author: Alexander Korotkov
Backpatch-through: 14, where multiranges were introduced
parent 690339fc
...@@ -347,16 +347,15 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -347,16 +347,15 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata,
switch (operator) switch (operator)
{ {
/* these return false if either argument is empty */ /* these return false if either argument is empty */
case OID_RANGE_OVERLAPS_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_LEFT_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_LEFT_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_LEFT_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RIGHT_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_RIGHT_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RIGHT_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
case OID_MULTIRANGE_LEFT_RANGE_OP:
case OID_MULTIRANGE_LEFT_MULTIRANGE_OP: case OID_MULTIRANGE_LEFT_MULTIRANGE_OP:
case OID_MULTIRANGE_RIGHT_RANGE_OP:
case OID_MULTIRANGE_RIGHT_MULTIRANGE_OP: case OID_MULTIRANGE_RIGHT_MULTIRANGE_OP:
/* nothing is less than an empty multirange */ /* nothing is less than an empty multirange */
case OID_MULTIRANGE_LESS_OP: case OID_MULTIRANGE_LESS_OP:
...@@ -367,7 +366,7 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -367,7 +366,7 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata,
* only empty multiranges can be contained by an empty * only empty multiranges can be contained by an empty
* multirange * multirange
*/ */
case OID_MULTIRANGE_RANGE_CONTAINED_OP: case OID_RANGE_MULTIRANGE_CONTAINED_OP:
case OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP: case OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP:
/* only empty ranges are <= an empty multirange */ /* only empty ranges are <= an empty multirange */
case OID_MULTIRANGE_LESS_EQUAL_OP: case OID_MULTIRANGE_LESS_EQUAL_OP:
...@@ -388,8 +387,18 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -388,8 +387,18 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata,
break; break;
/* an element cannot be empty */ /* an element cannot be empty */
case OID_MULTIRANGE_ELEM_CONTAINED_OP:
case OID_MULTIRANGE_CONTAINS_ELEM_OP: case OID_MULTIRANGE_CONTAINS_ELEM_OP:
/* filtered out by multirangesel() */
case OID_RANGE_OVERLAPS_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
case OID_RANGE_LEFT_MULTIRANGE_OP:
case OID_RANGE_RIGHT_MULTIRANGE_OP:
case OID_RANGE_CONTAINS_MULTIRANGE_OP:
case OID_MULTIRANGE_ELEM_CONTAINED_OP:
case OID_MULTIRANGE_RANGE_CONTAINED_OP:
default: default:
elog(ERROR, "unexpected operator %u", operator); elog(ERROR, "unexpected operator %u", operator);
selec = 0.0; /* keep compiler quiet */ selec = 0.0; /* keep compiler quiet */
...@@ -416,8 +425,7 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -416,8 +425,7 @@ calc_multirangesel(TypeCacheEntry *typcache, VariableStatData *vardata,
* calculations, realizing that the histogram covers only the * calculations, realizing that the histogram covers only the
* non-null, non-empty values. * non-null, non-empty values.
*/ */
if (operator == OID_MULTIRANGE_ELEM_CONTAINED_OP || if (operator == OID_RANGE_MULTIRANGE_CONTAINED_OP ||
operator == OID_MULTIRANGE_RANGE_CONTAINED_OP ||
operator == OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP) operator == OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP)
{ {
/* empty is contained by anything non-empty */ /* empty is contained by anything non-empty */
...@@ -575,7 +583,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -575,7 +583,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
hist_lower, nhist, true); hist_lower, nhist, true);
break; break;
case OID_RANGE_LEFT_MULTIRANGE_OP:
case OID_MULTIRANGE_LEFT_RANGE_OP: case OID_MULTIRANGE_LEFT_RANGE_OP:
case OID_MULTIRANGE_LEFT_MULTIRANGE_OP: case OID_MULTIRANGE_LEFT_MULTIRANGE_OP:
/* var << const when upper(var) < lower(const) */ /* var << const when upper(var) < lower(const) */
...@@ -584,7 +591,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -584,7 +591,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
hist_upper, nhist, false); hist_upper, nhist, false);
break; break;
case OID_RANGE_RIGHT_MULTIRANGE_OP:
case OID_MULTIRANGE_RIGHT_RANGE_OP: case OID_MULTIRANGE_RIGHT_RANGE_OP:
case OID_MULTIRANGE_RIGHT_MULTIRANGE_OP: case OID_MULTIRANGE_RIGHT_MULTIRANGE_OP:
/* var >> const when lower(var) > upper(const) */ /* var >> const when lower(var) > upper(const) */
...@@ -593,7 +599,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -593,7 +599,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
hist_lower, nhist, true); hist_lower, nhist, true);
break; break;
case OID_RANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RIGHT_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_RIGHT_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RIGHT_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
/* compare lower bounds */ /* compare lower bounds */
...@@ -602,7 +607,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -602,7 +607,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
hist_lower, nhist, false); hist_lower, nhist, false);
break; break;
case OID_RANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_LEFT_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_LEFT_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_LEFT_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
/* compare upper bounds */ /* compare upper bounds */
...@@ -611,7 +615,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -611,7 +615,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
hist_upper, nhist, true); hist_upper, nhist, true);
break; break;
case OID_RANGE_OVERLAPS_MULTIRANGE_OP:
case OID_MULTIRANGE_OVERLAPS_RANGE_OP: case OID_MULTIRANGE_OVERLAPS_RANGE_OP:
case OID_MULTIRANGE_OVERLAPS_MULTIRANGE_OP: case OID_MULTIRANGE_OVERLAPS_MULTIRANGE_OP:
case OID_MULTIRANGE_CONTAINS_ELEM_OP: case OID_MULTIRANGE_CONTAINS_ELEM_OP:
...@@ -647,7 +650,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -647,7 +650,6 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
lslot.values, lslot.nvalues); lslot.values, lslot.nvalues);
break; break;
case OID_MULTIRANGE_RANGE_CONTAINED_OP:
case OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP: case OID_MULTIRANGE_MULTIRANGE_CONTAINED_OP:
case OID_RANGE_MULTIRANGE_CONTAINED_OP: case OID_RANGE_MULTIRANGE_CONTAINED_OP:
if (const_lower.infinite) if (const_lower.infinite)
...@@ -675,6 +677,16 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata, ...@@ -675,6 +677,16 @@ calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
} }
break; break;
/* filtered out by multirangesel() */
case OID_RANGE_OVERLAPS_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_LEFT_MULTIRANGE_OP:
case OID_RANGE_OVERLAPS_RIGHT_MULTIRANGE_OP:
case OID_RANGE_LEFT_MULTIRANGE_OP:
case OID_RANGE_RIGHT_MULTIRANGE_OP:
case OID_RANGE_CONTAINS_MULTIRANGE_OP:
case OID_MULTIRANGE_ELEM_CONTAINED_OP:
case OID_MULTIRANGE_RANGE_CONTAINED_OP:
default: default:
elog(ERROR, "unknown multirange operator %u", operator); elog(ERROR, "unknown multirange operator %u", operator);
hist_selec = -1.0; /* keep compiler quiet */ hist_selec = -1.0; /* keep compiler quiet */
......
...@@ -2241,12 +2241,114 @@ analyze test_multirange_gist; ...@@ -2241,12 +2241,114 @@ analyze test_multirange_gist;
SET enable_seqscan = t; SET enable_seqscan = t;
SET enable_indexscan = f; SET enable_indexscan = f;
SET enable_bitmapscan = f; SET enable_bitmapscan = f;
select count(*) from test_multirange_gist where mr = '{}'::int4multirange;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr @> 'empty'::int4range; select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
count count
------- -------
3700 3700
(1 row) (1 row)
select count(*) from test_multirange_gist where mr && 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr <@ 'empty'::int4range;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr << 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr >> 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &< 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &> 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr -|- 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
count
-------
3700
(1 row)
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
count
-------
3700
(1 row)
select count(*) from test_multirange_gist where mr && '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr <@ '{}'::int4multirange;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr << '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr >> '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &< '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &> '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr -|- '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60)); select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60));
count count
------- -------
...@@ -2365,6 +2467,114 @@ select count(*) from test_multirange_gist where mr -|- int4multirange(int4range( ...@@ -2365,6 +2467,114 @@ select count(*) from test_multirange_gist where mr -|- int4multirange(int4range(
SET enable_seqscan = f; SET enable_seqscan = f;
SET enable_indexscan = t; SET enable_indexscan = t;
SET enable_bitmapscan = f; SET enable_bitmapscan = f;
select count(*) from test_multirange_gist where mr = '{}'::int4multirange;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
count
-------
3700
(1 row)
select count(*) from test_multirange_gist where mr && 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr <@ 'empty'::int4range;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr << 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr >> 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &< 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &> 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr -|- 'empty'::int4range;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
count
-------
3700
(1 row)
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
count
-------
3700
(1 row)
select count(*) from test_multirange_gist where mr && '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr <@ '{}'::int4multirange;
count
-------
500
(1 row)
select count(*) from test_multirange_gist where mr << '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr >> '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &< '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr &> '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr -|- '{}'::int4multirange;
count
-------
0
(1 row)
select count(*) from test_multirange_gist where mr @> 'empty'::int4range; select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
count count
------- -------
......
...@@ -435,7 +435,25 @@ SET enable_seqscan = t; ...@@ -435,7 +435,25 @@ SET enable_seqscan = t;
SET enable_indexscan = f; SET enable_indexscan = f;
SET enable_bitmapscan = f; SET enable_bitmapscan = f;
select count(*) from test_multirange_gist where mr = '{}'::int4multirange;
select count(*) from test_multirange_gist where mr @> 'empty'::int4range; select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
select count(*) from test_multirange_gist where mr && 'empty'::int4range;
select count(*) from test_multirange_gist where mr <@ 'empty'::int4range;
select count(*) from test_multirange_gist where mr << 'empty'::int4range;
select count(*) from test_multirange_gist where mr >> 'empty'::int4range;
select count(*) from test_multirange_gist where mr &< 'empty'::int4range;
select count(*) from test_multirange_gist where mr &> 'empty'::int4range;
select count(*) from test_multirange_gist where mr -|- 'empty'::int4range;
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr && '{}'::int4multirange;
select count(*) from test_multirange_gist where mr <@ '{}'::int4multirange;
select count(*) from test_multirange_gist where mr << '{}'::int4multirange;
select count(*) from test_multirange_gist where mr >> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr &< '{}'::int4multirange;
select count(*) from test_multirange_gist where mr &> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr -|- '{}'::int4multirange;
select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60)); select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60));
select count(*) from test_multirange_gist where mr @> 10; select count(*) from test_multirange_gist where mr @> 10;
select count(*) from test_multirange_gist where mr @> int4range(10,20); select count(*) from test_multirange_gist where mr @> int4range(10,20);
...@@ -461,6 +479,25 @@ SET enable_seqscan = f; ...@@ -461,6 +479,25 @@ SET enable_seqscan = f;
SET enable_indexscan = t; SET enable_indexscan = t;
SET enable_bitmapscan = f; SET enable_bitmapscan = f;
select count(*) from test_multirange_gist where mr = '{}'::int4multirange;
select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
select count(*) from test_multirange_gist where mr && 'empty'::int4range;
select count(*) from test_multirange_gist where mr <@ 'empty'::int4range;
select count(*) from test_multirange_gist where mr << 'empty'::int4range;
select count(*) from test_multirange_gist where mr >> 'empty'::int4range;
select count(*) from test_multirange_gist where mr &< 'empty'::int4range;
select count(*) from test_multirange_gist where mr &> 'empty'::int4range;
select count(*) from test_multirange_gist where mr -|- 'empty'::int4range;
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr @> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr && '{}'::int4multirange;
select count(*) from test_multirange_gist where mr <@ '{}'::int4multirange;
select count(*) from test_multirange_gist where mr << '{}'::int4multirange;
select count(*) from test_multirange_gist where mr >> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr &< '{}'::int4multirange;
select count(*) from test_multirange_gist where mr &> '{}'::int4multirange;
select count(*) from test_multirange_gist where mr -|- '{}'::int4multirange;
select count(*) from test_multirange_gist where mr @> 'empty'::int4range; select count(*) from test_multirange_gist where mr @> 'empty'::int4range;
select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60)); select count(*) from test_multirange_gist where mr = int4multirange(int4range(10,20), int4range(30,40), int4range(50,60));
select count(*) from test_multirange_gist where mr @> 10; select count(*) from test_multirange_gist where mr @> 10;
......
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