Commit e6f7edb9 authored by Tom Lane's avatar Tom Lane

Install some slightly realistic cost estimation for bitmap index scans.

parent 2f8c7c86
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.247 2005/04/19 22:35:14 tgl Exp $ * $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.248 2005/04/21 02:28:01 tgl Exp $
* *
* NOTES * NOTES
* Every node type that can appear in stored rules' parsetrees *must* * Every node type that can appear in stored rules' parsetrees *must*
...@@ -1024,6 +1024,8 @@ _outIndexPath(StringInfo str, IndexPath *node) ...@@ -1024,6 +1024,8 @@ _outIndexPath(StringInfo str, IndexPath *node)
WRITE_NODE_FIELD(indexquals); WRITE_NODE_FIELD(indexquals);
WRITE_BOOL_FIELD(isjoininner); WRITE_BOOL_FIELD(isjoininner);
WRITE_ENUM_FIELD(indexscandir, ScanDirection); WRITE_ENUM_FIELD(indexscandir, ScanDirection);
WRITE_FLOAT_FIELD(indextotalcost, "%.2f");
WRITE_FLOAT_FIELD(indexselectivity, "%.4f");
WRITE_FLOAT_FIELD(rows, "%.0f"); WRITE_FLOAT_FIELD(rows, "%.0f");
} }
......
...@@ -49,7 +49,7 @@ ...@@ -49,7 +49,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/costsize.c,v 1.142 2005/04/19 22:35:15 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/costsize.c,v 1.143 2005/04/21 02:28:01 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -103,6 +103,7 @@ bool enable_hashjoin = true; ...@@ -103,6 +103,7 @@ bool enable_hashjoin = true;
static bool cost_qual_eval_walker(Node *node, QualCost *total); static bool cost_qual_eval_walker(Node *node, QualCost *total);
static Selectivity cost_bitmap_qual(Node *bitmapqual, Cost *totalCost);
static Selectivity approx_selectivity(Query *root, List *quals, static Selectivity approx_selectivity(Query *root, List *quals,
JoinType jointype); JoinType jointype);
static Selectivity join_in_selectivity(JoinPath *path, Query *root); static Selectivity join_in_selectivity(JoinPath *path, Query *root);
...@@ -126,7 +127,7 @@ clamp_row_est(double nrows) ...@@ -126,7 +127,7 @@ clamp_row_est(double nrows)
if (nrows < 1.0) if (nrows < 1.0)
nrows = 1.0; nrows = 1.0;
else else
nrows = ceil(nrows); nrows = rint(nrows);
return nrows; return nrows;
} }
...@@ -232,6 +233,10 @@ cost_nonsequential_access(double relpages) ...@@ -232,6 +233,10 @@ cost_nonsequential_access(double relpages)
* 'is_injoin' is T if we are considering using the index scan as the inside * 'is_injoin' is T if we are considering using the index scan as the inside
* of a nestloop join (hence, some of the indexQuals are join clauses) * of a nestloop join (hence, some of the indexQuals are join clauses)
* *
* cost_index() takes an IndexPath not just a Path, because it sets a few
* additional fields of the IndexPath besides startup_cost and total_cost.
* These fields are needed if the IndexPath is used in a BitmapIndexScan.
*
* NOTE: 'indexQuals' must contain only clauses usable as index restrictions. * NOTE: 'indexQuals' must contain only clauses usable as index restrictions.
* Any additional quals evaluated as qpquals may reduce the number of returned * Any additional quals evaluated as qpquals may reduce the number of returned
* tuples, but they won't reduce the number of tuples we have to fetch from * tuples, but they won't reduce the number of tuples we have to fetch from
...@@ -241,7 +246,7 @@ cost_nonsequential_access(double relpages) ...@@ -241,7 +246,7 @@ cost_nonsequential_access(double relpages)
* it was a list of bare clause expressions. * it was a list of bare clause expressions.
*/ */
void void
cost_index(Path *path, Query *root, cost_index(IndexPath *path, Query *root,
IndexOptInfo *index, IndexOptInfo *index,
List *indexQuals, List *indexQuals,
bool is_injoin) bool is_injoin)
...@@ -286,6 +291,14 @@ cost_index(Path *path, Query *root, ...@@ -286,6 +291,14 @@ cost_index(Path *path, Query *root,
PointerGetDatum(&indexSelectivity), PointerGetDatum(&indexSelectivity),
PointerGetDatum(&indexCorrelation)); PointerGetDatum(&indexCorrelation));
/*
* Save amcostestimate's results for possible use by cost_bitmap_scan.
* We don't bother to save indexStartupCost or indexCorrelation, because
* a bitmap scan doesn't care about either.
*/
path->indextotalcost = indexTotalCost;
path->indexselectivity = indexSelectivity;
/* all costs for touching index itself included here */ /* all costs for touching index itself included here */
startup_cost += indexStartupCost; startup_cost += indexStartupCost;
run_cost += indexTotalCost - indexStartupCost; run_cost += indexTotalCost - indexStartupCost;
...@@ -396,8 +409,8 @@ cost_index(Path *path, Query *root, ...@@ -396,8 +409,8 @@ cost_index(Path *path, Query *root,
run_cost += cpu_per_tuple * tuples_fetched; run_cost += cpu_per_tuple * tuples_fetched;
path->startup_cost = startup_cost; path->path.startup_cost = startup_cost;
path->total_cost = startup_cost + run_cost; path->path.total_cost = startup_cost + run_cost;
} }
/* /*
...@@ -417,19 +430,151 @@ cost_bitmap_scan(Path *path, Query *root, RelOptInfo *baserel, ...@@ -417,19 +430,151 @@ cost_bitmap_scan(Path *path, Query *root, RelOptInfo *baserel,
{ {
Cost startup_cost = 0; Cost startup_cost = 0;
Cost run_cost = 0; Cost run_cost = 0;
Cost indexTotalCost;
Selectivity indexSelectivity;
Cost cpu_per_tuple;
Cost cost_per_page;
double tuples_fetched;
double pages_fetched;
double T;
/* Should only be applied to base relations */ /* Should only be applied to base relations */
Assert(IsA(baserel, RelOptInfo)); Assert(IsA(baserel, RelOptInfo));
Assert(baserel->relid > 0); Assert(baserel->relid > 0);
Assert(baserel->rtekind == RTE_RELATION); Assert(baserel->rtekind == RTE_RELATION);
/* XXX lots to do here */ if (!enable_indexscan) /* XXX use a separate enable flag? */
run_cost += 10; startup_cost += disable_cost;
/*
* Estimate total cost of obtaining the bitmap, as well as its total
* selectivity.
*/
indexTotalCost = 0;
indexSelectivity = cost_bitmap_qual(bitmapqual, &indexTotalCost);
startup_cost += indexTotalCost;
/*
* The number of heap pages that need to be fetched is the same as the
* Mackert and Lohman formula for the case T <= b (ie, no re-reads
* needed).
*/
tuples_fetched = clamp_row_est(indexSelectivity * baserel->tuples);
T = (baserel->pages > 1) ? (double) baserel->pages : 1.0;
pages_fetched = (2.0 * T * tuples_fetched) / (2.0 * T + tuples_fetched);
if (pages_fetched > T)
pages_fetched = T;
/*
* For small numbers of pages we should charge random_page_cost apiece,
* while if nearly all the table's pages are being read, it's more
* appropriate to charge 1.0 apiece. The effect is nonlinear, too.
* For lack of a better idea, interpolate like this to determine the
* cost per page.
*/
cost_per_page = random_page_cost -
(random_page_cost - 1.0) * sqrt(pages_fetched / T);
run_cost += pages_fetched * cost_per_page;
/*
* Estimate CPU costs per tuple.
*
* Often the indexquals don't need to be rechecked at each tuple ...
* but not always, especially not if there are enough tuples involved
* that the bitmaps become lossy. For the moment, just assume they
* will be rechecked always.
*/
startup_cost += baserel->baserestrictcost.startup;
cpu_per_tuple = cpu_tuple_cost + baserel->baserestrictcost.per_tuple;
run_cost += cpu_per_tuple * tuples_fetched;
path->startup_cost = startup_cost; path->startup_cost = startup_cost;
path->total_cost = startup_cost + run_cost; path->total_cost = startup_cost + run_cost;
} }
/*
* cost_bitmap_qual
* Recursively examine the AND/OR/IndexPath tree for a bitmap scan
*
* Total execution costs are added to *totalCost (so caller must be sure
* to initialize that to zero). Estimated total selectivity of the bitmap
* is returned as the function result.
*/
static Selectivity
cost_bitmap_qual(Node *bitmapqual, Cost *totalCost)
{
Selectivity result;
Selectivity subresult;
ListCell *l;
if (and_clause(bitmapqual))
{
/*
* We estimate AND selectivity on the assumption that the inputs
* are independent. This is probably often wrong, but we don't
* have the info to do better.
*
* The runtime cost of the BitmapAnd itself is estimated at 100x
* cpu_operator_cost for each tbm_intersect needed. Probably too
* small, definitely too simplistic?
*
* This must agree with make_bitmap_and in createplan.c.
*/
result = 1.0;
foreach(l, ((BoolExpr *) bitmapqual)->args)
{
subresult = cost_bitmap_qual((Node *) lfirst(l), totalCost);
result *= subresult;
if (l != list_head(((BoolExpr *) bitmapqual)->args))
*totalCost += 100.0 * cpu_operator_cost;
}
}
else if (or_clause(bitmapqual))
{
/*
* We estimate OR selectivity on the assumption that the inputs
* are non-overlapping, since that's often the case in "x IN (list)"
* type situations. Of course, we clamp to 1.0 at the end.
*
* The runtime cost of the BitmapOr itself is estimated at 100x
* cpu_operator_cost for each tbm_union needed. Probably too
* small, definitely too simplistic? We are aware that the tbm_unions
* are optimized out when the inputs are BitmapIndexScans.
*
* This must agree with make_bitmap_or in createplan.c.
*/
result = 0.0;
foreach(l, ((BoolExpr *) bitmapqual)->args)
{
subresult = cost_bitmap_qual((Node *) lfirst(l), totalCost);
result += subresult;
if (l != list_head(((BoolExpr *) bitmapqual)->args) &&
!IsA((Node *) lfirst(l), IndexPath))
*totalCost += 100.0 * cpu_operator_cost;
}
result = Min(result, 1.0);
}
else if (IsA(bitmapqual, IndexPath))
{
IndexPath *ipath = (IndexPath *) bitmapqual;
/* this must agree with create_bitmap_subplan in createplan.c */
*totalCost += ipath->indextotalcost;
result = ipath->indexselectivity;
}
else
{
elog(ERROR, "unrecognized node type: %d", nodeTag(bitmapqual));
result = 0.0; /* keep compiler quiet */
}
return result;
}
/* /*
* cost_tidscan * cost_tidscan
* Determines and returns the cost of scanning a relation using TIDs. * Determines and returns the cost of scanning a relation using TIDs.
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/indxpath.c,v 1.174 2005/04/20 21:48:04 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/indxpath.c,v 1.175 2005/04/21 02:28:01 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1710,7 +1710,7 @@ make_innerjoin_index_path(Query *root, ...@@ -1710,7 +1710,7 @@ make_innerjoin_index_path(Query *root,
/* Like costsize.c, force estimate to be at least one row */ /* Like costsize.c, force estimate to be at least one row */
pathnode->rows = clamp_row_est(pathnode->rows); pathnode->rows = clamp_row_est(pathnode->rows);
cost_index(&pathnode->path, root, index, indexquals, true); cost_index(pathnode, root, index, indexquals, true);
return (Path *) pathnode; return (Path *) pathnode;
} }
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/orindxpath.c,v 1.67 2005/03/27 06:29:36 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/orindxpath.c,v 1.68 2005/04/21 02:28:01 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -353,7 +353,7 @@ best_or_subclause_index(Query *root, ...@@ -353,7 +353,7 @@ best_or_subclause_index(Query *root,
IndexOptInfo *index = (IndexOptInfo *) lfirst(ilist); IndexOptInfo *index = (IndexOptInfo *) lfirst(ilist);
List *indexclauses; List *indexclauses;
List *indexquals; List *indexquals;
Path subclause_path; IndexPath subclause_path;
/* /*
* Ignore partial indexes that do not match the query. If predOK * Ignore partial indexes that do not match the query. If predOK
...@@ -402,13 +402,13 @@ best_or_subclause_index(Query *root, ...@@ -402,13 +402,13 @@ best_or_subclause_index(Query *root,
cost_index(&subclause_path, root, index, indexquals, false); cost_index(&subclause_path, root, index, indexquals, false);
if (!found || subclause_path.total_cost < *retTotalCost) if (!found || subclause_path.path.total_cost < *retTotalCost)
{ {
*retIndexInfo = index; *retIndexInfo = index;
*retIndexClauses = flatten_clausegroups_list(indexclauses); *retIndexClauses = flatten_clausegroups_list(indexclauses);
*retIndexQuals = indexquals; *retIndexQuals = indexquals;
*retStartupCost = subclause_path.startup_cost; *retStartupCost = subclause_path.path.startup_cost;
*retTotalCost = subclause_path.total_cost; *retTotalCost = subclause_path.path.total_cost;
found = true; found = true;
} }
} }
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/plan/createplan.c,v 1.180 2005/04/19 22:35:16 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/plan/createplan.c,v 1.181 2005/04/21 02:28:01 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -976,10 +976,12 @@ create_bitmap_subplan(Query *root, Node *bitmapqual) ...@@ -976,10 +976,12 @@ create_bitmap_subplan(Query *root, Node *bitmapqual)
linitial(iscan->indxqualorig), linitial(iscan->indxqualorig),
linitial(iscan->indxstrategy), linitial(iscan->indxstrategy),
linitial(iscan->indxsubtype)); linitial(iscan->indxsubtype));
/* XXX this cost is wrong: */ /* this must agree with cost_bitmap_qual in costsize.c */
copy_path_costsize(&bscan->scan.plan, &ipath->path); bscan->scan.plan.startup_cost = 0.0;
/* use the indexscan-specific rows estimate, not the parent rel's */ bscan->scan.plan.total_cost = ipath->indextotalcost;
bscan->scan.plan.plan_rows = ipath->rows; bscan->scan.plan.plan_rows =
clamp_row_est(ipath->indexselectivity * ipath->path.parent->tuples);
bscan->scan.plan.plan_width = 0; /* meaningless */
plan = (Plan *) bscan; plan = (Plan *) bscan;
} }
else else
...@@ -2068,8 +2070,9 @@ make_bitmap_and(List *bitmapplans) ...@@ -2068,8 +2070,9 @@ make_bitmap_and(List *bitmapplans)
ListCell *subnode; ListCell *subnode;
/* /*
* Compute cost as sum of subplan costs, plus 10x cpu_operator_cost * Compute cost as sum of subplan costs, plus 100x cpu_operator_cost
* (a pretty arbitrary amount, agreed) for each tbm_intersect needed. * (a pretty arbitrary amount, agreed) for each tbm_intersect needed.
* This must agree with cost_bitmap_qual in costsize.c.
*/ */
plan->startup_cost = 0; plan->startup_cost = 0;
plan->total_cost = 0; plan->total_cost = 0;
...@@ -2085,7 +2088,10 @@ make_bitmap_and(List *bitmapplans) ...@@ -2085,7 +2088,10 @@ make_bitmap_and(List *bitmapplans)
plan->plan_rows = subplan->plan_rows; plan->plan_rows = subplan->plan_rows;
} }
else else
{
plan->total_cost += cpu_operator_cost * 100.0;
plan->plan_rows = Min(plan->plan_rows, subplan->plan_rows); plan->plan_rows = Min(plan->plan_rows, subplan->plan_rows);
}
plan->total_cost += subplan->total_cost; plan->total_cost += subplan->total_cost;
} }
...@@ -2106,10 +2112,12 @@ make_bitmap_or(List *bitmapplans) ...@@ -2106,10 +2112,12 @@ make_bitmap_or(List *bitmapplans)
ListCell *subnode; ListCell *subnode;
/* /*
* Compute cost as sum of subplan costs, plus 10x cpu_operator_cost * Compute cost as sum of subplan costs, plus 100x cpu_operator_cost
* (a pretty arbitrary amount, agreed) for each tbm_union needed. * (a pretty arbitrary amount, agreed) for each tbm_union needed.
* We assume that tbm_union can be optimized away for BitmapIndexScan * We assume that tbm_union can be optimized away for BitmapIndexScan
* subplans. * subplans.
*
* This must agree with cost_bitmap_qual in costsize.c.
*/ */
plan->startup_cost = 0; plan->startup_cost = 0;
plan->total_cost = 0; plan->total_cost = 0;
...@@ -2122,7 +2130,7 @@ make_bitmap_or(List *bitmapplans) ...@@ -2122,7 +2130,7 @@ make_bitmap_or(List *bitmapplans)
if (subnode == list_head(bitmapplans)) /* first node? */ if (subnode == list_head(bitmapplans)) /* first node? */
plan->startup_cost = subplan->startup_cost; plan->startup_cost = subplan->startup_cost;
else if (!IsA(subplan, BitmapIndexScan)) else if (!IsA(subplan, BitmapIndexScan))
plan->total_cost += cpu_operator_cost * 10; plan->total_cost += cpu_operator_cost * 100.0;
plan->total_cost += subplan->total_cost; plan->total_cost += subplan->total_cost;
plan->plan_rows += subplan->plan_rows; /* ignore overlap */ plan->plan_rows += subplan->plan_rows; /* ignore overlap */
} }
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/util/pathnode.c,v 1.116 2005/04/19 22:35:17 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/util/pathnode.c,v 1.117 2005/04/21 02:28:01 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -466,7 +466,7 @@ create_index_path(Query *root, ...@@ -466,7 +466,7 @@ create_index_path(Query *root,
*/ */
pathnode->rows = index->rel->rows; pathnode->rows = index->rel->rows;
cost_index(&pathnode->path, root, index, indexquals, false); cost_index(pathnode, root, index, indexquals, false);
return pathnode; return pathnode;
} }
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/relation.h,v 1.105 2005/04/19 22:35:17 tgl Exp $ * $PostgreSQL: pgsql/src/include/nodes/relation.h,v 1.106 2005/04/21 02:28:02 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -374,6 +374,10 @@ typedef struct Path ...@@ -374,6 +374,10 @@ typedef struct Path
* NoMovementScanDirection for an indexscan, but the planner wants to * NoMovementScanDirection for an indexscan, but the planner wants to
* distinguish ordered from unordered indexes for building pathkeys.) * distinguish ordered from unordered indexes for building pathkeys.)
* *
* 'indextotalcost' and 'indexselectivity' are saved in the IndexPath so that
* we need not recompute them when considering using the same index in a
* bitmap index/heap scan (see BitmapHeapPath).
*
* 'rows' is the estimated result tuple count for the indexscan. This * 'rows' is the estimated result tuple count for the indexscan. This
* is the same as path.parent->rows for a simple indexscan, but it is * is the same as path.parent->rows for a simple indexscan, but it is
* different for a nestloop inner scan, because the additional indexquals * different for a nestloop inner scan, because the additional indexquals
...@@ -389,6 +393,8 @@ typedef struct IndexPath ...@@ -389,6 +393,8 @@ typedef struct IndexPath
List *indexquals; List *indexquals;
bool isjoininner; bool isjoininner;
ScanDirection indexscandir; ScanDirection indexscandir;
Cost indextotalcost;
Selectivity indexselectivity;
double rows; /* estimated number of result tuples */ double rows; /* estimated number of result tuples */
} IndexPath; } IndexPath;
...@@ -401,9 +407,12 @@ typedef struct IndexPath ...@@ -401,9 +407,12 @@ typedef struct IndexPath
* *
* The individual indexscans are represented by IndexPath nodes, and any * The individual indexscans are represented by IndexPath nodes, and any
* logic on top of them is represented by regular AND and OR expressions. * logic on top of them is represented by regular AND and OR expressions.
* Notice that we can use the same IndexPath node both to represent an * Notice that we can use the same IndexPath node both to represent a regular
* ordered index scan, and as the child of a BitmapHeapPath that represents * IndexScan plan, and as the child of a BitmapHeapPath that represents
* scanning the same index in an unordered way. * scanning the same index using a BitmapIndexScan. The startup_cost and
* total_cost figures of an IndexPath always represent the costs to use it
* as a regular IndexScan. The costs of a BitmapIndexScan can be computed
* using the IndexPath's indextotalcost and indexselectivity.
* *
* BitmapHeapPaths can be nestloop inner indexscans. The isjoininner and * BitmapHeapPaths can be nestloop inner indexscans. The isjoininner and
* rows fields serve the same purpose as for plain IndexPaths. * rows fields serve the same purpose as for plain IndexPaths.
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/optimizer/cost.h,v 1.64 2005/04/19 22:35:18 tgl Exp $ * $PostgreSQL: pgsql/src/include/optimizer/cost.h,v 1.65 2005/04/21 02:28:02 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -51,7 +51,7 @@ extern bool enable_hashjoin; ...@@ -51,7 +51,7 @@ extern bool enable_hashjoin;
extern double clamp_row_est(double nrows); extern double clamp_row_est(double nrows);
extern void cost_seqscan(Path *path, Query *root, RelOptInfo *baserel); extern void cost_seqscan(Path *path, Query *root, RelOptInfo *baserel);
extern void cost_index(Path *path, Query *root, IndexOptInfo *index, extern void cost_index(IndexPath *path, Query *root, IndexOptInfo *index,
List *indexQuals, bool is_injoin); List *indexQuals, bool is_injoin);
extern void cost_bitmap_scan(Path *path, Query *root, RelOptInfo *baserel, extern void cost_bitmap_scan(Path *path, Query *root, RelOptInfo *baserel,
Node *bitmapqual, bool is_injoin); Node *bitmapqual, bool is_injoin);
......
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