Commit f9aefcb9 authored by Tom Lane's avatar Tom Lane

Support using index-only scans with partial indexes in more cases.

Previously, the planner would reject an index-only scan if any restriction
clause for its table used a column not available from the index, even
if that restriction clause would later be dropped from the plan entirely
because it's implied by the index's predicate.  This is a fairly common
situation for partial indexes because predicates using columns not included
in the index are often the most useful kind of predicate, and we have to
duplicate (or at least imply) the predicate in the WHERE clause in order
to get the index to be considered at all.  So index-only scans were
essentially unavailable with such partial indexes.

To fix, we have to do detection of implied-by-predicate clauses much
earlier in the planner.  This patch puts it in check_index_predicates
(nee check_partial_indexes), meaning it gets done for every partial index,
whereas we previously only considered this issue at createplan time,
so that the work was only done for an index actually selected for use.
That could result in a noticeable planning slowdown for queries against
tables with many partial indexes.  However, testing suggested that there
isn't really a significant cost, especially not with reasonable numbers
of partial indexes.  We do get a small additional benefit, which is that
cost_index is more accurate since it correctly discounts the evaluation
cost of clauses that will be removed.  We can also avoid considering such
clauses as potential indexquals, which saves useless matching cycles in
the case where the predicate columns aren't in the index, and prevents
generating bogus plans that double-count the clause's selectivity when
the columns are in the index.

