PostgreSQL Source Code git master
Loading...
Searching...
No Matches
planner.h File Reference
#include "nodes/pathnodes.h"
#include "nodes/plannodes.h"
Include dependency graph for planner.h:
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Typedefs

typedef struct ExplainState ExplainState
 
typedef PlannedStmt *(* planner_hook_type) (Query *parse, const char *query_string, int cursorOptions, ParamListInfo boundParams, ExplainState *es)
 
typedef void(* planner_setup_hook_type) (PlannerGlobal *glob, Query *parse, const char *query_string, int cursorOptions, double *tuple_fraction, ExplainState *es)
 
typedef void(* planner_shutdown_hook_type) (PlannerGlobal *glob, Query *parse, const char *query_string, PlannedStmt *pstmt)
 
typedef void(* create_upper_paths_hook_type) (PlannerInfo *root, UpperRelationKind stage, RelOptInfo *input_rel, RelOptInfo *output_rel, void *extra)
 

Functions

PlannedStmtstandard_planner (Query *parse, const char *query_string, int cursorOptions, ParamListInfo boundParams, ExplainState *es)
 
PlannerInfosubquery_planner (PlannerGlobal *glob, Query *parse, char *plan_name, PlannerInfo *parent_root, PlannerInfo *alternative_root, bool hasRecursion, double tuple_fraction, SetOperationStmt *setops)
 
RowMarkType select_rowmark_type (RangeTblEntry *rte, LockClauseStrength strength)
 
bool limit_needed (Query *parse)
 
void mark_partial_aggref (Aggref *agg, AggSplit aggsplit)
 
Pathget_cheapest_fractional_path (RelOptInfo *rel, double tuple_fraction)
 
Exprpreprocess_phv_expression (PlannerInfo *root, Expr *expr)
 
RelOptInfocreate_unique_paths (PlannerInfo *root, RelOptInfo *rel, SpecialJoinInfo *sjinfo)
 
charchoose_plan_name (PlannerGlobal *glob, const char *name, bool always_number)
 

Variables

PGDLLIMPORT planner_hook_type planner_hook
 
PGDLLIMPORT planner_setup_hook_type planner_setup_hook
 
PGDLLIMPORT planner_shutdown_hook_type planner_shutdown_hook
 
PGDLLIMPORT create_upper_paths_hook_type create_upper_paths_hook
 

Typedef Documentation

◆ create_upper_paths_hook_type

typedef void(* create_upper_paths_hook_type) (PlannerInfo *root, UpperRelationKind stage, RelOptInfo *input_rel, RelOptInfo *output_rel, void *extra)

Definition at line 50 of file planner.h.

◆ ExplainState

Definition at line 25 of file planner.h.

◆ planner_hook_type

typedef PlannedStmt *(* planner_hook_type) (Query *parse, const char *query_string, int cursorOptions, ParamListInfo boundParams, ExplainState *es)

Definition at line 28 of file planner.h.

◆ planner_setup_hook_type

typedef void(* planner_setup_hook_type) (PlannerGlobal *glob, Query *parse, const char *query_string, int cursorOptions, double *tuple_fraction, ExplainState *es)

Definition at line 36 of file planner.h.

◆ planner_shutdown_hook_type

typedef void(* planner_shutdown_hook_type) (PlannerGlobal *glob, Query *parse, const char *query_string, PlannedStmt *pstmt)

Definition at line 44 of file planner.h.

Function Documentation

◆ choose_plan_name()

char * choose_plan_name ( PlannerGlobal glob,
const char name,
bool  always_number 
)
extern

Definition at line 9232 of file planner.c.

9233{
9234 unsigned n;
9235
9236 /*
9237 * If a numeric suffix is not required, then search the list of
9238 * previously-assigned names for a match. If none is found, then we can
9239 * use the provided name without modification.
9240 */
9241 if (!always_number)
9242 {
9243 bool found = false;
9244
9245 foreach_ptr(char, subplan_name, glob->subplanNames)
9246 {
9247 if (strcmp(subplan_name, name) == 0)
9248 {
9249 found = true;
9250 break;
9251 }
9252 }
9253
9254 if (!found)
9255 {
9256 /* pstrdup here is just to avoid cast-away-const */
9257 char *chosen_name = pstrdup(name);
9258
9259 glob->subplanNames = lappend(glob->subplanNames, chosen_name);
9260 return chosen_name;
9261 }
9262 }
9263
9264 /*
9265 * If a numeric suffix is required or if the un-suffixed name is already
9266 * in use, then loop until we find a positive integer that produces a
9267 * novel name.
9268 */
9269 for (n = 1; true; ++n)
9270 {
9271 char *proposed_name = psprintf("%s_%u", name, n);
9272 bool found = false;
9273
9274 foreach_ptr(char, subplan_name, glob->subplanNames)
9275 {
9277 {
9278 found = true;
9279 break;
9280 }
9281 }
9282
9283 if (!found)
9284 {
9285 glob->subplanNames = lappend(glob->subplanNames, proposed_name);
9286 return proposed_name;
9287 }
9288
9290 }
9291}
List * lappend(List *list, void *datum)
Definition list.c:339
char * pstrdup(const char *in)
Definition mcxt.c:1781
void pfree(void *pointer)
Definition mcxt.c:1616
#define foreach_ptr(type, var, lst)
Definition pg_list.h:501
static int fb(int x)
char * psprintf(const char *fmt,...)
Definition psprintf.c:43
const char * name

References fb(), foreach_ptr, lappend(), name, pfree(), psprintf(), and pstrdup().

Referenced by build_minmax_path(), make_subplan(), recurse_set_operations(), set_subquery_pathlist(), and SS_process_ctes().

◆ create_unique_paths()

RelOptInfo * create_unique_paths ( PlannerInfo root,
RelOptInfo rel,
SpecialJoinInfo sjinfo 
)
extern

Definition at line 8680 of file planner.c.

