From 2a301f59561ac9f5d836122ae26ec4609033392e Mon Sep 17 00:00:00 2001 From: "dgrowley@gmail.com" Date: Sun, 1 Apr 2018 01:20:47 +1300 Subject: [PATCH v18 5/5] Improve planning speed of partitioned table UPDATE/DELETEs By making a call to grouping_planner for the complete parse of the query we can make use of the faster partition pruning code used there. This will identify all partitions which could be pruned as IS_DUMMY_RELs, of which we can skip performing each individual grouping_planner call inside inheritance_planner. This can improve planner performance significantly when there are many partitions. There may be a slight slowdown when no partitions could be pruned or when there are very few (1 or 2) partitions. However it seems better to optimize the case when partitions are pruned, rather than the case where they're not, as those queries are less likely to be fast to execute. The case for partitioned tables with just 1 or 2 leaf partitions does not seem worth worrying about too much. The measured regression on 1 partition was just 10% of overall planning time. This commit also implements run-time partition pruning for UPDATE/DELETE. --- src/backend/commands/explain.c | 4 +- src/backend/executor/execPartition.c | 14 +-- src/backend/executor/nodeModifyTable.c | 159 ++++++++++++++++++++++++-------- src/backend/nodes/copyfuncs.c | 1 + src/backend/nodes/outfuncs.c | 1 + src/backend/nodes/readfuncs.c | 1 + src/backend/optimizer/plan/createplan.c | 32 ++++++- src/backend/optimizer/plan/planner.c | 59 ++++++++++++ src/include/nodes/execnodes.h | 11 ++- src/include/nodes/plannodes.h | 2 + 10 files changed, 233 insertions(+), 51 deletions(-) diff --git a/src/backend/commands/explain.c b/src/backend/commands/explain.c index a3db51e660..865ba928d3 100644 --- a/src/backend/commands/explain.c +++ b/src/backend/commands/explain.c @@ -3021,14 +3021,14 @@ show_modifytable_info(ModifyTableState *mtstate, List *ancestors, /* Should we explicitly label target relations? */ labeltargets = (mtstate->mt_nplans > 1 || (mtstate->mt_nplans == 1 && - mtstate->resultRelInfo->ri_RangeTableIndex != node->nominalRelation)); + mtstate->resultRelInfos[0]->ri_RangeTableIndex != node->nominalRelation)); if (labeltargets) ExplainOpenGroup("Target Tables", "Target Tables", false, es); for (j = 0; j < mtstate->mt_nplans; j++) { - ResultRelInfo *resultRelInfo = mtstate->resultRelInfo + j; + ResultRelInfo *resultRelInfo = mtstate->resultRelInfos[j]; FdwRoutine *fdwroutine = resultRelInfo->ri_FdwRoutine; if (labeltargets) diff --git a/src/backend/executor/execPartition.c b/src/backend/executor/execPartition.c index 7c4f56c319..fb46a458b0 100644 --- a/src/backend/executor/execPartition.c +++ b/src/backend/executor/execPartition.c @@ -96,7 +96,7 @@ ExecSetupPartitionTupleRouting(ModifyTableState *mtstate, Relation rel) /* Set up details specific to the type of tuple routing we are doing. */ if (node && node->operation == CMD_UPDATE) { - update_rri = mtstate->resultRelInfo; + update_rri = mtstate->resultRelInfos[0]; num_update_rri = list_length(node->plans); proute->subplan_partition_offsets = palloc(num_update_rri * sizeof(int)); @@ -386,8 +386,8 @@ ExecInitPartitionInfo(ModifyTableState *mtstate, List *wcoList; List *wcoExprs = NIL; ListCell *ll; - int firstVarno = mtstate->resultRelInfo[0].ri_RangeTableIndex; - Relation firstResultRel = mtstate->resultRelInfo[0].ri_RelationDesc; + int firstVarno = mtstate->resultRelInfos[0]->ri_RangeTableIndex; + Relation firstResultRel = mtstate->resultRelInfos[0]->ri_RelationDesc; /* * In the case of INSERT on a partitioned table, there is only one @@ -442,8 +442,8 @@ ExecInitPartitionInfo(ModifyTableState *mtstate, TupleTableSlot *slot; ExprContext *econtext; List *returningList; - int firstVarno = mtstate->resultRelInfo[0].ri_RangeTableIndex; - Relation firstResultRel = mtstate->resultRelInfo[0].ri_RelationDesc; + int firstVarno = mtstate->resultRelInfos[0]->ri_RangeTableIndex; + Relation firstResultRel = mtstate->resultRelInfos[0]->ri_RelationDesc; /* See the comment above for WCO lists. */ Assert((node->operation == CMD_INSERT && @@ -498,8 +498,8 @@ ExecInitPartitionInfo(ModifyTableState *mtstate, if (node && node->onConflictAction != ONCONFLICT_NONE) { TupleConversionMap *map = proute->parent_child_tupconv_maps[partidx]; - int firstVarno = mtstate->resultRelInfo[0].ri_RangeTableIndex; - Relation firstResultRel = mtstate->resultRelInfo[0].ri_RelationDesc; + int firstVarno = mtstate->resultRelInfos[0]->ri_RangeTableIndex; + Relation firstResultRel = mtstate->resultRelInfos[0]->ri_RelationDesc; TupleDesc partrelDesc = RelationGetDescr(partrel); ExprContext *econtext = mtstate->ps.ps_ExprContext; ListCell *lc; diff --git a/src/backend/executor/nodeModifyTable.c b/src/backend/executor/nodeModifyTable.c index 1b09868ff8..3afa9d07c8 100644 --- a/src/backend/executor/nodeModifyTable.c +++ b/src/backend/executor/nodeModifyTable.c @@ -1116,14 +1116,10 @@ lreplace:; saved_tcs_map = mtstate->mt_transition_capture->tcs_map; /* - * resultRelInfo is one of the per-subplan resultRelInfos. So we - * should convert the tuple into root's tuple descriptor, since - * ExecInsert() starts the search from root. The tuple conversion - * map list is in the order of mtstate->resultRelInfo[], so to - * retrieve the one for this resultRel, we need to know the - * position of the resultRel in mtstate->resultRelInfo[]. + * Convert the tuple into root's tuple descriptor, since + * ExecInsert() starts the search from root. */ - map_index = resultRelInfo - mtstate->resultRelInfo; + map_index = mtstate->mt_whichplan; Assert(map_index >= 0 && map_index < mtstate->mt_nplans); tupconv_map = tupconv_map_for_subplan(mtstate, map_index); tuple = ConvertPartitionTupleSlot(tupconv_map, @@ -1503,12 +1499,12 @@ static void fireBSTriggers(ModifyTableState *node) { ModifyTable *plan = (ModifyTable *) node->ps.plan; - ResultRelInfo *resultRelInfo = node->resultRelInfo; + ResultRelInfo *resultRelInfo = node->resultRelInfos[0]; /* * If the node modifies a partitioned table, we must fire its triggers. - * Note that in that case, node->resultRelInfo points to the first leaf - * partition, not the root table. + * Note that in that case, node->resultRelInfos[0] points to the first + * leaf partition, not the root table. */ if (node->rootResultRelInfo != NULL) resultRelInfo = node->rootResultRelInfo; @@ -1546,13 +1542,14 @@ static ResultRelInfo * getTargetResultRelInfo(ModifyTableState *node) { /* - * Note that if the node modifies a partitioned table, node->resultRelInfo - * points to the first leaf partition, not the root table. + * Note that if the node modifies a partitioned table, + * node->resultRelInfos[0] points to the first leaf partition, not the + * root table. */ if (node->rootResultRelInfo != NULL) return node->rootResultRelInfo; else - return node->resultRelInfo; + return node->resultRelInfos[0]; } /* @@ -1762,7 +1759,7 @@ static void ExecSetupChildParentMapForSubplan(ModifyTableState *mtstate) { ResultRelInfo *targetRelInfo = getTargetResultRelInfo(mtstate); - ResultRelInfo *resultRelInfos = mtstate->resultRelInfo; + ResultRelInfo **resultRelInfos = mtstate->resultRelInfos; TupleDesc outdesc; int numResultRelInfos = mtstate->mt_nplans; int i; @@ -1793,7 +1790,7 @@ ExecSetupChildParentMapForSubplan(ModifyTableState *mtstate) for (i = 0; i < numResultRelInfos; ++i) { mtstate->mt_per_subplan_tupconv_maps[i] = - convert_tuples_by_name(RelationGetDescr(resultRelInfos[i].ri_RelationDesc), + convert_tuples_by_name(RelationGetDescr(resultRelInfos[i]->ri_RelationDesc), outdesc, gettext_noop("could not convert row type")); } @@ -1932,7 +1929,7 @@ ExecModifyTable(PlanState *pstate) } /* Preload local variables */ - resultRelInfo = node->resultRelInfo + node->mt_whichplan; + resultRelInfo = node->resultRelInfos[node->mt_whichplan]; subplanstate = node->mt_plans[node->mt_whichplan]; junkfilter = resultRelInfo->ri_junkFilter; @@ -1969,7 +1966,7 @@ ExecModifyTable(PlanState *pstate) node->mt_whichplan++; if (node->mt_whichplan < node->mt_nplans) { - resultRelInfo++; + resultRelInfo = node->resultRelInfos[node->mt_whichplan]; subplanstate = node->mt_plans[node->mt_whichplan]; junkfilter = resultRelInfo->ri_junkFilter; estate->es_result_relation_info = resultRelInfo; @@ -2153,9 +2150,10 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) ResultRelInfo *resultRelInfo; Plan *subplan; ListCell *l; - int i; + int i, j; Relation rel; bool update_tuple_routing_needed = node->partColsUpdated; + Bitmapset *validsubplans; /* check for unsupported flags */ Assert(!(eflags & (EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK))); @@ -2172,8 +2170,70 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) mtstate->canSetTag = node->canSetTag; mtstate->mt_done = false; + /* If run-time partition pruning is enabled, then setup that up now */ + if (node->part_prune_infos != NIL) + { + PartitionPruning *partprune; + + ExecAssignExprContext(estate, &mtstate->ps); + + partprune = ExecSetupPartitionPruning(&mtstate->ps, + node->part_prune_infos); + + /* + * When there are external params matching the partition key we may be + * able to prune away ModifyTable plans. + */ + if (!bms_is_empty(partprune->extparams)) + { + /* Determine which subplans match the external params */ + validsubplans = ExecFindInitialMatchingSubPlans(partprune, + list_length(node->plans)); + + /* + * If no plans match the given parameters then we must handle this + * case in a special way. The problem here is that code in + * explain.c requires a ModifyTable to have at least one plan in + * order for it to properly determine the Vars in that plan's + * targetlist. We sidestep this issue by just initializing the + * first subplan, but we set the mt_done flag so that we never + * actually bother scanning it. + */ + if (bms_is_empty(validsubplans)) + { + mtstate->mt_done = true; + + /* Mark the first as valid so that it's initialized below */ + validsubplans = bms_make_singleton(0); + } + + nplans = bms_num_members(validsubplans); + } + else + { + /* We'll need to initialize all subplans */ + nplans = list_length(node->plans); + validsubplans = bms_add_range(NULL, 0, nplans - 1); + } + + mtstate->partition_pruning = partprune; + } + else + { + nplans = list_length(node->plans); + + /* + * When run-time partition pruning is not enabled we can just mark + * all plans as valid, they must also all be initialized. + */ + validsubplans = bms_add_range(NULL, 0, nplans - 1); + mtstate->partition_pruning = NULL; + } + + mtstate->mt_plans = (PlanState **) palloc0(sizeof(PlanState *) * nplans); - mtstate->resultRelInfo = estate->es_result_relations + node->resultRelIndex; + mtstate->resultRelInfos = (ResultRelInfo **) + palloc(sizeof(ResultRelInfo *) * nplans); /* If modifying a partitioned table, initialize the root table info */ if (node->rootResultRelIndex >= 0) @@ -2197,12 +2257,20 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) */ saved_resultRelInfo = estate->es_result_relation_info; - resultRelInfo = mtstate->resultRelInfo; - i = 0; + j = i = 0; foreach(l, node->plans) { + if (!bms_is_member(i, validsubplans)) + { + i++; + continue; + } + subplan = (Plan *) lfirst(l); + resultRelInfo = estate->es_result_relations + node->resultRelIndex + i; + mtstate->resultRelInfos[j] = resultRelInfo; + /* Initialize the usesFdwDirectModify flag */ resultRelInfo->ri_usesFdwDirectModify = bms_is_member(i, node->fdwDirectModifyPlans); @@ -2239,7 +2307,7 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) /* Now init the plan for this result rel */ estate->es_result_relation_info = resultRelInfo; - mtstate->mt_plans[i] = ExecInitNode(subplan, estate, eflags); + mtstate->mt_plans[j] = ExecInitNode(subplan, estate, eflags); /* Also let FDWs init themselves for foreign-table result rels */ if (!resultRelInfo->ri_usesFdwDirectModify && @@ -2255,8 +2323,8 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) eflags); } - resultRelInfo++; i++; + j++; } estate->es_result_relation_info = saved_resultRelInfo; @@ -2301,26 +2369,34 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) /* * Initialize any WITH CHECK OPTION constraints if needed. */ - resultRelInfo = mtstate->resultRelInfo; - i = 0; + j = i = 0; foreach(l, node->withCheckOptionLists) { - List *wcoList = (List *) lfirst(l); + List *wcoList; List *wcoExprs = NIL; ListCell *ll; + if (!bms_is_member(i, validsubplans)) + { + i++; + continue; + } + + wcoList = (List *) lfirst(l); + foreach(ll, wcoList) { WithCheckOption *wco = (WithCheckOption *) lfirst(ll); ExprState *wcoExpr = ExecInitQual((List *) wco->qual, - mtstate->mt_plans[i]); + mtstate->mt_plans[j]); wcoExprs = lappend(wcoExprs, wcoExpr); } - + resultRelInfo = mtstate->resultRelInfos[j]; resultRelInfo->ri_WithCheckOptions = wcoList; resultRelInfo->ri_WithCheckOptionExprs = wcoExprs; - resultRelInfo++; + + j++; i++; } @@ -2350,15 +2426,25 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) /* * Build a projection for each result rel. */ - resultRelInfo = mtstate->resultRelInfo; + j = i = 0; foreach(l, node->returningLists) { - List *rlist = (List *) lfirst(l); + List *rlist; + + if (!bms_is_member(i, validsubplans)) + { + i++; + continue; + } + + rlist = (List *) lfirst(l); + resultRelInfo = mtstate->resultRelInfos[j]; resultRelInfo->ri_projectReturning = ExecBuildProjectionInfo(rlist, econtext, slot, &mtstate->ps, resultRelInfo->ri_RelationDesc->rd_att); - resultRelInfo++; + j++; + i++; } } else @@ -2374,7 +2460,7 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) } /* Set the list of arbiter indexes if needed for ON CONFLICT */ - resultRelInfo = mtstate->resultRelInfo; + resultRelInfo = mtstate->resultRelInfos[0]; if (node->onConflictAction != ONCONFLICT_NONE) resultRelInfo->ri_onConflictArbiterIndexes = node->arbiterIndexes; @@ -2528,11 +2614,11 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) if (junk_filter_needed) { - resultRelInfo = mtstate->resultRelInfo; for (i = 0; i < nplans; i++) { JunkFilter *j; + resultRelInfo = mtstate->resultRelInfos[i]; subplan = mtstate->mt_plans[i]->plan; if (operation == CMD_INSERT || operation == CMD_UPDATE) ExecCheckPlanOutput(resultRelInfo->ri_RelationDesc, @@ -2573,13 +2659,12 @@ ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags) } resultRelInfo->ri_junkFilter = j; - resultRelInfo++; } } else { if (operation == CMD_INSERT) - ExecCheckPlanOutput(mtstate->resultRelInfo->ri_RelationDesc, + ExecCheckPlanOutput(mtstate->resultRelInfos[0]->ri_RelationDesc, subplan->targetlist); } } @@ -2626,7 +2711,7 @@ ExecEndModifyTable(ModifyTableState *node) */ for (i = 0; i < node->mt_nplans; i++) { - ResultRelInfo *resultRelInfo = node->resultRelInfo + i; + ResultRelInfo *resultRelInfo = node->resultRelInfos[i]; if (!resultRelInfo->ri_usesFdwDirectModify && resultRelInfo->ri_FdwRoutine != NULL && diff --git a/src/backend/nodes/copyfuncs.c b/src/backend/nodes/copyfuncs.c index 9f2fd52a0b..4b40b0f30f 100644 --- a/src/backend/nodes/copyfuncs.c +++ b/src/backend/nodes/copyfuncs.c @@ -222,6 +222,7 @@ _copyModifyTable(const ModifyTable *from) COPY_NODE_FIELD(onConflictWhere); COPY_SCALAR_FIELD(exclRelRTI); COPY_NODE_FIELD(exclRelTlist); + COPY_NODE_FIELD(part_prune_infos); return newnode; } diff --git a/src/backend/nodes/outfuncs.c b/src/backend/nodes/outfuncs.c index cace730efa..2348fc5369 100644 --- a/src/backend/nodes/outfuncs.c +++ b/src/backend/nodes/outfuncs.c @@ -390,6 +390,7 @@ _outModifyTable(StringInfo str, const ModifyTable *node) WRITE_NODE_FIELD(onConflictWhere); WRITE_UINT_FIELD(exclRelRTI); WRITE_NODE_FIELD(exclRelTlist); + WRITE_NODE_FIELD(part_prune_infos); } static void diff --git a/src/backend/nodes/readfuncs.c b/src/backend/nodes/readfuncs.c index 31fed08627..12da032528 100644 --- a/src/backend/nodes/readfuncs.c +++ b/src/backend/nodes/readfuncs.c @@ -1634,6 +1634,7 @@ _readModifyTable(void) READ_NODE_FIELD(onConflictWhere); READ_UINT_FIELD(exclRelRTI); READ_NODE_FIELD(exclRelTlist); + READ_NODE_FIELD(part_prune_infos); READ_DONE(); } diff --git a/src/backend/optimizer/plan/createplan.c b/src/backend/optimizer/plan/createplan.c index 090b22224c..4437e28902 100644 --- a/src/backend/optimizer/plan/createplan.c +++ b/src/backend/optimizer/plan/createplan.c @@ -291,7 +291,8 @@ static ModifyTable *make_modifytable(PlannerInfo *root, bool partColsUpdated, List *resultRelations, List *subplans, List *withCheckOptionLists, List *returningLists, - List *rowMarks, OnConflictExpr *onconflict, int epqParam); + List *rowMarks, OnConflictExpr *onconflict, int epqParam, + List *partpruneinfos); static GatherMerge *create_gather_merge_plan(PlannerInfo *root, GatherMergePath *best_path); @@ -2480,6 +2481,7 @@ create_modifytable_plan(PlannerInfo *root, ModifyTablePath *best_path) List *subplans = NIL; ListCell *subpaths, *subroots; + List *partpruneinfos = NIL; /* Build the plan for each input path */ forboth(subpaths, best_path->subpaths, @@ -2508,6 +2510,27 @@ create_modifytable_plan(PlannerInfo *root, ModifyTablePath *best_path) subplans = lappend(subplans, subplan); } + if (best_path->partitioned_rels != NIL) + { + int partrelid = linitial_int(best_path->partitioned_rels); + RelOptInfo *rel = root->simple_rel_array[partrelid]; + List *prunequal = NIL; + + prunequal = extract_actual_clauses(rel->baserestrictinfo, false); + + /* + * If any quals exist, then these may be useful to allow us to perform + * further partition pruning during execution. We'll generate a + * PartitionPruneInfo for each partitioned rel to store these quals + * and allow translation of partition indexes into subpath indexes. + */ + if (prunequal != NIL) + partpruneinfos = make_partition_pruneinfo(root, + best_path->partitioned_rels, + best_path->resultRelations, + best_path->subpaths, prunequal); + } + plan = make_modifytable(root, best_path->operation, best_path->canSetTag, @@ -2520,7 +2543,8 @@ create_modifytable_plan(PlannerInfo *root, ModifyTablePath *best_path) best_path->returningLists, best_path->rowMarks, best_path->onconflict, - best_path->epqParam); + best_path->epqParam, + partpruneinfos); copy_generic_path_info(&plan->plan, &best_path->path); @@ -6589,7 +6613,8 @@ make_modifytable(PlannerInfo *root, bool partColsUpdated, List *resultRelations, List *subplans, List *withCheckOptionLists, List *returningLists, - List *rowMarks, OnConflictExpr *onconflict, int epqParam) + List *rowMarks, OnConflictExpr *onconflict, int epqParam, + List *partpruneinfos) { ModifyTable *node = makeNode(ModifyTable); List *fdw_private_list; @@ -6648,6 +6673,7 @@ make_modifytable(PlannerInfo *root, node->returningLists = returningLists; node->rowMarks = rowMarks; node->epqParam = epqParam; + node->part_prune_infos = partpruneinfos; /* * For each result relation that is a foreign table, allow the FDW to diff --git a/src/backend/optimizer/plan/planner.c b/src/backend/optimizer/plan/planner.c index 556efd8ac6..41e0dc4e70 100644 --- a/src/backend/optimizer/plan/planner.c +++ b/src/backend/optimizer/plan/planner.c @@ -1179,6 +1179,7 @@ inheritance_planner(PlannerInfo *root) Query *parent_parse; Bitmapset *parent_relids = bms_make_singleton(top_parentRTindex); PlannerInfo **parent_roots = NULL; + PlannerInfo *partition_root = NULL; Assert(parse->commandType != CMD_INSERT); @@ -1256,6 +1257,32 @@ inheritance_planner(PlannerInfo *root) * the ModifyTable node, if one is needed at all. */ partitioned_relids = bms_make_singleton(top_parentRTindex); + + /* + * For partitioned tables, since we're able to determine the minimum + * set of partitions required much more easily than what we can do + * with an inheritance hierarchy, we invoke the grouping_planner on + * the entire given query in order to determine the minimum set of + * partitions which will be required below. This may mean that we + * invoke the grouping planner far fewer times, as otherwise we'd + * have to invoke it once for each partition. + */ + + /* + * Since the planner tends to scribble on the parse, we must make a + * copy of it. We also must make copies of the PlannerInfo and + * PlannerGlobal since these will also be modified from the call to + * grouping_planner. + */ + partition_root = makeNode(PlannerInfo); + partition_root->glob = makeNode(PlannerGlobal); + + memcpy(partition_root, root, sizeof(PlannerInfo)); + memcpy(partition_root->glob, root->glob, sizeof(PlannerGlobal)); + + partition_root->parse = copyObject(partition_root->parse); + + grouping_planner(partition_root, true, 0.0 /* retrieve all tuples */ ); } /* @@ -1286,6 +1313,21 @@ inheritance_planner(PlannerInfo *root) if (!bms_is_member(appinfo->parent_relid, parent_relids)) continue; + /* + * If the target rel is a partitioned table then skip any child + * partitions which were found to be dummies by the grouping_planner + * call performed above. + */ + if (partition_root) + { + RelOptInfo *rel; + + rel = find_base_rel(partition_root, appinfo->child_relid); + + if (IS_DUMMY_REL(rel)) + continue; + } + /* * expand_inherited_rtentry() always processes a parent before any of * that parent's children, so the parent_root for this relation should @@ -1610,6 +1652,23 @@ inheritance_planner(PlannerInfo *root) Assert(list_length(partitioned_rels) >= 1); } + /* + * The individual grouping_planner calls per partition above performed + * no planning on the actual partitioned tables, however, in order to + * allow partition pruning at run-time we must know the baserestrictinfo + * of each partition. We simply replace the RelOptInfos from the initial + * full plan which was generated and replace the non-complete RelOptInfos + * which are stored in root. + */ + if (partition_root) + { + int i; + + i = -1; + while ((i = bms_next_member(partitioned_relids, i)) >= 0) + root->simple_rel_array[i] = partition_root->simple_rel_array[i]; + } + /* Create Path representing a ModifyTable to do the UPDATE/DELETE work */ add_path(final_rel, (Path *) create_modifytable_path(root, final_rel, diff --git a/src/include/nodes/execnodes.h b/src/include/nodes/execnodes.h index 681779d42f..65de9ef6f8 100644 --- a/src/include/nodes/execnodes.h +++ b/src/include/nodes/execnodes.h @@ -1012,6 +1012,8 @@ typedef struct ProjectSetState MemoryContext argcontext; /* context for SRF arguments */ } ProjectSetState; +struct PartitionPruning; + /* ---------------- * ModifyTableState information * ---------------- @@ -1025,7 +1027,7 @@ typedef struct ModifyTableState PlanState **mt_plans; /* subplans (one per target rel) */ int mt_nplans; /* number of plans in the array */ int mt_whichplan; /* which one is being executed (0..n-1) */ - ResultRelInfo *resultRelInfo; /* per-subplan target relations */ + ResultRelInfo **resultRelInfos; /* per-subplan target relations */ ResultRelInfo *rootResultRelInfo; /* root target relation (partitioned * table root) */ List **mt_arowmarks; /* per-subplan ExecAuxRowMark lists */ @@ -1046,6 +1048,12 @@ typedef struct ModifyTableState /* Per plan map for tuple conversion from child to root */ TupleConversionMap **mt_per_subplan_tupconv_maps; + + /* + * Details required to allow partitions to be eliminated from the scan, or + * NULL if not possible. + */ + struct PartitionPruning *partition_pruning; } ModifyTableState; /* ---------------- @@ -1067,7 +1075,6 @@ struct AppendState; typedef struct AppendState AppendState; struct ParallelAppendState; typedef struct ParallelAppendState ParallelAppendState; -struct PartitionPruning; struct AppendState { diff --git a/src/include/nodes/plannodes.h b/src/include/nodes/plannodes.h index c90c763a86..d13e5117c9 100644 --- a/src/include/nodes/plannodes.h +++ b/src/include/nodes/plannodes.h @@ -238,6 +238,8 @@ typedef struct ModifyTable Node *onConflictWhere; /* WHERE for ON CONFLICT UPDATE */ Index exclRelRTI; /* RTI of the EXCLUDED pseudo relation */ List *exclRelTlist; /* tlist of the EXCLUDED pseudo relation */ + List *part_prune_infos; /* Mapping details for run-time subplan + * pruning, one per partitioned_rels */ } ModifyTable; /* ---------------- -- 2.16.2.windows.1