Tomas Vondra and Kyotaro Horiguchi, reviewed by Kevin Grittner and
Konstantin Knizhnik, and whacked around a little by me
parent 3501f71c
...@@ -2129,6 +2129,7 @@ _outIndexOptInfo(StringInfo str, const IndexOptInfo *node) ...@@ -2129,6 +2129,7 @@ _outIndexOptInfo(StringInfo str, const IndexOptInfo *node)
/* indexprs is redundant since we print indextlist */ /* indexprs is redundant since we print indextlist */
WRITE_NODE_FIELD(indpred); WRITE_NODE_FIELD(indpred);
WRITE_NODE_FIELD(indextlist); WRITE_NODE_FIELD(indextlist);
WRITE_NODE_FIELD(indrestrictinfo);
WRITE_BOOL_FIELD(predOK); WRITE_BOOL_FIELD(predOK);
WRITE_BOOL_FIELD(unique); WRITE_BOOL_FIELD(unique);
WRITE_BOOL_FIELD(immediate); WRITE_BOOL_FIELD(immediate);
......
...@@ -474,7 +474,7 @@ set_plain_rel_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte) ...@@ -474,7 +474,7 @@ set_plain_rel_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte)
* Test any partial indexes of rel for applicability. We must do this * Test any partial indexes of rel for applicability. We must do this
* first since partial unique indexes can affect size estimates. * first since partial unique indexes can affect size estimates.
*/ */
check_partial_indexes(root, rel); check_index_predicates(root, rel);
/* Mark rel with estimated output rows, width, etc */ /* Mark rel with estimated output rows, width, etc */
set_baserel_size_estimates(root, rel); set_baserel_size_estimates(root, rel);
...@@ -716,7 +716,7 @@ set_tablesample_rel_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte) ...@@ -716,7 +716,7 @@ set_tablesample_rel_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte)
* Test any partial indexes of rel for applicability. We must do this * Test any partial indexes of rel for applicability. We must do this
* first since partial unique indexes can affect size estimates. * first since partial unique indexes can affect size estimates.
*/ */
check_partial_indexes(root, rel); check_index_predicates(root, rel);
/* /*
* Call the sampling method's estimation function to estimate the number * Call the sampling method's estimation function to estimate the number
......
...@@ -433,14 +433,17 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count) ...@@ -433,14 +433,17 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count)
/* /*
* Mark the path with the correct row estimate, and identify which quals * Mark the path with the correct row estimate, and identify which quals
* will need to be enforced as qpquals. * will need to be enforced as qpquals. We need not check any quals that
* are implied by the index's predicate, so we can use indrestrictinfo not
* baserestrictinfo as the list of relevant restriction clauses for the
* rel.
*/ */
if (path->path.param_info) if (path->path.param_info)
{ {
path->path.rows = path->path.param_info->ppi_rows; path->path.rows = path->path.param_info->ppi_rows;
/* qpquals come from the rel's restriction clauses and ppi_clauses */ /* qpquals come from the rel's restriction clauses and ppi_clauses */
qpquals = list_concat( qpquals = list_concat(
extract_nonindex_conditions(baserel->baserestrictinfo, extract_nonindex_conditions(path->indexinfo->indrestrictinfo,
path->indexquals), path->indexquals),
extract_nonindex_conditions(path->path.param_info->ppi_clauses, extract_nonindex_conditions(path->path.param_info->ppi_clauses,
path->indexquals)); path->indexquals));
...@@ -449,7 +452,7 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count) ...@@ -449,7 +452,7 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count)
{ {
path->path.rows = baserel->rows; path->path.rows = baserel->rows;
/* qpquals come from just the rel's restriction clauses */ /* qpquals come from just the rel's restriction clauses */
qpquals = extract_nonindex_conditions(baserel->baserestrictinfo, qpquals = extract_nonindex_conditions(path->indexinfo->indrestrictinfo,
path->indexquals); path->indexquals);
} }
...@@ -631,11 +634,11 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count) ...@@ -631,11 +634,11 @@ cost_index(IndexPath *path, PlannerInfo *root, double loop_count)
* final plan. So we approximate it as quals that don't appear directly in * final plan. So we approximate it as quals that don't appear directly in
* indexquals and also are not redundant children of the same EquivalenceClass * indexquals and also are not redundant children of the same EquivalenceClass
* as some indexqual. This method neglects some infrequently-relevant * as some indexqual. This method neglects some infrequently-relevant
* considerations such as clauses that needn't be checked because they are * considerations, specifically clauses that needn't be checked because they
* implied by a partial index's predicate. It does not seem worth the cycles * are implied by an indexqual. It does not seem worth the cycles to try to
* to try to factor those things in at this stage, even though createplan.c * factor that in at this stage, even though createplan.c will take pains to
* will take pains to remove such unnecessary clauses from the qpquals list if * remove such unnecessary clauses from the qpquals list if this path is
* this path is selected for use. * selected for use.
*/ */
static List * static List *
extract_nonindex_conditions(List *qual_clauses, List *indexquals) extract_nonindex_conditions(List *qual_clauses, List *indexquals)
...@@ -654,7 +657,7 @@ extract_nonindex_conditions(List *qual_clauses, List *indexquals) ...@@ -654,7 +657,7 @@ extract_nonindex_conditions(List *qual_clauses, List *indexquals)
continue; /* simple duplicate */ continue; /* simple duplicate */
if (is_redundant_derived_clause(rinfo, indexquals)) if (is_redundant_derived_clause(rinfo, indexquals))
continue; /* derived from same EquivalenceClass */ continue; /* derived from same EquivalenceClass */
/* ... skip the predicate proof attempts createplan.c will try ... */ /* ... skip the predicate proof attempt createplan.c will try ... */
result = lappend(result, rinfo); result = lappend(result, rinfo);
} }
return result; return result;
......
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
#include "optimizer/pathnode.h" #include "optimizer/pathnode.h"
#include "optimizer/paths.h" #include "optimizer/paths.h"
#include "optimizer/predtest.h" #include "optimizer/predtest.h"
#include "optimizer/prep.h"
#include "optimizer/restrictinfo.h" #include "optimizer/restrictinfo.h"
#include "optimizer/var.h" #include "optimizer/var.h"
#include "utils/builtins.h" #include "utils/builtins.h"
...@@ -216,7 +217,7 @@ static Const *string_to_const(const char *str, Oid datatype); ...@@ -216,7 +217,7 @@ static Const *string_to_const(const char *str, Oid datatype);
* *
* 'rel' is the relation for which we want to generate index paths * 'rel' is the relation for which we want to generate index paths
* *
* Note: check_partial_indexes() must have been run previously for this rel. * Note: check_index_predicates() must have been run previously for this rel.
* *
* Note: in cases involving LATERAL references in the relation's tlist, it's * Note: in cases involving LATERAL references in the relation's tlist, it's
* possible that rel->lateral_relids is nonempty. Currently, we include * possible that rel->lateral_relids is nonempty. Currently, we include
...@@ -1800,25 +1801,27 @@ check_index_only(RelOptInfo *rel, IndexOptInfo *index) ...@@ -1800,25 +1801,27 @@ check_index_only(RelOptInfo *rel, IndexOptInfo *index)
/* /*
* Check that all needed attributes of the relation are available from the * Check that all needed attributes of the relation are available from the
* index. * index.
*
* XXX this is overly conservative for partial indexes, since we will
* consider attributes involved in the index predicate as required even
* though the predicate won't need to be checked at runtime. (The same is
* true for attributes used only in index quals, if we are certain that
* the index is not lossy.) However, it would be quite expensive to
* determine that accurately at this point, so for now we take the easy
* way out.
*/ */
/* /*
* Add all the attributes needed for joins or final output. Note: we must * First, identify all the attributes needed for joins or final output.
* look at rel's targetlist, not the attr_needed data, because attr_needed * Note: we must look at rel's targetlist, not the attr_needed data,
* isn't computed for inheritance child rels. * because attr_needed isn't computed for inheritance child rels.
*/ */
pull_varattnos((Node *) rel->reltarget->exprs, rel->relid, &attrs_used); pull_varattnos((Node *) rel->reltarget->exprs, rel->relid, &attrs_used);
/* Add all the attributes used by restriction clauses. */ /*
foreach(lc, rel->baserestrictinfo) * Add all the attributes used by restriction clauses; but consider only
* those clauses not implied by the index predicate, since ones that are
* so implied don't need to be checked explicitly in the plan.
*
* Note: attributes used only in index quals would not be needed at
* runtime either, if we are certain that the index is not lossy. However
* it'd be complicated to account for that accurately, and it doesn't
* matter in most cases, since we'd conclude that such attributes are
* available from the index anyway.
*/
foreach(lc, index->indrestrictinfo)
{ {
RestrictInfo *rinfo = (RestrictInfo *) lfirst(lc); RestrictInfo *rinfo = (RestrictInfo *) lfirst(lc);
...@@ -2023,7 +2026,8 @@ static void ...@@ -2023,7 +2026,8 @@ static void
match_restriction_clauses_to_index(RelOptInfo *rel, IndexOptInfo *index, match_restriction_clauses_to_index(RelOptInfo *rel, IndexOptInfo *index,
IndexClauseSet *clauseset) IndexClauseSet *clauseset)
{ {
match_clauses_to_index(index, rel->baserestrictinfo, clauseset); /* We can ignore clauses that are implied by the index predicate */
match_clauses_to_index(index, index->indrestrictinfo, clauseset);
} }
/* /*
...@@ -2664,39 +2668,48 @@ match_clause_to_ordering_op(IndexOptInfo *index, ...@@ -2664,39 +2668,48 @@ match_clause_to_ordering_op(IndexOptInfo *index,
****************************************************************************/ ****************************************************************************/
/* /*
* check_partial_indexes * check_index_predicates
* Check each partial index of the relation, and mark it predOK if * Set the predicate-derived IndexOptInfo fields for each index
* the index's predicate is satisfied for this query. * of the specified relation.
*
* predOK is set true if the index is partial and its predicate is satisfied
* for this query, ie the query's WHERE clauses imply the predicate.
*
* indrestrictinfo is set to the relation's baserestrictinfo list less any
* conditions that are implied by the index's predicate. (Obviously, for a
* non-partial index, this is the same as baserestrictinfo.) Such conditions
* can be dropped from the plan when using the index, in certain cases.
* *
* Note: it is possible for this to get re-run after adding more restrictions * At one time it was possible for this to get re-run after adding more
* to the rel; so we might be able to prove more indexes OK. We assume that * restrictions to the rel, thus possibly letting us prove more indexes OK.
* adding more restrictions can't make an index not OK. * That doesn't happen any more (at least not in the core code's usage),
* but this code still supports it in case extensions want to mess with the
* baserestrictinfo list. We assume that adding more restrictions can't make
* an index not predOK. We must recompute indrestrictinfo each time, though,
* to make sure any newly-added restrictions get into it if needed.
*/ */
void void
check_partial_indexes(PlannerInfo *root, RelOptInfo *rel) check_index_predicates(PlannerInfo *root, RelOptInfo *rel)
{ {
List *clauselist; List *clauselist;
bool have_partial; bool have_partial;
bool is_target_rel;
Relids otherrels; Relids otherrels;
ListCell *lc; ListCell *lc;
/* /*
* Frequently, there will be no partial indexes, so first check to make * Initialize the indrestrictinfo lists to be identical to
* sure there's something useful to do here. * baserestrictinfo, and check whether there are any partial indexes. If
* not, this is all we need to do.
*/ */
have_partial = false; have_partial = false;
foreach(lc, rel->indexlist) foreach(lc, rel->indexlist)
{ {
IndexOptInfo *index = (IndexOptInfo *) lfirst(lc); IndexOptInfo *index = (IndexOptInfo *) lfirst(lc);
if (index->indpred == NIL) index->indrestrictinfo = rel->baserestrictinfo;
continue; /* ignore non-partial indexes */ if (index->indpred)
if (index->predOK)
continue; /* don't repeat work if already proven OK */
have_partial = true; have_partial = true;
break;
} }
if (!have_partial) if (!have_partial)
return; return;
...@@ -2743,18 +2756,54 @@ check_partial_indexes(PlannerInfo *root, RelOptInfo *rel) ...@@ -2743,18 +2756,54 @@ check_partial_indexes(PlannerInfo *root, RelOptInfo *rel)
otherrels, otherrels,
rel)); rel));
/* Now try to prove each index predicate true */ /*
* Normally we remove quals that are implied by a partial index's
* predicate from indrestrictinfo, indicating that they need not be
* checked explicitly by an indexscan plan using this index. However, if
* the rel is a target relation of UPDATE/DELETE/SELECT FOR UPDATE, we
* cannot remove such quals from the plan, because they need to be in the
* plan so that they will be properly rechecked by EvalPlanQual testing.
* Some day we might want to remove such quals from the main plan anyway
* and pass them through to EvalPlanQual via a side channel; but for now,
* we just don't remove implied quals at all for target relations.
*/
is_target_rel = (rel->relid == root->parse->resultRelation ||
get_plan_rowmark(root->rowMarks, rel->relid) != NULL);
/*
* Now try to prove each index predicate true, and compute the
* indrestrictinfo lists for partial indexes. Note that we compute the
* indrestrictinfo list even for non-predOK indexes; this might seem
* wasteful, but we may be able to use such indexes in OR clauses, cf
* generate_bitmap_or_paths().
*/
foreach(lc, rel->indexlist) foreach(lc, rel->indexlist)
{ {
IndexOptInfo *index = (IndexOptInfo *) lfirst(lc); IndexOptInfo *index = (IndexOptInfo *) lfirst(lc);
ListCell *lcr;
if (index->indpred == NIL) if (index->indpred == NIL)
continue; /* ignore non-partial indexes */ continue; /* ignore non-partial indexes here */
if (index->predOK)
continue; /* don't repeat work if already proven OK */
if (!index->predOK) /* don't repeat work if already proven OK */
index->predOK = predicate_implied_by(index->indpred, clauselist); index->predOK = predicate_implied_by(index->indpred, clauselist);
/* If rel is an update target, leave indrestrictinfo as set above */
if (is_target_rel)
continue;
/* Else compute indrestrictinfo as the non-implied quals */
index->indrestrictinfo = NIL;
foreach(lcr, rel->baserestrictinfo)
{
RestrictInfo *rinfo = (RestrictInfo *) lfirst(lcr);
/* predicate_implied_by() assumes first arg is immutable */
if (contain_mutable_functions((Node *) rinfo->clause) ||
!predicate_implied_by(list_make1(rinfo->clause),
index->indpred))
index->indrestrictinfo = lappend(index->indrestrictinfo, rinfo);
}
} }
} }
......
...@@ -35,7 +35,6 @@ ...@@ -35,7 +35,6 @@
#include "optimizer/planmain.h" #include "optimizer/planmain.h"
#include "optimizer/planner.h" #include "optimizer/planner.h"
#include "optimizer/predtest.h" #include "optimizer/predtest.h"
#include "optimizer/prep.h"
#include "optimizer/restrictinfo.h" #include "optimizer/restrictinfo.h"
#include "optimizer/subselect.h" #include "optimizer/subselect.h"
#include "optimizer/tlist.h" #include "optimizer/tlist.h"
...@@ -494,8 +493,25 @@ create_scan_plan(PlannerInfo *root, Path *best_path, int flags) ...@@ -494,8 +493,25 @@ create_scan_plan(PlannerInfo *root, Path *best_path, int flags)
* Extract the relevant restriction clauses from the parent relation. The * Extract the relevant restriction clauses from the parent relation. The
* executor must apply all these restrictions during the scan, except for * executor must apply all these restrictions during the scan, except for
* pseudoconstants which we'll take care of below. * pseudoconstants which we'll take care of below.
*
* If this is a plain indexscan or index-only scan, we need not consider
* restriction clauses that are implied by the index's predicate, so use
* indrestrictinfo not baserestrictinfo. Note that we can't do that for
* bitmap indexscans, since there's not necessarily a single index
* involved; but it doesn't matter since create_bitmap_scan_plan() will be
* able to get rid of such clauses anyway via predicate proof.
*/ */
switch (best_path->pathtype)
{
case T_IndexScan:
case T_IndexOnlyScan:
Assert(IsA(best_path, IndexPath));
scan_clauses = ((IndexPath *) best_path)->indexinfo->indrestrictinfo;
break;
default:
scan_clauses = rel->baserestrictinfo; scan_clauses = rel->baserestrictinfo;
break;
}
/* /*
* If this is a parameterized scan, we also need to enforce all the join * If this is a parameterized scan, we also need to enforce all the join
...@@ -2385,11 +2401,6 @@ create_indexscan_plan(PlannerInfo *root, ...@@ -2385,11 +2401,6 @@ create_indexscan_plan(PlannerInfo *root,
* first input contains only immutable functions, so we have to check * first input contains only immutable functions, so we have to check
* that.) * that.)
* *
* We can also discard quals that are implied by a partial index's
* predicate, but only in a plain SELECT; when scanning a target relation
* of UPDATE/DELETE/SELECT FOR UPDATE, we must leave such quals in the
* plan so that they'll be properly rechecked by EvalPlanQual testing.
*
* Note: if you change this bit of code you should also look at * Note: if you change this bit of code you should also look at
* extract_nonindex_conditions() in costsize.c. * extract_nonindex_conditions() in costsize.c.
*/ */
...@@ -2405,21 +2416,9 @@ create_indexscan_plan(PlannerInfo *root, ...@@ -2405,21 +2416,9 @@ create_indexscan_plan(PlannerInfo *root,
continue; /* simple duplicate */ continue; /* simple duplicate */
if (is_redundant_derived_clause(rinfo, indexquals)) if (is_redundant_derived_clause(rinfo, indexquals))
continue; /* derived from same EquivalenceClass */ continue; /* derived from same EquivalenceClass */
if (!contain_mutable_functions((Node *) rinfo->clause)) if (!contain_mutable_functions((Node *) rinfo->clause) &&
{ predicate_implied_by(list_make1(rinfo->clause), indexquals))
List *clausel = list_make1(rinfo->clause);
if (predicate_implied_by(clausel, indexquals))
continue; /* provably implied by indexquals */ continue; /* provably implied by indexquals */
if (best_path->indexinfo->indpred)
{
if (baserelid != root->parse->resultRelation &&
get_plan_rowmark(root->rowMarks, baserelid) == NULL)
if (predicate_implied_by(clausel,
best_path->indexinfo->indpred))
continue; /* implied by index predicate */
}
}
qpqual = lappend(qpqual, rinfo); qpqual = lappend(qpqual, rinfo);
} }
...@@ -2556,11 +2555,12 @@ create_bitmap_scan_plan(PlannerInfo *root, ...@@ -2556,11 +2555,12 @@ create_bitmap_scan_plan(PlannerInfo *root,
* redundant with any top-level indexqual by virtue of being generated * redundant with any top-level indexqual by virtue of being generated
* from the same EC. After that, try predicate_implied_by(). * from the same EC. After that, try predicate_implied_by().
* *
* Unlike create_indexscan_plan(), we need take no special thought here * Unlike create_indexscan_plan(), the predicate_implied_by() test here is
* for partial index predicates; this is because the predicate conditions * useful for getting rid of qpquals that are implied by index predicates,
* are already listed in bitmapqualorig and indexquals. Bitmap scans have * because the predicate conditions are included in the "indexquals"
* to do it that way because predicate conditions need to be rechecked if * returned by create_bitmap_subplan(). Bitmap scans have to do it that
* the scan becomes lossy, so they have to be included in bitmapqualorig. * way because predicate conditions need to be rechecked if the scan
* becomes lossy, so they have to be included in bitmapqualorig.
*/ */
qpqual = NIL; qpqual = NIL;
foreach(l, scan_clauses) foreach(l, scan_clauses)
...@@ -2575,13 +2575,9 @@ create_bitmap_scan_plan(PlannerInfo *root, ...@@ -2575,13 +2575,9 @@ create_bitmap_scan_plan(PlannerInfo *root,
continue; /* simple duplicate */ continue; /* simple duplicate */
if (rinfo->parent_ec && list_member_ptr(indexECs, rinfo->parent_ec)) if (rinfo->parent_ec && list_member_ptr(indexECs, rinfo->parent_ec))
continue; /* derived from same EquivalenceClass */ continue; /* derived from same EquivalenceClass */
if (!contain_mutable_functions(clause)) if (!contain_mutable_functions(clause) &&
{ predicate_implied_by(list_make1(clause), indexquals))
List *clausel = list_make1(clause);
if (predicate_implied_by(clausel, indexquals))
continue; /* provably implied by indexquals */ continue; /* provably implied by indexquals */
}
qpqual = lappend(qpqual, rinfo); qpqual = lappend(qpqual, rinfo);
} }
......
...@@ -339,7 +339,8 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent, ...@@ -339,7 +339,8 @@ get_relation_info(PlannerInfo *root, Oid relationObjectId, bool inhparent,
/* Build targetlist using the completed indexprs data */ /* Build targetlist using the completed indexprs data */
info->indextlist = build_index_tlist(root, info, relation); info->indextlist = build_index_tlist(root, info, relation);
info->predOK = false; /* set later in indxpath.c */ info->indrestrictinfo = NIL; /* set later, in indxpath.c */
info->predOK = false; /* set later, in indxpath.c */
info->unique = index->indisunique; info->unique = index->indisunique;
info->immediate = index->indimmediate; info->immediate = index->indimmediate;
info->hypothetical = false; info->hypothetical = false;
......
...@@ -563,6 +563,10 @@ typedef struct RelOptInfo ...@@ -563,6 +563,10 @@ typedef struct RelOptInfo
* indextlist is a TargetEntry list representing the index columns. * indextlist is a TargetEntry list representing the index columns.
* It provides an equivalent base-relation Var for each simple column, * It provides an equivalent base-relation Var for each simple column,
* and links to the matching indexprs element for each expression column. * and links to the matching indexprs element for each expression column.
*
* While most of these fields are filled when the IndexOptInfo is created
* (by plancat.c), indrestrictinfo and predOK are set later, in
* check_index_predicates().
*/ */
typedef struct IndexOptInfo typedef struct IndexOptInfo
{ {
...@@ -595,7 +599,12 @@ typedef struct IndexOptInfo ...@@ -595,7 +599,12 @@ typedef struct IndexOptInfo
List *indextlist; /* targetlist representing index columns */ List *indextlist; /* targetlist representing index columns */
bool predOK; /* true if predicate matches query */ List *indrestrictinfo;/* parent relation's baserestrictinfo list,
* less any conditions implied by the index's
* predicate (unless it's a target rel, see
* comments in check_index_predicates()) */
bool predOK; /* true if index predicate matches query */
bool unique; /* true if a unique index */ bool unique; /* true if a unique index */
bool immediate; /* is uniqueness enforced immediately? */ bool immediate; /* is uniqueness enforced immediately? */
bool hypothetical; /* true if index doesn't really exist */ bool hypothetical; /* true if index doesn't really exist */
......
...@@ -70,7 +70,7 @@ extern bool match_index_to_operand(Node *operand, int indexcol, ...@@ -70,7 +70,7 @@ extern bool match_index_to_operand(Node *operand, int indexcol,
extern void expand_indexqual_conditions(IndexOptInfo *index, extern void expand_indexqual_conditions(IndexOptInfo *index,
List *indexclauses, List *indexclausecols, List *indexclauses, List *indexclausecols,
List **indexquals_p, List **indexqualcols_p); List **indexquals_p, List **indexqualcols_p);
extern void check_partial_indexes(PlannerInfo *root, RelOptInfo *rel); extern void check_index_predicates(PlannerInfo *root, RelOptInfo *rel);
extern Expr *adjust_rowcompare_for_index(RowCompareExpr *clause, extern Expr *adjust_rowcompare_for_index(RowCompareExpr *clause,
IndexOptInfo *index, IndexOptInfo *index,
int indexcol, int indexcol,
......
...@@ -780,7 +780,6 @@ explain (costs off) ...@@ -780,7 +780,6 @@ explain (costs off)
-> Index Only Scan Backward using minmaxtest2i on minmaxtest2 -> Index Only Scan Backward using minmaxtest2i on minmaxtest2
Index Cond: (f1 IS NOT NULL) Index Cond: (f1 IS NOT NULL)
-> Index Only Scan using minmaxtest3i on minmaxtest3 -> Index Only Scan using minmaxtest3i on minmaxtest3
Index Cond: (f1 IS NOT NULL)
InitPlan 2 (returns $1) InitPlan 2 (returns $1)
-> Limit -> Limit
-> Merge Append -> Merge Append
...@@ -792,8 +791,7 @@ explain (costs off) ...@@ -792,8 +791,7 @@ explain (costs off)
-> Index Only Scan using minmaxtest2i on minmaxtest2 minmaxtest2_1 -> Index Only Scan using minmaxtest2i on minmaxtest2 minmaxtest2_1
Index Cond: (f1 IS NOT NULL) Index Cond: (f1 IS NOT NULL)
-> Index Only Scan Backward using minmaxtest3i on minmaxtest3 minmaxtest3_1 -> Index Only Scan Backward using minmaxtest3i on minmaxtest3 minmaxtest3_1
Index Cond: (f1 IS NOT NULL) (23 rows)
(25 rows)
select min(f1), max(f1) from minmaxtest; select min(f1), max(f1) from minmaxtest;
min | max min | max
...@@ -818,7 +816,6 @@ explain (costs off) ...@@ -818,7 +816,6 @@ explain (costs off)
-> Index Only Scan Backward using minmaxtest2i on minmaxtest2 -> Index Only Scan Backward using minmaxtest2i on minmaxtest2
Index Cond: (f1 IS NOT NULL) Index Cond: (f1 IS NOT NULL)
-> Index Only Scan using minmaxtest3i on minmaxtest3 -> Index Only Scan using minmaxtest3i on minmaxtest3
Index Cond: (f1 IS NOT NULL)
InitPlan 2 (returns $1) InitPlan 2 (returns $1)
-> Limit -> Limit
-> Merge Append -> Merge Append
...@@ -830,11 +827,10 @@ explain (costs off) ...@@ -830,11 +827,10 @@ explain (costs off)
-> Index Only Scan using minmaxtest2i on minmaxtest2 minmaxtest2_1 -> Index Only Scan using minmaxtest2i on minmaxtest2 minmaxtest2_1
Index Cond: (f1 IS NOT NULL) Index Cond: (f1 IS NOT NULL)
-> Index Only Scan Backward using minmaxtest3i on minmaxtest3 minmaxtest3_1 -> Index Only Scan Backward using minmaxtest3i on minmaxtest3 minmaxtest3_1
Index Cond: (f1 IS NOT NULL)
-> Sort -> Sort
Sort Key: ($0), ($1) Sort Key: ($0), ($1)
-> Result -> Result
(28 rows) (26 rows)
select distinct min(f1), max(f1) from minmaxtest; select distinct min(f1), max(f1) from minmaxtest;
min | max min | max
......
...@@ -733,6 +733,166 @@ SELECT * FROM foo ORDER BY f1 DESC NULLS LAST; ...@@ -733,6 +733,166 @@ SELECT * FROM foo ORDER BY f1 DESC NULLS LAST;
(7 rows) (7 rows)
--
-- Test planning of some cases with partial indexes
--
-- partial index is usable
explain (costs off)
select * from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
QUERY PLAN
-----------------------------------------
Index Scan using onek2_u2_prtl on onek2
Index Cond: (unique2 = 11)
Filter: (stringu1 = 'ATAAAA'::name)
(3 rows)
select * from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
unique1 | unique2 | two | four | ten | twenty | hundred | thousand | twothousand | fivethous | tenthous | odd | even | stringu1 | stringu2 | string4
---------+---------+-----+------+-----+--------+---------+----------+-------------+-----------+----------+-----+------+----------+----------+---------
494 | 11 | 0 | 2 | 4 | 14 | 4 | 94 | 94 | 494 | 494 | 8 | 9 | ATAAAA | LAAAAA | VVVVxx
(1 row)
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
QUERY PLAN
-----------------------------------------
Index Scan using onek2_u2_prtl on onek2
Index Cond: (unique2 = 11)
Filter: (stringu1 = 'ATAAAA'::name)
(3 rows)
select unique2 from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
unique2
---------
11
(1 row)
-- partial index predicate implies clause, so no need for retest
explain (costs off)
select * from onek2 where unique2 = 11 and stringu1 < 'B';
QUERY PLAN
-----------------------------------------
Index Scan using onek2_u2_prtl on onek2
Index Cond: (unique2 = 11)
(2 rows)
select * from onek2 where unique2 = 11 and stringu1 < 'B';
unique1 | unique2 | two | four | ten | twenty | hundred | thousand | twothousand | fivethous | tenthous | odd | even | stringu1 | stringu2 | string4
---------+---------+-----+------+-----+--------+---------+----------+-------------+-----------+----------+-----+------+----------+----------+---------
494 | 11 | 0 | 2 | 4 | 14 | 4 | 94 | 94 | 494 | 494 | 8 | 9 | ATAAAA | LAAAAA | VVVVxx
(1 row)
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
QUERY PLAN
----------------------------------------------
Index Only Scan using onek2_u2_prtl on onek2
Index Cond: (unique2 = 11)
(2 rows)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
unique2
---------
11
(1 row)
-- but if it's an update target, must retest anyway
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B' for update;
QUERY PLAN
-----------------------------------------------
LockRows
-> Index Scan using onek2_u2_prtl on onek2
Index Cond: (unique2 = 11)
Filter: (stringu1 < 'B'::name)
(4 rows)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B' for update;
unique2
---------
11
(1 row)
-- partial index is not applicable
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'C';
QUERY PLAN
-------------------------------------------------------
Seq Scan on onek2
Filter: ((stringu1 < 'C'::name) AND (unique2 = 11))
(2 rows)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'C';
unique2
---------
11
(1 row)
-- partial index implies clause, but bitmap scan must recheck predicate anyway
SET enable_indexscan TO off;
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
QUERY PLAN
-------------------------------------------------------------
Bitmap Heap Scan on onek2
Recheck Cond: ((unique2 = 11) AND (stringu1 < 'B'::name))
-> Bitmap Index Scan on onek2_u2_prtl
Index Cond: (unique2 = 11)
(4 rows)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
unique2
---------
11
(1 row)
RESET enable_indexscan;
-- check multi-index cases too
explain (costs off)
select unique1, unique2 from onek2
where (unique2 = 11 or unique1 = 0) and stringu1 < 'B';
QUERY PLAN
--------------------------------------------------------------------------------
Bitmap Heap Scan on onek2
Recheck Cond: (((unique2 = 11) AND (stringu1 < 'B'::name)) OR (unique1 = 0))
Filter: (stringu1 < 'B'::name)
-> BitmapOr
-> Bitmap Index Scan on onek2_u2_prtl
Index Cond: (unique2 = 11)
-> Bitmap Index Scan on onek2_u1_prtl
Index Cond: (unique1 = 0)
(8 rows)
select unique1, unique2 from onek2
where (unique2 = 11 or unique1 = 0) and stringu1 < 'B';
unique1 | unique2
---------+---------
494 | 11
0 | 998
(2 rows)
explain (costs off)
select unique1, unique2 from onek2
where (unique2 = 11 and stringu1 < 'B') or unique1 = 0;
QUERY PLAN
--------------------------------------------------------------------------------
Bitmap Heap Scan on onek2
Recheck Cond: (((unique2 = 11) AND (stringu1 < 'B'::name)) OR (unique1 = 0))
-> BitmapOr
-> Bitmap Index Scan on onek2_u2_prtl
Index Cond: (unique2 = 11)
-> Bitmap Index Scan on onek2_u1_prtl
Index Cond: (unique1 = 0)
(7 rows)
select unique1, unique2 from onek2
where (unique2 = 11 and stringu1 < 'B') or unique1 = 0;
unique1 | unique2
---------+---------
494 | 11
0 | 998
(2 rows)
-- --
-- Test some corner cases that have been known to confuse the planner -- Test some corner cases that have been known to confuse the planner
-- --
......
...@@ -187,6 +187,50 @@ SELECT * FROM foo ORDER BY f1 NULLS FIRST; ...@@ -187,6 +187,50 @@ SELECT * FROM foo ORDER BY f1 NULLS FIRST;
SELECT * FROM foo ORDER BY f1 DESC; SELECT * FROM foo ORDER BY f1 DESC;
SELECT * FROM foo ORDER BY f1 DESC NULLS LAST; SELECT * FROM foo ORDER BY f1 DESC NULLS LAST;
--
-- Test planning of some cases with partial indexes
--
-- partial index is usable
explain (costs off)
select * from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
select * from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
select unique2 from onek2 where unique2 = 11 and stringu1 = 'ATAAAA';
-- partial index predicate implies clause, so no need for retest
explain (costs off)
select * from onek2 where unique2 = 11 and stringu1 < 'B';
select * from onek2 where unique2 = 11 and stringu1 < 'B';
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
-- but if it's an update target, must retest anyway
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B' for update;
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B' for update;
-- partial index is not applicable
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'C';
select unique2 from onek2 where unique2 = 11 and stringu1 < 'C';
-- partial index implies clause, but bitmap scan must recheck predicate anyway
SET enable_indexscan TO off;
explain (costs off)
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
select unique2 from onek2 where unique2 = 11 and stringu1 < 'B';
RESET enable_indexscan;
-- check multi-index cases too
explain (costs off)
select unique1, unique2 from onek2
where (unique2 = 11 or unique1 = 0) and stringu1 < 'B';
select unique1, unique2 from onek2
where (unique2 = 11 or unique1 = 0) and stringu1 < 'B';
explain (costs off)
select unique1, unique2 from onek2
where (unique2 = 11 and stringu1 < 'B') or unique1 = 0;
select unique1, unique2 from onek2
where (unique2 = 11 and stringu1 < 'B') or unique1 = 0;
-- --
-- Test some corner cases that have been known to confuse the planner -- Test some corner cases that have been known to confuse the planner
-- --
......
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