8681{
8682 RelOptInfo *unique_rel;
8684 List *groupClause = NIL;
8685 MemoryContext oldcontext;
8686
8687 /* Caller made a mistake if SpecialJoinInfo is the wrong one */
8688 Assert(sjinfo->jointype == JOIN_SEMI);
8689 Assert(bms_equal(rel->relids, sjinfo->syn_righthand));
8690
8691 /* If result already cached, return it */
8692 if (rel->unique_rel)
8693 return rel->unique_rel;
8694
8695 /* If it's not possible to unique-ify, return NULL */
8696 if (!(sjinfo->semi_can_btree || sjinfo->semi_can_hash))
8697 return NULL;
8698
8699 /*
8700 * Punt if this is a child relation and we failed to build a unique-ified
8701 * relation for its parent. This can happen if all the RHS columns were
8702 * found to be equated to constants when unique-ifying the parent table,
8703 * leaving no columns to unique-ify.
8704 */
8705 if (IS_OTHER_REL(rel) && rel->top_parent->unique_rel == NULL)
8706 return NULL;
8707
8708 /*
8709 * When called during GEQO join planning, we are in a short-lived memory
8710 * context. We must make sure that the unique rel and any subsidiary data
8711 * structures created for a baserel survive the GEQO cycle, else the
8712 * baserel is trashed for future GEQO cycles. On the other hand, when we
8713 * are creating those for a joinrel during GEQO, we don't want them to
8714 * clutter the main planning context. Upshot is that the best solution is
8715 * to explicitly allocate memory in the same context the given RelOptInfo
8716 * is in.
8717 */
8719
8720 unique_rel = makeNode(RelOptInfo);
8721 memcpy(unique_rel, rel, sizeof(RelOptInfo));
8722
8723 /*
8724 * clear path info
8725 */
8726 unique_rel->pathlist = NIL;
8727 unique_rel->ppilist = NIL;
8728 unique_rel->partial_pathlist = NIL;
8729 unique_rel->cheapest_startup_path = NULL;
8730 unique_rel->cheapest_total_path = NULL;
8731 unique_rel->cheapest_parameterized_paths = NIL;
8732
8733 /*
8734 * Build the target list for the unique rel. We also build the pathkeys
8735 * that represent the ordering requirements for the sort-based
8736 * implementation, and the list of SortGroupClause nodes that represent
8737 * the columns to be grouped on for the hash-based implementation.
8738 *
8739 * For a child rel, we can construct these fields from those of its
8740 * parent.
8741 */
8742 if (IS_OTHER_REL(rel))
8743 {
8746
8747 parent_unique_target = rel->top_parent->unique_rel->reltarget;
8748
8750
8751 /* Translate the target expressions */
8752 child_unique_target->exprs = (List *)
8754 (Node *) parent_unique_target->exprs,
8755 rel,
8756 rel->top_parent);
8757
8758 unique_rel->reltarget = child_unique_target;
8759
8760 sortPathkeys = rel->top_parent->unique_pathkeys;
8761 groupClause = rel->top_parent->unique_groupclause;
8762 }
8763 else
8764 {
8765 List *newtlist;
8766 int nextresno;
8767 List *sortList = NIL;
8768 ListCell *lc1;
8769 ListCell *lc2;
8770
8771 /*
8772 * The values we are supposed to unique-ify may be expressions in the
8773 * variables of the input rel's targetlist. We have to add any such
8774 * expressions to the unique rel's targetlist.
8775 *
8776 * To complicate matters, some of the values to be unique-ified may be
8777 * known redundant by the EquivalenceClass machinery (e.g., because
8778 * they have been equated to constants). There is no need to compare
8779 * such values during unique-ification, and indeed we had better not
8780 * try because the Vars involved may not have propagated as high as
8781 * the semijoin's level. We use make_pathkeys_for_sortclauses to
8782 * detect such cases, which is a tad inefficient but it doesn't seem
8783 * worth building specialized infrastructure for this.
8784 */
8787
8788 forboth(lc1, sjinfo->semi_rhs_exprs, lc2, sjinfo->semi_operators)
8789 {
8790 Expr *uniqexpr = lfirst(lc1);
8792 Oid sortop;
8794 bool made_tle = false;
8795
8797 if (!tle)
8798 {
8800 nextresno,
8801 NULL,
8802 false);
8804 nextresno++;
8805 made_tle = true;
8806 }
8807
8808 /*
8809 * Try to build an ORDER BY list to sort the input compatibly. We
8810 * do this for each sortable clause even when the clauses are not
8811 * all sortable, so that we can detect clauses that are redundant
8812 * according to the pathkey machinery.
8813 */
8815 if (OidIsValid(sortop))
8816 {
8817 Oid eqop;
8819
8820 /*
8821 * The Unique node will need equality operators. Normally
8822 * these are the same as the IN clause operators, but if those
8823 * are cross-type operators then the equality operators are
8824 * the ones for the IN clause operators' RHS datatype.
8825 */
8826 eqop = get_equality_op_for_ordering_op(sortop, NULL);
8827 if (!OidIsValid(eqop)) /* shouldn't happen */
8828 elog(ERROR, "could not find equality operator for ordering operator %u",
8829 sortop);
8830
8832 sortcl->tleSortGroupRef = assignSortGroupRef(tle, newtlist);
8833 sortcl->eqop = eqop;
8834 sortcl->sortop = sortop;
8835 sortcl->reverse_sort = false;
8836 sortcl->nulls_first = false;
8837 sortcl->hashable = false; /* no need to make this accurate */
8839
8840 /*
8841 * At each step, convert the SortGroupClause list to pathkey
8842 * form. If the just-added SortGroupClause is redundant, the
8843 * result will be shorter than the SortGroupClause list.
8844 */
8846 newtlist);
8848 {
8849 /* Drop the redundant SortGroupClause */
8852 /* Undo tlist addition, if we made one */
8853 if (made_tle)
8854 {
8856 nextresno--;
8857 }
8858 /* We need not consider this clause for hashing, either */
8859 continue;
8860 }
8861 }
8862 else if (sjinfo->semi_can_btree) /* shouldn't happen */
8863 elog(ERROR, "could not find ordering operator for equality operator %u",
8864 in_oper);
8865
8866 if (sjinfo->semi_can_hash)
8867 {
8868 /* Create a GROUP BY list for the Agg node to use */
8869 Oid eq_oper;
8871
8872 /*
8873 * Get the hashable equality operators for the Agg node to
8874 * use. Normally these are the same as the IN clause
8875 * operators, but if those are cross-type operators then the
8876 * equality operators are the ones for the IN clause
8877 * operators' RHS datatype.
8878 */
8880 elog(ERROR, "could not find compatible hash operator for operator %u",
8881 in_oper);
8882
8884 groupcl->tleSortGroupRef = assignSortGroupRef(tle, newtlist);
8885 groupcl->eqop = eq_oper;
8886 groupcl->sortop = sortop;
8887 groupcl->reverse_sort = false;
8888 groupcl->nulls_first = false;
8889 groupcl->hashable = true;
8890 groupClause = lappend(groupClause, groupcl);
8891 }
8892 }
8893
8894 /*
8895 * Done building the sortPathkeys and groupClause. But the
8896 * sortPathkeys are bogus if not all the clauses were sortable.
8897 */
8898 if (!sjinfo->semi_can_btree)
8899 sortPathkeys = NIL;
8900
8901 /*
8902 * It can happen that all the RHS columns are equated to constants.
8903 * We'd have to do something special to unique-ify in that case, and
8904 * it's such an unlikely-in-the-real-world case that it's not worth
8905 * the effort. So just punt if we found no columns to unique-ify.
8906 */
8907 if (sortPathkeys == NIL && groupClause == NIL)
8908 {
8909 MemoryContextSwitchTo(oldcontext);
8910 return NULL;
8911 }
8912
8913 /* Convert the required targetlist back to PathTarget form */
8914 unique_rel->reltarget = create_pathtarget(root, newtlist);
8915 }
8916
8917 /* build unique paths based on input rel's pathlist */
8918 create_final_unique_paths(root, rel, sortPathkeys, groupClause,
8919 sjinfo, unique_rel);
8920
8921 /* build unique paths based on input rel's partial_pathlist */
8923 sjinfo, unique_rel);
8924
8925 /* Now choose the best path(s) */
8926 set_cheapest(unique_rel);
8927
8928 /*
8929 * There shouldn't be any partial paths for the unique relation;
8930 * otherwise, we won't be able to properly guarantee uniqueness.
8931 */
8932 Assert(unique_rel->partial_pathlist == NIL);
8933
8934 /* Cache the result */
8935 rel->unique_rel = unique_rel;
8937 rel->unique_groupclause = groupClause;
8938
8939 MemoryContextSwitchTo(oldcontext);
8940
8941 return unique_rel;
8942}
Node * adjust_appendrel_attrs_multilevel(PlannerInfo *root, Node *node, RelOptInfo *childrel, RelOptInfo *parentrel)
Definition appendinfo.c:597
bool bms_equal(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:142
#define Assert(condition)
Definition c.h:943
#define OidIsValid(objectId)
Definition c.h:858
memcpy(sums, checksumBaseOffsets, sizeof(checksumBaseOffsets))
#define ERROR
Definition elog.h:40
#define elog(elevel,...)
Definition elog.h:228
List * list_delete_last(List *list)
Definition list.c:957
bool get_compatible_hash_operators(Oid opno, Oid *lhs_opno, Oid *rhs_opno)
Definition lsyscache.c:477
Oid get_equality_op_for_ordering_op(Oid opno, bool *reverse)
Definition lsyscache.c:326
Oid get_ordering_op_for_equality_op(Oid opno, bool use_lhs_type)
Definition lsyscache.c:364
TargetEntry * makeTargetEntry(Expr *expr, AttrNumber resno, char *resname, bool resjunk)
Definition makefuncs.c:289
MemoryContext GetMemoryChunkContext(void *pointer)
Definition mcxt.c:756
#define makeNode(_type_)
Definition nodes.h:161
@ JOIN_SEMI
Definition nodes.h:317
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
Definition palloc.h:124
Index assignSortGroupRef(TargetEntry *tle, List *tlist)
List * make_pathkeys_for_sortclauses(PlannerInfo *root, List *sortclauses, List *tlist)
Definition pathkeys.c:1336
void set_cheapest(RelOptInfo *parent_rel)
Definition pathnode.c:268
#define IS_OTHER_REL(rel)
Definition pathnodes.h:1004
#define lfirst(lc)
Definition pg_list.h:172
static int list_length(const List *l)
Definition pg_list.h:152
#define NIL
Definition pg_list.h:68
#define forboth(cell1, list1, cell2, list2)
Definition pg_list.h:550
#define lfirst_oid(lc)
Definition pg_list.h:174
static void create_final_unique_paths(PlannerInfo *root, RelOptInfo *input_rel, List *sortPathkeys, List *groupClause, SpecialJoinInfo *sjinfo, RelOptInfo *unique_rel)
Definition planner.c:8949
static void create_partial_unique_paths(PlannerInfo *root, RelOptInfo *input_rel, List *sortPathkeys, List *groupClause, SpecialJoinInfo *sjinfo, RelOptInfo *unique_rel)
Definition planner.c:9074
unsigned int Oid
tree ctl root
Definition radixtree.h:1857
Definition pg_list.h:54
Definition nodes.h:135
List * ppilist
Definition pathnodes.h:1051
Relids relids
Definition pathnodes.h:1021
struct PathTarget * reltarget
Definition pathnodes.h:1045
List * unique_pathkeys
Definition pathnodes.h:1134
List * cheapest_parameterized_paths
Definition pathnodes.h:1055
List * pathlist
Definition pathnodes.h:1050
struct Path * cheapest_startup_path
Definition pathnodes.h:1053
struct Path * cheapest_total_path
Definition pathnodes.h:1054
List * unique_groupclause
Definition pathnodes.h:1136
List * partial_pathlist
Definition pathnodes.h:1052
struct RelOptInfo * unique_rel
Definition pathnodes.h:1132
List * semi_rhs_exprs
Definition pathnodes.h:3241
JoinType jointype
Definition pathnodes.h:3230
Relids syn_righthand
Definition pathnodes.h:3229
List * semi_operators
Definition pathnodes.h:3240
TargetEntry * tlist_member(Expr *node, List *targetlist)
Definition tlist.c:88
List * make_tlist_from_pathtarget(PathTarget *target)
Definition tlist.c:633
PathTarget * copy_pathtarget(PathTarget *src)
Definition tlist.c:666
#define create_pathtarget(root, tlist)
Definition tlist.h:58

References adjust_appendrel_attrs_multilevel(), Assert, assignSortGroupRef(), bms_equal(), RelOptInfo::cheapest_parameterized_paths, RelOptInfo::cheapest_startup_path, RelOptInfo::cheapest_total_path, copy_pathtarget(), create_final_unique_paths(), create_partial_unique_paths(), create_pathtarget, elog, ERROR, fb(), forboth, get_compatible_hash_operators(), get_equality_op_for_ordering_op(), get_ordering_op_for_equality_op(), GetMemoryChunkContext(), IS_OTHER_REL, JOIN_SEMI, SpecialJoinInfo::jointype, lappend(), lfirst, lfirst_oid, list_delete_last(), list_length(), make_pathkeys_for_sortclauses(), make_tlist_from_pathtarget(), makeNode, makeTargetEntry(), memcpy(), MemoryContextSwitchTo(), NIL, OidIsValid, RelOptInfo::partial_pathlist, RelOptInfo::pathlist, RelOptInfo::ppilist, RelOptInfo::relids, RelOptInfo::reltarget, root, SpecialJoinInfo::semi_can_btree, SpecialJoinInfo::semi_can_hash, SpecialJoinInfo::semi_operators, SpecialJoinInfo::semi_rhs_exprs, set_cheapest(), SpecialJoinInfo::syn_righthand, tlist_member(), RelOptInfo::unique_groupclause, RelOptInfo::unique_pathkeys, and RelOptInfo::unique_rel.

Referenced by join_is_legal(), and populate_joinrel_with_paths().

◆ get_cheapest_fractional_path()

Path * get_cheapest_fractional_path ( RelOptInfo rel,
double  tuple_fraction 
)
extern

Definition at line 6865 of file planner.c.

6866{
6868 ListCell *l;
6869
6870 /* If all tuples will be retrieved, just return the cheapest-total path */
6871 if (tuple_fraction <= 0.0)
6872 return best_path;
6873
6874 /* Convert absolute # of tuples to a fraction; no need to clamp to 0..1 */
6875 if (tuple_fraction >= 1.0 && best_path->rows > 0)
6876 tuple_fraction /= best_path->rows;
6877
6878 foreach(l, rel->pathlist)
6879 {
6880 Path *path = (Path *) lfirst(l);
6881
6882 if (path->param_info)
6883 continue;
6884
6885 if (path == rel->cheapest_total_path ||
6886 compare_fractional_path_costs(best_path, path, tuple_fraction) <= 0)
6887 continue;
6888
6889 best_path = path;
6890 }
6891
6892 return best_path;
6893}
int compare_fractional_path_costs(Path *path1, Path *path2, double fraction)
Definition pathnode.c:123

References RelOptInfo::cheapest_total_path, compare_fractional_path_costs(), fb(), lfirst, and RelOptInfo::pathlist.

Referenced by add_paths_to_append_rel(), make_subplan(), and standard_planner().

◆ limit_needed()

bool limit_needed ( Query parse)
extern

Definition at line 3049 of file planner.c.

3050{
3051 Node *node;
3052
3053 node = parse->limitCount;
3054 if (node)
3055 {
3056 if (IsA(node, Const))
3057 {
3058 /* NULL indicates LIMIT ALL, ie, no limit */
3059 if (!((Const *) node)->constisnull)
3060 return true; /* LIMIT with a constant value */
3061 }
3062 else
3063 return true; /* non-constant LIMIT */
3064 }
3065
3066 node = parse->limitOffset;
3067 if (node)
3068 {
3069 if (IsA(node, Const))
3070 {
3071 /* Treat NULL as no offset; the executor would too */
3072 if (!((Const *) node)->constisnull)
3073 {
3074 int64 offset = DatumGetInt64(((Const *) node)->constvalue);
3075
3076 if (offset != 0)
3077 return true; /* OFFSET with a nonzero value */
3078 }
3079 }
3080 else
3081 return true; /* non-constant OFFSET */
3082 }
3083
3084 return false; /* don't need a Limit plan node */
3085}
int64_t int64
Definition c.h:621
void parse(int)
Definition parse.c:49
#define IsA(nodeptr, _type_)
Definition nodes.h:164
static int64 DatumGetInt64(Datum X)
Definition postgres.h:416

References DatumGetInt64(), fb(), IsA, and parse().

Referenced by grouping_planner(), and set_rel_consider_parallel().

◆ mark_partial_aggref()

void mark_partial_aggref ( Aggref agg,
AggSplit  aggsplit 
)
extern

Definition at line 6026 of file planner.c.

6027{
6028 /* aggtranstype should be computed by this point */
6029 Assert(OidIsValid(agg->aggtranstype));
6030 /* ... but aggsplit should still be as the parser left it */
6031 Assert(agg->aggsplit == AGGSPLIT_SIMPLE);
6032
6033 /* Mark the Aggref with the intended partial-aggregation mode */
6034 agg->aggsplit = aggsplit;
6035
6036 /*
6037 * Adjust result type if needed. Normally, a partial aggregate returns
6038 * the aggregate's transition type; but if that's INTERNAL and we're
6039 * serializing, it returns BYTEA instead.
6040 */
6041 if (DO_AGGSPLIT_SKIPFINAL(aggsplit))
6042 {
6043 if (agg->aggtranstype == INTERNALOID && DO_AGGSPLIT_SERIALIZE(aggsplit))
6044 agg->aggtype = BYTEAOID;
6045 else
6046 agg->aggtype = agg->aggtranstype;
6047 }
6048}
#define DO_AGGSPLIT_SKIPFINAL(as)
Definition nodes.h:396
#define DO_AGGSPLIT_SERIALIZE(as)
Definition nodes.h:397
@ AGGSPLIT_SIMPLE
Definition nodes.h:387

References AGGSPLIT_SIMPLE, Assert, DO_AGGSPLIT_SERIALIZE, DO_AGGSPLIT_SKIPFINAL, fb(), and OidIsValid.

Referenced by convert_combining_aggrefs(), create_rel_agg_info(), and make_partial_grouping_target().

◆ preprocess_phv_expression()

Expr * preprocess_phv_expression ( PlannerInfo root,
Expr expr 
)
extern

Definition at line 1688 of file planner.c.

1689{
1690 return (Expr *) preprocess_expression(root, (Node *) expr, EXPRKIND_PHV);
1691}
#define EXPRKIND_PHV
Definition planner.c:95
static Node * preprocess_expression(PlannerInfo *root, Node *expr, int kind)
Definition planner.c:1407

References EXPRKIND_PHV, preprocess_expression(), and root.

Referenced by extract_lateral_references().

◆ select_rowmark_type()

RowMarkType select_rowmark_type ( RangeTblEntry rte,
LockClauseStrength  strength 
)
extern

Definition at line 2798 of file planner.c.

2799{
2800 if (rte->rtekind != RTE_RELATION)
2801 {
2802 /* If it's not a table at all, use ROW_MARK_COPY */
2803 return ROW_MARK_COPY;
2804 }
2805 else if (rte->relkind == RELKIND_FOREIGN_TABLE)
2806 {
2807 /* Let the FDW select the rowmark type, if it wants to */
2808 FdwRoutine *fdwroutine = GetFdwRoutineByRelId(rte->relid);
2809
2810 if (fdwroutine->GetForeignRowMarkType != NULL)
2811 return fdwroutine->GetForeignRowMarkType(rte, strength);
2812 /* Otherwise, use ROW_MARK_COPY by default */
2813 return ROW_MARK_COPY;
2814 }
2815 else
2816 {
2817 /* Regular table, apply the appropriate lock type */
2818 switch (strength)
2819 {
2820 case LCS_NONE:
2821
2822 /*
2823 * We don't need a tuple lock, only the ability to re-fetch
2824 * the row.
2825 */
2826 return ROW_MARK_REFERENCE;
2827 break;
2828 case LCS_FORKEYSHARE:
2829 return ROW_MARK_KEYSHARE;
2830 break;
2831 case LCS_FORSHARE:
2832 return ROW_MARK_SHARE;
2833 break;
2834 case LCS_FORNOKEYUPDATE:
2836 break;
2837 case LCS_FORUPDATE:
2838 return ROW_MARK_EXCLUSIVE;
2839 break;
2840 }
2841 elog(ERROR, "unrecognized LockClauseStrength %d", (int) strength);
2842 return ROW_MARK_EXCLUSIVE; /* keep compiler quiet */
2843 }
2844}
FdwRoutine * GetFdwRoutineByRelId(Oid relid)
Definition foreign.c:451
@ LCS_FORUPDATE
Definition lockoptions.h:28
@ LCS_NONE
Definition lockoptions.h:23
@ LCS_FORSHARE
Definition lockoptions.h:26
@ LCS_FORKEYSHARE
Definition lockoptions.h:25
@ LCS_FORNOKEYUPDATE
Definition lockoptions.h:27
@ RTE_RELATION
@ ROW_MARK_COPY
Definition plannodes.h:1562
@ ROW_MARK_REFERENCE
Definition plannodes.h:1561
@ ROW_MARK_SHARE
Definition plannodes.h:1559
@ ROW_MARK_EXCLUSIVE
Definition plannodes.h:1557
@ ROW_MARK_NOKEYEXCLUSIVE
Definition plannodes.h:1558
@ ROW_MARK_KEYSHARE
Definition plannodes.h:1560
GetForeignRowMarkType_function GetForeignRowMarkType
Definition fdwapi.h:251

References elog, ERROR, fb(), GetFdwRoutineByRelId(), FdwRoutine::GetForeignRowMarkType, LCS_FORKEYSHARE, LCS_FORNOKEYUPDATE, LCS_FORSHARE, LCS_FORUPDATE, LCS_NONE, ROW_MARK_COPY, ROW_MARK_EXCLUSIVE, ROW_MARK_KEYSHARE, ROW_MARK_NOKEYEXCLUSIVE, ROW_MARK_REFERENCE, ROW_MARK_SHARE, and RTE_RELATION.

Referenced by expand_single_inheritance_child(), and preprocess_rowmarks().

◆ standard_planner()

PlannedStmt * standard_planner ( Query parse,
const char query_string,
int  cursorOptions,
ParamListInfo  boundParams,
ExplainState es 
)
extern

Definition at line 351 of file planner.c.

353{
355 PlannerGlobal *glob;
356 double tuple_fraction;
360 Plan *top_plan;
361 ListCell *lp,
362 *lr,
363 *lc;
364
365 /*
366 * Set up global state for this planner invocation. This data is needed
367 * across all levels of sub-Query that might exist in the given command,
368 * so we keep it in a separate struct that's linked to by each per-Query
369 * PlannerInfo.
370 */
371 glob = makeNode(PlannerGlobal);
372
373 glob->boundParams = boundParams;
374 glob->subplans = NIL;
375 glob->subpaths = NIL;
376 glob->subroots = NIL;
377 glob->rewindPlanIDs = NULL;
378 glob->finalrtable = NIL;
379 glob->allRelids = NULL;
380 glob->prunableRelids = NULL;
381 glob->finalrteperminfos = NIL;
382 glob->finalrowmarks = NIL;
383 glob->resultRelations = NIL;
384 glob->appendRelations = NIL;
385 glob->partPruneInfos = NIL;
386 glob->relationOids = NIL;
387 glob->invalItems = NIL;
388 glob->paramExecTypes = NIL;
389 glob->lastPHId = 0;
390 glob->lastRowMarkId = 0;
391 glob->lastPlanNodeId = 0;
392 glob->transientPlan = false;
393 glob->dependsOnRole = false;
394 glob->partition_directory = NULL;
395 glob->rel_notnullatts_hash = NULL;
396
397 /*
398 * Assess whether it's feasible to use parallel mode for this query. We
399 * can't do this in a standalone backend, or if the command will try to
400 * modify any data, or if this is a cursor operation, or if GUCs are set
401 * to values that don't permit parallelism, or if parallel-unsafe
402 * functions are present in the query tree.
403 *
404 * (Note that we do allow CREATE TABLE AS, SELECT INTO, and CREATE
405 * MATERIALIZED VIEW to use parallel plans, but this is safe only because
406 * the command is writing into a completely new table which workers won't
407 * be able to see. If the workers could see the table, the fact that
408 * group locking would cause them to ignore the leader's heavyweight GIN
409 * page locks would make this unsafe. We'll have to fix that somehow if
410 * we want to allow parallel inserts in general; updates and deletes have
411 * additional problems especially around combo CIDs.)
412 *
413 * For now, we don't try to use parallel mode if we're running inside a
414 * parallel worker. We might eventually be able to relax this
415 * restriction, but for now it seems best not to have parallel workers
416 * trying to create their own parallel workers.
417 */
418 if ((cursorOptions & CURSOR_OPT_PARALLEL_OK) != 0 &&
420 parse->commandType == CMD_SELECT &&
421 !parse->hasModifyingCTE &&
424 {
425 /* all the cheap tests pass, so scan the query tree */
428 }
429 else
430 {
431 /* skip the query tree scan, just assume it's unsafe */
433 glob->parallelModeOK = false;
434 }
435
436 /*
437 * glob->parallelModeNeeded is normally set to false here and changed to
438 * true during plan creation if a Gather or Gather Merge plan is actually
439 * created (cf. create_gather_plan, create_gather_merge_plan).
440 *
441 * However, if debug_parallel_query = on or debug_parallel_query =
442 * regress, then we impose parallel mode whenever it's safe to do so, even
443 * if the final plan doesn't use parallelism. It's not safe to do so if
444 * the query contains anything parallel-unsafe; parallelModeOK will be
445 * false in that case. Note that parallelModeOK can't change after this
446 * point. Otherwise, everything in the query is either parallel-safe or
447 * parallel-restricted, and in either case it should be OK to impose
448 * parallel-mode restrictions. If that ends up breaking something, then
449 * either some function the user included in the query is incorrectly
450 * labeled as parallel-safe or parallel-restricted when in reality it's
451 * parallel-unsafe, or else the query planner itself has a bug.
452 */
453 glob->parallelModeNeeded = glob->parallelModeOK &&
455
456 /* Determine what fraction of the plan is likely to be scanned */
457 if (cursorOptions & CURSOR_OPT_FAST_PLAN)
458 {
459 /*
460 * We have no real idea how many tuples the user will ultimately FETCH
461 * from a cursor, but it is often the case that he doesn't want 'em
462 * all, or would prefer a fast-start plan anyway so that he can
463 * process some of the tuples sooner. Use a GUC parameter to decide
464 * what fraction to optimize for.
465 */
466 tuple_fraction = cursor_tuple_fraction;
467
468 /*
469 * We document cursor_tuple_fraction as simply being a fraction, which
470 * means the edge cases 0 and 1 have to be treated specially here. We
471 * convert 1 to 0 ("all the tuples") and 0 to a very small fraction.
472 */
473 if (tuple_fraction >= 1.0)
474 tuple_fraction = 0.0;
475 else if (tuple_fraction <= 0.0)
476 tuple_fraction = 1e-10;
477 }
478 else
479 {
480 /* Default assumption is we need all the tuples */
481 tuple_fraction = 0.0;
482 }
483
484 /*
485 * Compute the initial path generation strategy mask.
486 *
487 * Some strategies, such as PGS_FOREIGNJOIN, have no corresponding enable_*
488 * GUC, and so the corresponding bits are always set in the default
489 * strategy mask.
490 *
491 * It may seem surprising that enable_indexscan sets both PGS_INDEXSCAN
492 * and PGS_INDEXONLYSCAN. However, the historical behavior of this GUC
493 * corresponds to this exactly: enable_indexscan=off disables both
494 * index-scan and index-only scan paths, whereas enable_indexonlyscan=off
495 * converts the index-only scan paths that we would have considered into
496 * index scan paths.
497 */
500 if (enable_tidscan)
502 if (enable_seqscan)
511 {
513 if (enable_material)
515 }
516 if (enable_nestloop)
517 {
519 if (enable_material)
521 if (enable_memoize)
523 }
524 if (enable_hashjoin)
530
531 /* Allow plugins to take control after we've initialized "glob" */
533 (*planner_setup_hook) (glob, parse, query_string, cursorOptions,
534 &tuple_fraction, es);
535
536 /* primary planning entry point (may recurse for subqueries) */
537 root = subquery_planner(glob, parse, NULL, NULL, NULL, false,
538 tuple_fraction, NULL);
539
540 /* Select best Path and turn it into a Plan */
543
545
546 /*
547 * If creating a plan for a scrollable cursor, make sure it can run
548 * backwards on demand. Add a Material node at the top at need.
549 */
550 if (cursorOptions & CURSOR_OPT_SCROLL)
551 {
554 }
555
556 /*
557 * Optionally add a Gather node for testing purposes, provided this is
558 * actually a safe thing to do.
559 *
560 * We can add Gather even when top_plan has parallel-safe initPlans, but
561 * then we have to move the initPlans to the Gather node because of
562 * SS_finalize_plan's limitations. That would cause cosmetic breakage of
563 * regression tests when debug_parallel_query = regress, because initPlans
564 * that would normally appear on the top_plan move to the Gather, causing
565 * them to disappear from EXPLAIN output. That doesn't seem worth kluging
566 * EXPLAIN to hide, so skip it when debug_parallel_query = regress.
567 */
569 top_plan->parallel_safe &&
570 (top_plan->initPlan == NIL ||
572 {
575 bool unsafe_initplans;
576
577 gather->plan.targetlist = top_plan->targetlist;
578 gather->plan.qual = NIL;
579 gather->plan.lefttree = top_plan;
580 gather->plan.righttree = NULL;
581 gather->num_workers = 1;
582 gather->single_copy = true;
584
585 /* Transfer any initPlans to the new top node */
586 gather->plan.initPlan = top_plan->initPlan;
587 top_plan->initPlan = NIL;
588
589 /*
590 * Since this Gather has no parallel-aware descendants to signal to,
591 * we don't need a rescan Param.
592 */
593 gather->rescan_param = -1;
594
595 /*
596 * Ideally we'd use cost_gather here, but setting up dummy path data
597 * to satisfy it doesn't seem much cleaner than knowing what it does.
598 */
599 gather->plan.startup_cost = top_plan->startup_cost +
601 gather->plan.total_cost = top_plan->total_cost +
603 gather->plan.plan_rows = top_plan->plan_rows;
604 gather->plan.plan_width = top_plan->plan_width;
605 gather->plan.parallel_aware = false;
606 gather->plan.parallel_safe = false;
607
608 /*
609 * Delete the initplans' cost from top_plan. We needn't add it to the
610 * Gather node, since the above coding already included it there.
611 */
612 SS_compute_initplan_cost(gather->plan.initPlan,
614 top_plan->startup_cost -= initplan_cost;
615 top_plan->total_cost -= initplan_cost;
616
617 /* use parallel mode for parallel plans. */
618 root->glob->parallelModeNeeded = true;
619
620 top_plan = &gather->plan;
621 }
622
623 /*
624 * If any Params were generated, run through the plan tree and compute
625 * each plan node's extParam/allParam sets. Ideally we'd merge this into
626 * set_plan_references' tree traversal, but for now it has to be separate
627 * because we need to visit subplans before not after main plan.
628 */
629 if (glob->paramExecTypes != NIL)
630 {
631 Assert(list_length(glob->subplans) == list_length(glob->subroots));
632 forboth(lp, glob->subplans, lr, glob->subroots)
633 {
634 Plan *subplan = (Plan *) lfirst(lp);
636
637 SS_finalize_plan(subroot, subplan);
638 }
640 }
641
642 /* final cleanup of the plan */
643 Assert(glob->finalrtable == NIL);
644 Assert(glob->finalrteperminfos == NIL);
645 Assert(glob->finalrowmarks == NIL);
646 Assert(glob->resultRelations == NIL);
647 Assert(glob->appendRelations == NIL);
649 /* ... and the subplans (both regular subplans and initplans) */
650 Assert(list_length(glob->subplans) == list_length(glob->subroots));
651 forboth(lp, glob->subplans, lr, glob->subroots)
652 {
653 Plan *subplan = (Plan *) lfirst(lp);
655
656 lfirst(lp) = set_plan_references(subroot, subplan);
657 }
658
659 /* build the PlannedStmt result */
661
662 result->commandType = parse->commandType;
663 result->queryId = parse->queryId;
664 result->planOrigin = PLAN_STMT_STANDARD;
665 result->hasReturning = (parse->returningList != NIL);
666 result->hasModifyingCTE = parse->hasModifyingCTE;
667 result->canSetTag = parse->canSetTag;
668 result->transientPlan = glob->transientPlan;
669 result->dependsOnRole = glob->dependsOnRole;
670 result->parallelModeNeeded = glob->parallelModeNeeded;
671 result->planTree = top_plan;
672 result->partPruneInfos = glob->partPruneInfos;
673 result->rtable = glob->finalrtable;
674 result->unprunableRelids = bms_difference(glob->allRelids,
675 glob->prunableRelids);
676 result->permInfos = glob->finalrteperminfos;
677 result->subrtinfos = glob->subrtinfos;
678 result->appendRelations = glob->appendRelations;
679 result->subplans = glob->subplans;
680 result->rewindPlanIDs = glob->rewindPlanIDs;
681 result->rowMarks = glob->finalrowmarks;
682
683 /*
684 * Compute resultRelationRelids and rowMarkRelids from resultRelations and
685 * rowMarks. These can be used for cheap membership checks.
686 */
687 foreach(lc, glob->resultRelations)
688 result->resultRelationRelids = bms_add_member(result->resultRelationRelids,
689 lfirst_int(lc));
690 foreach(lc, glob->finalrowmarks)
691 result->rowMarkRelids = bms_add_member(result->rowMarkRelids,
692 ((PlanRowMark *) lfirst(lc))->rti);
693
694 result->relationOids = glob->relationOids;
695 result->invalItems = glob->invalItems;
696 result->paramExecTypes = glob->paramExecTypes;
697 /* utilityStmt should be null, but we might as well copy it */
698 result->utilityStmt = parse->utilityStmt;
699 result->elidedNodes = glob->elidedNodes;
700 result->stmt_location = parse->stmt_location;
701 result->stmt_len = parse->stmt_len;
702
703 result->jitFlags = PGJIT_NONE;
704 if (jit_enabled && jit_above_cost >= 0 &&
705 top_plan->total_cost > jit_above_cost)
706 {
707 result->jitFlags |= PGJIT_PERFORM;
708
709 /*
710 * Decide how much effort should be put into generating better code.
711 */
712 if (jit_optimize_above_cost >= 0 &&
713 top_plan->total_cost > jit_optimize_above_cost)
714 result->jitFlags |= PGJIT_OPT3;
715 if (jit_inline_above_cost >= 0 &&
716 top_plan->total_cost > jit_inline_above_cost)
717 result->jitFlags |= PGJIT_INLINE;
718
719 /*
720 * Decide which operations should be JITed.
721 */
722 if (jit_expressions)
723 result->jitFlags |= PGJIT_EXPR;
725 result->jitFlags |= PGJIT_DEFORM;
726 }
727
728 /* Allow plugins to take control before we discard "glob" */
730 (*planner_shutdown_hook) (glob, parse, query_string, result);
731
732 if (glob->partition_directory != NULL)
733 DestroyPartitionDirectory(glob->partition_directory);
734
735 return result;
736}
Bitmapset * bms_difference(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:346
Bitmapset * bms_add_member(Bitmapset *a, int x)
Definition bitmapset.c:799
uint32 result
char max_parallel_hazard(Query *parse)
Definition clauses.c:747
bool enable_seqscan
Definition costsize.c:146
int max_parallel_workers_per_gather
Definition costsize.c:144
bool enable_memoize
Definition costsize.c:156
double parallel_setup_cost
Definition costsize.c:137
bool enable_gathermerge
Definition costsize.c:159
double parallel_tuple_cost
Definition costsize.c:136
bool enable_indexonlyscan
Definition costsize.c:148
bool enable_tidscan
Definition costsize.c:150
bool enable_material
Definition costsize.c:155
bool enable_hashjoin
Definition costsize.c:158
bool enable_mergejoin
Definition costsize.c:157
bool enable_partitionwise_join
Definition costsize.c:160
bool enable_nestloop
Definition costsize.c:154
bool enable_bitmapscan
Definition costsize.c:149
bool enable_indexscan
Definition costsize.c:147
Plan * materialize_finished_plan(Plan *subplan)
Plan * create_plan(PlannerInfo *root, Path *best_path)
Definition createplan.c:339
bool ExecSupportsBackwardScan(Plan *node)
Definition execAmi.c:512
bool IsUnderPostmaster
Definition globals.c:122
#define IsParallelWorker()
Definition parallel.h:62
double jit_optimize_above_cost
Definition jit.c:42
bool jit_enabled
Definition jit.c:33
bool jit_expressions
Definition jit.c:37
bool jit_tuple_deforming
Definition jit.c:39
double jit_above_cost
Definition jit.c:40
double jit_inline_above_cost
Definition jit.c:41
#define PGJIT_OPT3
Definition jit.h:21
#define PGJIT_NONE
Definition jit.h:19
#define PGJIT_EXPR
Definition jit.h:23
#define PGJIT_DEFORM
Definition jit.h:24
#define PGJIT_INLINE
Definition jit.h:22
#define PGJIT_PERFORM
Definition jit.h:20
double Cost
Definition nodes.h:261
@ CMD_SELECT
Definition nodes.h:275
@ DEBUG_PARALLEL_REGRESS
Definition optimizer.h:98
@ DEBUG_PARALLEL_OFF
Definition optimizer.h:96
#define CURSOR_OPT_SCROLL
#define CURSOR_OPT_FAST_PLAN
#define CURSOR_OPT_PARALLEL_OK
void DestroyPartitionDirectory(PartitionDirectory pdir)
Definition partdesc.c:484
#define PGS_NESTLOOP_MEMOIZE
Definition pathnodes.h:76
#define PGS_TIDSCAN
Definition pathnodes.h:70
#define PGS_FOREIGNJOIN
Definition pathnodes.h:71
#define PGS_APPEND
Definition pathnodes.h:78
#define PGS_MERGE_APPEND
Definition pathnodes.h:79
#define PGS_SEQSCAN
Definition pathnodes.h:66
#define PGS_CONSIDER_INDEXONLY
Definition pathnodes.h:82
#define PGS_NESTLOOP_MATERIALIZE
Definition pathnodes.h:75
#define PGS_MERGEJOIN_PLAIN
Definition pathnodes.h:72
#define PGS_MERGEJOIN_MATERIALIZE
Definition pathnodes.h:73
#define PGS_HASHJOIN
Definition pathnodes.h:77
#define PGS_CONSIDER_NONPARTIAL
Definition pathnodes.h:84
#define PGS_BITMAPSCAN
Definition pathnodes.h:69
#define PGS_GATHER
Definition pathnodes.h:80
#define PGS_CONSIDER_PARTITIONWISE
Definition pathnodes.h:83
#define PGS_GATHER_MERGE
Definition pathnodes.h:81
@ UPPERREL_FINAL
Definition pathnodes.h:152
#define PGS_INDEXONLYSCAN
Definition pathnodes.h:68
#define PGS_INDEXSCAN
Definition pathnodes.h:67
#define PGS_NESTLOOP_PLAIN
Definition pathnodes.h:74
#define lfirst_node(type, lc)
Definition pg_list.h:176
#define lfirst_int(lc)
Definition pg_list.h:173
PlannerInfo * subquery_planner(PlannerGlobal *glob, Query *parse, char *plan_name, PlannerInfo *parent_root, PlannerInfo *alternative_root, bool hasRecursion, double tuple_fraction, SetOperationStmt *setops)
Definition planner.c:775
double cursor_tuple_fraction
Definition planner.c:68
planner_shutdown_hook_type planner_shutdown_hook
Definition planner.c:80
Path * get_cheapest_fractional_path(RelOptInfo *rel, double tuple_fraction)
Definition planner.c:6865
planner_setup_hook_type planner_setup_hook
Definition planner.c:77
int debug_parallel_query
Definition planner.c:69
@ PLAN_STMT_STANDARD
Definition plannodes.h:39
e
RelOptInfo * fetch_upper_rel(PlannerInfo *root, UpperRelationKind kind, Relids relids)
Definition relnode.c:1617
Plan * set_plan_references(PlannerInfo *root, Plan *plan)
Definition setrefs.c:291
Bitmapset * prunableRelids
Definition pathnodes.h:206
char maxParallelHazard
Definition pathnodes.h:260
List * subplans
Definition pathnodes.h:178
bool dependsOnRole
Definition pathnodes.h:251
Bitmapset * allRelids
Definition pathnodes.h:199
List * appendRelations
Definition pathnodes.h:221
List * finalrowmarks
Definition pathnodes.h:215
List * invalItems
Definition pathnodes.h:230
List * relationOids
Definition pathnodes.h:227
List * paramExecTypes
Definition pathnodes.h:233
bool parallelModeOK
Definition pathnodes.h:254
bool transientPlan
Definition pathnodes.h:248
Bitmapset * rewindPlanIDs
Definition pathnodes.h:190
List * finalrteperminfos
Definition pathnodes.h:209
List * subpaths
Definition pathnodes.h:181
Index lastRowMarkId
Definition pathnodes.h:242
List * resultRelations
Definition pathnodes.h:218
List * partPruneInfos
Definition pathnodes.h:224
List * finalrtable
Definition pathnodes.h:193
uint64 default_pgs_mask
Definition pathnodes.h:263
bool parallelModeNeeded
Definition pathnodes.h:257
void SS_finalize_plan(PlannerInfo *root, Plan *plan)
Definition subselect.c:2549
void SS_compute_initplan_cost(List *init_plans, Cost *initplan_cost_p, bool *unsafe_initplans_p)
Definition subselect.c:2493

References PlannerGlobal::allRelids, PlannerGlobal::appendRelations, Assert, bms_add_member(), bms_difference(), CMD_SELECT, create_plan(), CURSOR_OPT_FAST_PLAN, CURSOR_OPT_PARALLEL_OK, CURSOR_OPT_SCROLL, cursor_tuple_fraction, DEBUG_PARALLEL_OFF, debug_parallel_query, DEBUG_PARALLEL_REGRESS, PlannerGlobal::default_pgs_mask, PlannerGlobal::dependsOnRole, DestroyPartitionDirectory(), PlannerGlobal::elidedNodes, enable_bitmapscan, enable_gathermerge, enable_hashjoin, enable_indexonlyscan, enable_indexscan, enable_material, enable_memoize, enable_mergejoin, enable_nestloop, enable_partitionwise_join, enable_seqscan, enable_tidscan, ExecSupportsBackwardScan(), fb(), fetch_upper_rel(), PlannerGlobal::finalrowmarks, PlannerGlobal::finalrtable, PlannerGlobal::finalrteperminfos, forboth, get_cheapest_fractional_path(), PlannerGlobal::invalItems, IsParallelWorker, IsUnderPostmaster, jit_above_cost, jit_enabled, jit_expressions, jit_inline_above_cost, jit_optimize_above_cost, jit_tuple_deforming, PlannerGlobal::lastPHId, PlannerGlobal::lastPlanNodeId, PlannerGlobal::lastRowMarkId, lfirst, lfirst_int, lfirst_node, list_length(), makeNode, materialize_finished_plan(), max_parallel_hazard(), max_parallel_workers_per_gather, PlannerGlobal::maxParallelHazard, NIL, parallel_setup_cost, parallel_tuple_cost, PlannerGlobal::parallelModeNeeded, PlannerGlobal::parallelModeOK, PlannerGlobal::paramExecTypes, parse(), PlannerGlobal::partPruneInfos, PGJIT_DEFORM, PGJIT_EXPR, PGJIT_INLINE, PGJIT_NONE, PGJIT_OPT3, PGJIT_PERFORM, PGS_APPEND, PGS_BITMAPSCAN, PGS_CONSIDER_INDEXONLY, PGS_CONSIDER_NONPARTIAL, PGS_CONSIDER_PARTITIONWISE, PGS_FOREIGNJOIN, PGS_GATHER, PGS_GATHER_MERGE, PGS_HASHJOIN, PGS_INDEXONLYSCAN, PGS_INDEXSCAN, PGS_MERGE_APPEND, PGS_MERGEJOIN_MATERIALIZE, PGS_MERGEJOIN_PLAIN, PGS_NESTLOOP_MATERIALIZE, PGS_NESTLOOP_MEMOIZE, PGS_NESTLOOP_PLAIN, PGS_SEQSCAN, PGS_TIDSCAN, PLAN_STMT_STANDARD, planner_setup_hook, planner_shutdown_hook, PlannerGlobal::prunableRelids, PlannerGlobal::relationOids, result, PlannerGlobal::resultRelations, PlannerGlobal::rewindPlanIDs, root, set_plan_references(), SS_compute_initplan_cost(), SS_finalize_plan(), PlannerGlobal::subpaths, PlannerGlobal::subplans, subquery_planner(), PlannerGlobal::subrtinfos, PlannerGlobal::transientPlan, and UPPERREL_FINAL.

Referenced by delay_execution_planner(), pgss_planner(), and planner().

◆ subquery_planner()

PlannerInfo * subquery_planner ( PlannerGlobal glob,
Query parse,
char plan_name,
PlannerInfo parent_root,
PlannerInfo alternative_root,
bool  hasRecursion,
double  tuple_fraction,
SetOperationStmt setops 
)
extern

Definition at line 775 of file planner.c.

779{
784 int havingIdx;
785 bool hasOuterJoins;
786 bool hasResultRTEs;
788 ListCell *l;
789
790 /* Create a PlannerInfo data structure for this subquery */
792 root->parse = parse;
793 root->glob = glob;
794 root->query_level = parent_root ? parent_root->query_level + 1 : 1;
795 root->plan_name = plan_name;
796 if (alternative_root != NULL)
797 root->alternative_plan_name = alternative_root->plan_name;
798 else
799 root->alternative_plan_name = plan_name;
800 root->parent_root = parent_root;
801 root->plan_params = NIL;
802 root->outer_params = NULL;
803 root->planner_cxt = CurrentMemoryContext;
804 root->init_plans = NIL;
805 root->cte_plan_ids = NIL;
806 root->multiexpr_params = NIL;
807 root->join_domains = NIL;
808 root->eq_classes = NIL;
809 root->ec_merging_done = false;
810 root->last_rinfo_serial = 0;
811 root->all_result_relids =
812 parse->resultRelation ? bms_make_singleton(parse->resultRelation) : NULL;
813 root->leaf_result_relids = NULL; /* we'll find out leaf-ness later */
814 root->append_rel_list = NIL;
815 root->row_identity_vars = NIL;
816 root->rowMarks = NIL;
817 memset(root->upper_rels, 0, sizeof(root->upper_rels));
818 memset(root->upper_targets, 0, sizeof(root->upper_targets));
819 root->processed_groupClause = NIL;
820 root->processed_distinctClause = NIL;
821 root->processed_tlist = NIL;
822 root->update_colnos = NIL;
823 root->grouping_map = NULL;
824 root->minmax_aggs = NIL;
825 root->qual_security_level = 0;
826 root->hasPseudoConstantQuals = false;
827 root->hasAlternativeSubPlans = false;
828 root->placeholdersFrozen = false;
829 root->hasRecursion = hasRecursion;
830 root->assumeReplanning = false;
831 if (hasRecursion)
832 root->wt_param_id = assign_special_exec_param(root);
833 else
834 root->wt_param_id = -1;
835 root->non_recursive_path = NULL;
836
837 /*
838 * Create the top-level join domain. This won't have valid contents until
839 * deconstruct_jointree fills it in, but the node needs to exist before
840 * that so we can build EquivalenceClasses referencing it.
841 */
842 root->join_domains = list_make1(makeNode(JoinDomain));
843
844 /*
845 * If there is a WITH list, process each WITH query and either convert it
846 * to RTE_SUBQUERY RTE(s) or build an initplan SubPlan structure for it.
847 */
848 if (parse->cteList)
850
851 /*
852 * If it's a MERGE command, transform the joinlist as appropriate.
853 */
855
856 /*
857 * Scan the rangetable for relation RTEs and retrieve the necessary
858 * catalog information for each relation. Using this information, clear
859 * the inh flag for any relation that has no children, collect not-null
860 * attribute numbers for any relation that has column not-null
861 * constraints, and expand virtual generated columns for any relation that
862 * contains them. Note that this step does not descend into sublinks and
863 * subqueries; if we pull up any sublinks or subqueries below, their
864 * relation RTEs are processed just before pulling them up.
865 */
867
868 /*
869 * If the FROM clause is empty, replace it with a dummy RTE_RESULT RTE, so
870 * that we don't need so many special cases to deal with that situation.
871 */
873
874 /*
875 * Look for ANY and EXISTS SubLinks in WHERE and JOIN/ON clauses, and try
876 * to transform them into joins. Note that this step does not descend
877 * into subqueries; if we pull up any subqueries below, their SubLinks are
878 * processed just before pulling them up.
879 */
880 if (parse->hasSubLinks)
882
883 /*
884 * Scan the rangetable for function RTEs, do const-simplification on them,
885 * and then inline them if possible (producing subqueries that might get
886 * pulled up next). Recursion issues here are handled in the same way as
887 * for SubLinks.
888 */
890
891 /*
892 * Check to see if any subqueries in the jointree can be merged into this
893 * query.
894 */
896
897 /*
898 * If this is a simple UNION ALL query, flatten it into an appendrel. We
899 * do this now because it requires applying pull_up_subqueries to the leaf
900 * queries of the UNION ALL, which weren't touched above because they
901 * weren't referenced by the jointree (they will be after we do this).
902 */
903 if (parse->setOperations)
905
906 /*
907 * Survey the rangetable to see what kinds of entries are present. We can
908 * skip some later processing if relevant SQL features are not used; for
909 * example if there are no JOIN RTEs we can avoid the expense of doing
910 * flatten_join_alias_vars(). This must be done after we have finished
911 * adding rangetable entries, of course. (Note: actually, processing of
912 * inherited or partitioned rels can cause RTEs for their child tables to
913 * get added later; but those must all be RTE_RELATION entries, so they
914 * don't invalidate the conclusions drawn here.)
915 */
916 root->hasJoinRTEs = false;
917 root->hasLateralRTEs = false;
918 root->group_rtindex = 0;
919 hasOuterJoins = false;
920 hasResultRTEs = false;
921 foreach(l, parse->rtable)
922 {
924
925 switch (rte->rtekind)
926 {
927 case RTE_JOIN:
928 root->hasJoinRTEs = true;
929 if (IS_OUTER_JOIN(rte->jointype))
930 hasOuterJoins = true;
931 break;
932 case RTE_RESULT:
933 hasResultRTEs = true;
934 break;
935 case RTE_GROUP:
936 Assert(parse->hasGroupRTE);
937 root->group_rtindex = list_cell_number(parse->rtable, l) + 1;
938 break;
939 default:
940 /* No work here for other RTE types */
941 break;
942 }
943
944 if (rte->lateral)
945 root->hasLateralRTEs = true;
946
947 /*
948 * We can also determine the maximum security level required for any
949 * securityQuals now. Addition of inheritance-child RTEs won't affect
950 * this, because child tables don't have their own securityQuals; see
951 * expand_single_inheritance_child().
952 */
953 if (rte->securityQuals)
954 root->qual_security_level = Max(root->qual_security_level,
955 list_length(rte->securityQuals));
956 }
957
958 /*
959 * If we have now verified that the query target relation is
960 * non-inheriting, mark it as a leaf target.
961 */
962 if (parse->resultRelation)
963 {
964 RangeTblEntry *rte = rt_fetch(parse->resultRelation, parse->rtable);
965
966 if (!rte->inh)
967 root->leaf_result_relids =
968 bms_make_singleton(parse->resultRelation);
969 }
970
971 /*
972 * This would be a convenient time to check access permissions for all
973 * relations mentioned in the query, since it would be better to fail now,
974 * before doing any detailed planning. However, for historical reasons,
975 * we leave this to be done at executor startup.
976 *
977 * Note, however, that we do need to check access permissions for any view
978 * relations mentioned in the query, in order to prevent information being
979 * leaked by selectivity estimation functions, which only check view owner
980 * permissions on underlying tables (see all_rows_selectable() and its
981 * callers). This is a little ugly, because it means that access
982 * permissions for views will be checked twice, which is another reason
983 * why it would be better to do all the ACL checks here.
984 */
985 foreach(l, parse->rtable)
986 {
988
989 if (rte->perminfoindex != 0 &&
990 rte->relkind == RELKIND_VIEW)
991 {
993 bool result;
994
995 perminfo = getRTEPermissionInfo(parse->rteperminfos, rte);
997 if (!result)
999 get_rel_name(perminfo->relid));
1000 }
1001 }
1002
1003 /*
1004 * Preprocess RowMark information. We need to do this after subquery
1005 * pullup, so that all base relations are present.
1006 */
1008
1009 /*
1010 * Set hasHavingQual to remember if HAVING clause is present. Needed
1011 * because preprocess_expression will reduce a constant-true condition to
1012 * an empty qual list ... but "HAVING TRUE" is not a semantic no-op.
1013 */
1014 root->hasHavingQual = (parse->havingQual != NULL);
1015
1016 /*
1017 * Do expression preprocessing on targetlist and quals, as well as other
1018 * random expressions in the querytree. Note that we do not need to
1019 * handle sort/group expressions explicitly, because they are actually
1020 * part of the targetlist.
1021 */
1022 parse->targetList = (List *)
1023 preprocess_expression(root, (Node *) parse->targetList,
1025
1027 foreach(l, parse->withCheckOptions)
1028 {
1030
1031 wco->qual = preprocess_expression(root, wco->qual,
1033 if (wco->qual != NULL)
1035 }
1036 parse->withCheckOptions = newWithCheckOptions;
1037
1038 parse->returningList = (List *)
1039 preprocess_expression(root, (Node *) parse->returningList,
1041
1042 preprocess_qual_conditions(root, (Node *) parse->jointree);
1043
1044 parse->havingQual = preprocess_expression(root, parse->havingQual,
1046
1047 foreach(l, parse->windowClause)
1048 {
1050
1051 /* partitionClause/orderClause are sort/group expressions */
1056 }
1057
1058 parse->limitOffset = preprocess_expression(root, parse->limitOffset,
1060 parse->limitCount = preprocess_expression(root, parse->limitCount,
1062
1063 if (parse->onConflict)
1064 {
1065 parse->onConflict->arbiterElems = (List *)
1067 (Node *) parse->onConflict->arbiterElems,
1069 parse->onConflict->arbiterWhere =
1071 parse->onConflict->arbiterWhere,
1073 parse->onConflict->onConflictSet = (List *)
1075 (Node *) parse->onConflict->onConflictSet,
1077 parse->onConflict->onConflictWhere =
1079 parse->onConflict->onConflictWhere,
1081 /* exclRelTlist contains only Vars, so no preprocessing needed */
1082 }
1083
1084 foreach(l, parse->mergeActionList)
1085 {
1087
1088 action->targetList = (List *)
1090 (Node *) action->targetList,
1092 action->qual =
1094 (Node *) action->qual,
1096 }
1097
1098 parse->mergeJoinCondition =
1099 preprocess_expression(root, parse->mergeJoinCondition, EXPRKIND_QUAL);
1100
1101 root->append_rel_list = (List *)
1102 preprocess_expression(root, (Node *) root->append_rel_list,
1104
1105 /* Also need to preprocess expressions within RTEs */
1106 foreach(l, parse->rtable)
1107 {
1109 int kind;
1110 ListCell *lcsq;
1111
1112 if (rte->rtekind == RTE_RELATION)
1113 {
1114 if (rte->tablesample)
1115 rte->tablesample = (TableSampleClause *)
1117 (Node *) rte->tablesample,
1119 }
1120 else if (rte->rtekind == RTE_SUBQUERY)
1121 {
1122 /*
1123 * We don't want to do all preprocessing yet on the subquery's
1124 * expressions, since that will happen when we plan it. But if it
1125 * contains any join aliases of our level, those have to get
1126 * expanded now, because planning of the subquery won't do it.
1127 * That's only possible if the subquery is LATERAL.
1128 */
1129 if (rte->lateral && root->hasJoinRTEs)
1130 rte->subquery = (Query *)
1132 (Node *) rte->subquery);
1133 }
1134 else if (rte->rtekind == RTE_FUNCTION)
1135 {
1136 /* Preprocess the function expression(s) fully */
1137 kind = rte->lateral ? EXPRKIND_RTFUNC_LATERAL : EXPRKIND_RTFUNC;
1138 rte->functions = (List *)
1139 preprocess_expression(root, (Node *) rte->functions, kind);
1140 }
1141 else if (rte->rtekind == RTE_TABLEFUNC)
1142 {
1143 /* Preprocess the function expression(s) fully */
1145 rte->tablefunc = (TableFunc *)
1146 preprocess_expression(root, (Node *) rte->tablefunc, kind);
1147 }
1148 else if (rte->rtekind == RTE_VALUES)
1149 {
1150 /* Preprocess the values lists fully */
1151 kind = rte->lateral ? EXPRKIND_VALUES_LATERAL : EXPRKIND_VALUES;
1152 rte->values_lists = (List *)
1153 preprocess_expression(root, (Node *) rte->values_lists, kind);
1154 }
1155 else if (rte->rtekind == RTE_GROUP)
1156 {
1157 /* Preprocess the groupexprs list fully */
1158 rte->groupexprs = (List *)
1159 preprocess_expression(root, (Node *) rte->groupexprs,
1161 }
1162
1163 /*
1164 * Process each element of the securityQuals list as if it were a
1165 * separate qual expression (as indeed it is). We need to do it this
1166 * way to get proper canonicalization of AND/OR structure. Note that
1167 * this converts each element into an implicit-AND sublist.
1168 */
1169 foreach(lcsq, rte->securityQuals)
1170 {
1172 (Node *) lfirst(lcsq),
1174 }
1175 }
1176
1177 /*
1178 * Now that we are done preprocessing expressions, and in particular done
1179 * flattening join alias variables, get rid of the joinaliasvars lists.
1180 * They no longer match what expressions in the rest of the tree look
1181 * like, because we have not preprocessed expressions in those lists (and
1182 * do not want to; for example, expanding a SubLink there would result in
1183 * a useless unreferenced subplan). Leaving them in place simply creates
1184 * a hazard for later scans of the tree. We could try to prevent that by
1185 * using QTW_IGNORE_JOINALIASES in every tree scan done after this point,
1186 * but that doesn't sound very reliable.
1187 */
1188 if (root->hasJoinRTEs)
1189 {
1190 foreach(l, parse->rtable)
1191 {
1193
1194 rte->joinaliasvars = NIL;
1195 }
1196 }
1197
1198 /*
1199 * Before we flatten GROUP Vars, check which HAVING clauses have collation
1200 * conflicts. When GROUP BY uses a nondeterministic collation, values
1201 * that are "equal" for grouping may be distinguishable under a different
1202 * collation. If such a HAVING clause were moved to WHERE, it would
1203 * filter individual rows before grouping, potentially eliminating some
1204 * members of a group and thereby changing aggregate results.
1205 *
1206 * We do this check before flatten_group_exprs because we can easily
1207 * identify grouping expressions by checking whether a Var references
1208 * RTE_GROUP, and such Vars directly carry the GROUP BY collation as their
1209 * varcollid. After flattening, these Vars are replaced by the underlying
1210 * expressions, and we would have to match expressions in the HAVING
1211 * clause back to grouping expressions, which is much more complex.
1212 */
1213 if (parse->hasGroupRTE)
1215 find_having_collation_conflicts(parse, root->group_rtindex);
1216 else
1218
1219 /*
1220 * Replace any Vars in the subquery's targetlist and havingQual that
1221 * reference GROUP outputs with the underlying grouping expressions.
1222 *
1223 * Note that we need to perform this replacement after we've preprocessed
1224 * the grouping expressions. This is to ensure that there is only one
1225 * instance of SubPlan for each SubLink contained within the grouping
1226 * expressions.
1227 */
1228 if (parse->hasGroupRTE)
1229 {
1230 parse->targetList = (List *)
1231 flatten_group_exprs(root, root->parse, (Node *) parse->targetList);
1232 parse->havingQual =
1233 flatten_group_exprs(root, root->parse, parse->havingQual);
1234 }
1235
1236 /* Constant-folding might have removed all set-returning functions */
1237 if (parse->hasTargetSRFs)
1238 parse->hasTargetSRFs = expression_returns_set((Node *) parse->targetList);
1239
1240 /*
1241 * If we have grouping sets, expand the groupingSets tree of this query to
1242 * a flat list of grouping sets. We need to do this before optimizing
1243 * HAVING, since we can't easily tell if there's an empty grouping set
1244 * until we have this representation.
1245 */
1246 if (parse->groupingSets)
1247 {
1248 parse->groupingSets =
1249 expand_grouping_sets(parse->groupingSets, parse->groupDistinct, -1);
1250 }
1251
1252 /*
1253 * In some cases we may want to transfer a HAVING clause into WHERE. We
1254 * cannot do so if the HAVING clause contains aggregates (obviously) or
1255 * volatile functions (since a HAVING clause is supposed to be executed
1256 * only once per group). We also can't do this if there are any grouping
1257 * sets and the clause references any columns that are nullable by the
1258 * grouping sets; the nulled values of those columns are not available
1259 * before the grouping step. (The test on groupClause might seem wrong,
1260 * but it's okay: it's just an optimization to avoid running pull_varnos
1261 * when there cannot be any Vars in the HAVING clause.)
1262 *
1263 * We also cannot do this if the HAVING clause uses a different collation
1264 * than the GROUP BY for any grouping expression whose GROUP BY collation
1265 * is nondeterministic. This is detected before flatten_group_exprs (see
1266 * find_having_collation_conflicts above) and recorded in the
1267 * havingCollationConflicts bitmapset. The bitmapset indexes remain valid
1268 * here because flatten_group_exprs uses expression_tree_mutator, which
1269 * preserves the list length and ordering of havingQual.
1270 *
1271 * Also, it may be that the clause is so expensive to execute that we're
1272 * better off doing it only once per group, despite the loss of
1273 * selectivity. This is hard to estimate short of doing the entire
1274 * planning process twice, so we use a heuristic: clauses containing
1275 * subplans are left in HAVING. Otherwise, we move or copy the HAVING
1276 * clause into WHERE, in hopes of eliminating tuples before aggregation
1277 * instead of after.
1278 *
1279 * If the query has no empty grouping set then we can simply move such a
1280 * clause into WHERE; any group that fails the clause will not be in the
1281 * output because none of its tuples will reach the grouping or
1282 * aggregation stage. Otherwise we have to keep the clause in HAVING to
1283 * ensure that we don't emit a bogus aggregated row. But then the HAVING
1284 * clause must be degenerate (variable-free), so we can copy it into WHERE
1285 * so that query_planner() can use it in a gating Result node. (This could
1286 * be done better, but it seems not worth optimizing.)
1287 *
1288 * Note that a HAVING clause may contain expressions that are not fully
1289 * preprocessed. This can happen if these expressions are part of
1290 * grouping items. In such cases, they are replaced with GROUP Vars in
1291 * the parser and then replaced back after we're done with expression
1292 * preprocessing on havingQual. This is not an issue if the clause
1293 * remains in HAVING, because these expressions will be matched to lower
1294 * target items in setrefs.c. However, if the clause is moved or copied
1295 * into WHERE, we need to ensure that these expressions are fully
1296 * preprocessed.
1297 *
1298 * Note that both havingQual and parse->jointree->quals are in
1299 * implicitly-ANDed-list form at this point, even though they are declared
1300 * as Node *.
1301 */
1302 newHaving = NIL;
1303 havingIdx = 0;
1304 foreach(l, (List *) parse->havingQual)
1305 {
1306 Node *havingclause = (Node *) lfirst(l);
1307
1312 (parse->groupClause && parse->groupingSets &&
1313 bms_is_member(root->group_rtindex, pull_varnos(root, havingclause))))
1314 {
1315 /* keep it in HAVING */
1317 }
1318 else if (parse->groupClause &&
1319 (parse->groupingSets == NIL ||
1320 (List *) linitial(parse->groupingSets) != NIL))
1321 {
1322 /* There is GROUP BY, but no empty grouping set */
1324
1325 /* Preprocess the HAVING clause fully */
1328 /* ... and move it to WHERE */
1329 parse->jointree->quals = (Node *)
1330 list_concat((List *) parse->jointree->quals,
1331 (List *) whereclause);
1332 }
1333 else
1334 {
1335 /* There is an empty grouping set (perhaps implicitly) */
1337
1338 /* Preprocess the HAVING clause fully */
1341 /* ... and put a copy in WHERE */
1342 parse->jointree->quals = (Node *)
1343 list_concat((List *) parse->jointree->quals,
1344 (List *) whereclause);
1345 /* ... and also keep it in HAVING */
1347 }
1348
1349 havingIdx++;
1350 }
1351 parse->havingQual = (Node *) newHaving;
1352
1353 /*
1354 * If we have any outer joins, try to reduce them to plain inner joins.
1355 * This step is most easily done after we've done expression
1356 * preprocessing.
1357 */
1358 if (hasOuterJoins)
1360
1361 /*
1362 * If we have any RTE_RESULT relations, see if they can be deleted from
1363 * the jointree. We also rely on this processing to flatten single-child
1364 * FromExprs underneath outer joins. This step is most effectively done
1365 * after we've done expression preprocessing and outer join reduction.
1366 */
1369
1370 /*
1371 * Do the main planning.
1372 */
1373 grouping_planner(root, tuple_fraction, setops);
1374
1375 /*
1376 * Capture the set of outer-level param IDs we have access to, for use in
1377 * extParam/allParam calculations later.
1378 */
1380
1381 /*
1382 * If any initPlans were created in this query level, adjust the surviving
1383 * Paths' costs and parallel-safety flags to account for them. The
1384 * initPlans won't actually get attached to the plan tree till
1385 * create_plan() runs, but we must include their effects now.
1386 */
1389
1390 /*
1391 * Make sure we've identified the cheapest Path for the final rel. (By
1392 * doing this here not in grouping_planner, we include initPlan costs in
1393 * the decision, though it's unlikely that will change anything.)
1394 */
1396
1397 return root;
1398}
@ ACLCHECK_NO_PRIV
Definition acl.h:185
void aclcheck_error(AclResult aclerr, ObjectType objtype, const char *objectname)
Definition aclchk.c:2672
Bitmapset * bms_make_singleton(int x)
Definition bitmapset.c:216
bool bms_is_member(int x, const Bitmapset *a)
Definition bitmapset.c:510
#define Max(x, y)
Definition c.h:1085
bool contain_agg_clause(Node *clause)
Definition clauses.c:194
bool contain_subplans(Node *clause)
Definition clauses.c:343
bool contain_volatile_functions(Node *clause)
Definition clauses.c:551
bool ExecCheckOneRelPerms(RTEPermissionInfo *perminfo)
Definition execMain.c:657
List * list_concat(List *list1, const List *list2)
Definition list.c:561
char * get_rel_name(Oid relid)
Definition lsyscache.c:2159
MemoryContext CurrentMemoryContext
Definition mcxt.c:160
bool expression_returns_set(Node *clause)
Definition nodeFuncs.c:768
#define copyObject(obj)
Definition nodes.h:232
#define IS_OUTER_JOIN(jointype)
Definition nodes.h:348
int assign_special_exec_param(PlannerInfo *root)
List * expand_grouping_sets(List *groupingSets, bool groupDistinct, int limit)
Definition parse_agg.c:2019
RTEPermissionInfo * getRTEPermissionInfo(List *rteperminfos, RangeTblEntry *rte)
@ RTE_JOIN
@ RTE_VALUES
@ RTE_SUBQUERY
@ RTE_RESULT
@ RTE_FUNCTION
@ RTE_TABLEFUNC
@ RTE_GROUP
@ OBJECT_VIEW
#define rt_fetch(rangetable_index, rangetable)
Definition parsetree.h:31
#define list_make1(x1)
Definition pg_list.h:244
#define linitial(l)
Definition pg_list.h:178
static int list_cell_number(const List *l, const ListCell *c)
Definition pg_list.h:365
#define EXPRKIND_TABLEFUNC_LATERAL
Definition planner.c:99
#define EXPRKIND_TARGET
Definition planner.c:88
#define EXPRKIND_APPINFO
Definition planner.c:94
static void preprocess_rowmarks(PlannerInfo *root)
Definition planner.c:2686
#define EXPRKIND_TABLESAMPLE
Definition planner.c:96
#define EXPRKIND_GROUPEXPR
Definition planner.c:100
static void preprocess_qual_conditions(PlannerInfo *root, Node *jtnode)
Definition planner.c:1509
#define EXPRKIND_RTFUNC_LATERAL
Definition planner.c:90
#define EXPRKIND_VALUES_LATERAL
Definition planner.c:92
#define EXPRKIND_LIMIT
Definition planner.c:93
#define EXPRKIND_VALUES
Definition planner.c:91
static Bitmapset * find_having_collation_conflicts(Query *parse, Index group_rtindex)
Definition planner.c:1558
#define EXPRKIND_QUAL
Definition planner.c:87
static void grouping_planner(PlannerInfo *root, double tuple_fraction, SetOperationStmt *setops)
Definition planner.c:1721
#define EXPRKIND_TABLEFUNC
Definition planner.c:98
#define EXPRKIND_RTFUNC
Definition planner.c:89
#define EXPRKIND_ARBITER_ELEM
Definition planner.c:97
void preprocess_function_rtes(PlannerInfo *root)
void flatten_simple_union_all(PlannerInfo *root)
void transform_MERGE_to_join(Query *parse)
void remove_useless_result_rtes(PlannerInfo *root)
void pull_up_sublinks(PlannerInfo *root)
void replace_empty_jointree(Query *parse)
void pull_up_subqueries(PlannerInfo *root)
Query * preprocess_relation_rtes(PlannerInfo *root)
void reduce_outer_joins(PlannerInfo *root)
Node * startOffset
Node * endOffset
void SS_process_ctes(PlannerInfo *root)
Definition subselect.c:884
void SS_identify_outer_params(PlannerInfo *root)
Definition subselect.c:2365
void SS_charge_for_initplans(PlannerInfo *root, RelOptInfo *final_rel)
Definition subselect.c:2429
Node * flatten_group_exprs(PlannerInfo *root, Query *query, Node *node)
Definition var.c:999
Relids pull_varnos(PlannerInfo *root, Node *node)
Definition var.c:114
Node * flatten_join_alias_vars(PlannerInfo *root, Query *query, Node *node)
Definition var.c:781

References aclcheck_error(), ACLCHECK_NO_PRIV, Assert, assign_special_exec_param(), bms_is_member(), bms_make_singleton(), contain_agg_clause(), contain_subplans(), contain_volatile_functions(), copyObject, CurrentMemoryContext, WindowClause::endOffset, ExecCheckOneRelPerms(), expand_grouping_sets(), expression_returns_set(), EXPRKIND_APPINFO, EXPRKIND_ARBITER_ELEM, EXPRKIND_GROUPEXPR, EXPRKIND_LIMIT, EXPRKIND_QUAL, EXPRKIND_RTFUNC, EXPRKIND_RTFUNC_LATERAL, EXPRKIND_TABLEFUNC, EXPRKIND_TABLEFUNC_LATERAL, EXPRKIND_TABLESAMPLE, EXPRKIND_TARGET, EXPRKIND_VALUES, EXPRKIND_VALUES_LATERAL, fb(), fetch_upper_rel(), find_having_collation_conflicts(), flatten_group_exprs(), flatten_join_alias_vars(), flatten_simple_union_all(), get_rel_name(), getRTEPermissionInfo(), grouping_planner(), IS_OUTER_JOIN, lappend(), lfirst, lfirst_node, linitial, list_cell_number(), list_concat(), list_length(), list_make1, makeNode, Max, NIL, OBJECT_VIEW, parse(), preprocess_expression(), preprocess_function_rtes(), preprocess_qual_conditions(), preprocess_relation_rtes(), preprocess_rowmarks(), pull_up_sublinks(), pull_up_subqueries(), pull_varnos(), reduce_outer_joins(), remove_useless_result_rtes(), replace_empty_jointree(), result, root, rt_fetch, RTE_FUNCTION, RTE_GROUP, RTE_JOIN, RTE_RELATION, RTE_RESULT, RTE_SUBQUERY, RTE_TABLEFUNC, RTE_VALUES, set_cheapest(), SS_charge_for_initplans(), SS_identify_outer_params(), SS_process_ctes(), WindowClause::startOffset, transform_MERGE_to_join(), and UPPERREL_FINAL.

Referenced by make_subplan(), recurse_set_operations(), set_subquery_pathlist(), SS_process_ctes(), and standard_planner().

Variable Documentation

◆ create_upper_paths_hook

◆ planner_hook

PGDLLIMPORT planner_hook_type planner_hook
extern

Definition at line 74 of file planner.c.

Referenced by _PG_init(), and planner().

◆ planner_setup_hook

PGDLLIMPORT planner_setup_hook_type planner_setup_hook
extern

Definition at line 77 of file planner.c.

Referenced by pgpa_planner_install_hooks(), and standard_planner().

◆ planner_shutdown_hook

PGDLLIMPORT planner_shutdown_hook_type planner_shutdown_hook
extern

Definition at line 80 of file planner.c.

Referenced by pgpa_planner_install_hooks(), and standard_planner().