PartitionDesc partdesc;
Relation rel;
PartitionKey partkey;
+ ListCell *lc2;
int partnatts;
+ int n_steps;
pprune->present_parts = bms_copy(pinfo->present_parts);
pprune->subnode_map = palloc(sizeof(int) * pinfo->nparts);
partkey = RelationGetPartitionKey(rel);
partdesc = RelationGetPartitionDesc(rel);
+ n_steps = list_length(pinfo->pruning_steps);
context->strategy = partkey->strategy;
context->partnatts = partnatts = partkey->partnatts;
context->boundinfo = partition_bounds_copy(partdesc->boundinfo, partkey);
context->planstate = planstate;
context->safeparams = NULL; /* empty for now */
+ context->exprstates = palloc0(sizeof(ExprState *) * n_steps * partnatts);
+
+ /* Initialize expression states for each expression */
+ foreach(lc2, pinfo->pruning_steps)
+ {
+ PartitionPruneStepOp *step = (PartitionPruneStepOp *) lfirst(lc2);
+ ListCell *lc3;
+ int keyno;
+
+ /* not needed for other step kinds */
+ if (!IsA(step, PartitionPruneStepOp))
+ continue;
+
+ Assert(list_length(step->exprs) <= partnatts);
+
+ keyno = 0;
+ foreach(lc3, step->exprs)
+ {
+ Expr *expr = (Expr *) lfirst(lc3);
+ int stateidx;
+
+ /* not needed for Consts */
+ if (!IsA(expr, Const))
+ {
+ stateidx = PruneCxtStateIdx(partnatts,
+ step->step.step_id, keyno);
+ context->exprstates[stateidx] =
+ ExecInitExpr(expr, context->planstate);
+ }
+ keyno++;
+ }
+ }
pprune->pruning_steps = pinfo->pruning_steps;
pprune->extparams = bms_copy(pinfo->extparams);
static bool match_boolean_partition_clause(Oid partopfamily, Expr *clause,
Expr *partkey, Expr **outconst);
static bool partkey_datum_from_expr(PartitionPruneContext *context,
- Expr *expr, Datum *value);
+ Expr *expr, int stateidx, Datum *value);
/*
* make_partition_pruneinfo
/* Not valid when being called from the planner */
context.planstate = NULL;
context.safeparams = NULL;
+ context.exprstates = NULL;
/* Actual pruning happens here. */
partindexes = get_matching_partitions(&context, pruning_steps);
if (lc1 != NULL)
{
Expr *expr;
+ int stateidx;
Datum datum;
expr = lfirst(lc1);
- if (partkey_datum_from_expr(context, expr, &datum))
+ stateidx = PruneCxtStateIdx(context->partnatts,
+ opstep->step.step_id, keyno);
+ if (partkey_datum_from_expr(context, expr, stateidx, &datum))
{
Oid cmpfn;
/*
* partkey_datum_from_expr
- * Evaluate 'expr', set *value to the resulting Datum. Return true if
- * evaluation was possible, otherwise false.
+ * Evaluate expression for potential partition pruning
+ *
+ * Evaluate 'expr', whose ExprState is stateidx of the context exprstate
+ * array; set *value to the resulting Datum. Return true if evaluation was
+ * possible, otherwise false.
*/
static bool
partkey_datum_from_expr(PartitionPruneContext *context,
- Expr *expr, Datum *value)
+ Expr *expr, int stateidx, Datum *value)
{
switch (nodeTag(expr))
{
bms_is_member(((Param *) expr)->paramid, context->safeparams))
{
ExprState *exprstate;
+ ExprContext *ectx;
bool isNull;
- exprstate = ExecInitExpr(expr, context->planstate);
-
- *value = ExecEvalExprSwitchContext(exprstate,
- context->planstate->ps_ExprContext,
- &isNull);
+ exprstate = context->exprstates[stateidx];
+ ectx = context->planstate->ps_ExprContext;
+ *value = ExecEvalExprSwitchContext(exprstate, ectx, &isNull);
if (isNull)
return false;
return true;
}
+ break;
default:
break;
* are not safe to use until the executor is running.
*/
Bitmapset *safeparams;
+
+ /*
+ * Array of ExprStates, indexed as per PruneCtxStateIdx; one for each
+ * partkey in each pruning step. Allocated if planstate is non-NULL,
+ * otherwise NULL.
+ */
+ ExprState **exprstates;
} PartitionPruneContext;
+#define PruneCxtStateIdx(partnatts, step_id, keyno) \
+ ((partnatts) * (step_id) + (keyno))
extern List *make_partition_pruneinfo(PlannerInfo *root, List *partition_rels,
List *subpaths, List *prunequal);