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

Go to the source code of this file.

Macros

#define DEFAULT_CURSOR_TUPLE_FRACTION   0.1
 

Typedefs

typedef void(* query_pathkeys_callback) (PlannerInfo *root, void *extra)
 

Functions

RelOptInfoquery_planner (PlannerInfo *root, query_pathkeys_callback qp_callback, void *qp_extra)
 
void preprocess_minmax_aggregates (PlannerInfo *root)
 
Plancreate_plan (PlannerInfo *root, Path *best_path)
 
ForeignScanmake_foreignscan (List *qptlist, List *qpqual, Index scanrelid, List *fdw_exprs, List *fdw_private, List *fdw_scan_tlist, List *fdw_recheck_quals, Plan *outer_plan)
 
Planchange_plan_targetlist (Plan *subplan, List *tlist, bool tlist_parallel_safe)
 
Planmaterialize_finished_plan (Plan *subplan)
 
bool is_projection_capable_path (Path *path)
 
bool is_projection_capable_plan (Plan *plan)
 
Sortmake_sort_from_sortclauses (List *sortcls, Plan *lefttree)
 
Aggmake_agg (List *tlist, List *qual, AggStrategy aggstrategy, AggSplit aggsplit, int numGroupCols, AttrNumber *grpColIdx, Oid *grpOperators, Oid *grpCollations, List *groupingSets, List *chain, Cardinality numGroups, Size transitionSpace, Plan *lefttree)
 
Limitmake_limit (Plan *lefttree, Node *limitOffset, Node *limitCount, LimitOption limitOption, int uniqNumCols, AttrNumber *uniqColIdx, Oid *uniqOperators, Oid *uniqCollations)
 
void add_base_rels_to_query (PlannerInfo *root, Node *jtnode)
 
void add_other_rels_to_query (PlannerInfo *root)
 
void build_base_rel_tlists (PlannerInfo *root, List *final_tlist)
 
void add_vars_to_targetlist (PlannerInfo *root, List *vars, Relids where_needed)
 
void add_vars_to_attr_needed (PlannerInfo *root, List *vars, Relids where_needed)
 
void remove_useless_groupby_columns (PlannerInfo *root)
 
void setup_eager_aggregation (PlannerInfo *root)
 
void find_lateral_references (PlannerInfo *root)
 
void rebuild_lateral_attr_needed (PlannerInfo *root)
 
void create_lateral_join_info (PlannerInfo *root)
 
Listdeconstruct_jointree (PlannerInfo *root)
 
bool restriction_is_always_true (PlannerInfo *root, RestrictInfo *restrictinfo)
 
bool restriction_is_always_false (PlannerInfo *root, RestrictInfo *restrictinfo)
 
void distribute_restrictinfo_to_rels (PlannerInfo *root, RestrictInfo *restrictinfo)
 
RestrictInfoprocess_implied_equality (PlannerInfo *root, Oid opno, Oid collation, Expr *item1, Expr *item2, Relids qualscope, Index security_level, bool both_const)
 
RestrictInfobuild_implied_join_equality (PlannerInfo *root, Oid opno, Oid collation, Expr *item1, Expr *item2, Relids qualscope, Index security_level)
 
void rebuild_joinclause_attr_needed (PlannerInfo *root)
 
void match_foreign_keys_to_quals (PlannerInfo *root)
 
Listremove_useless_joins (PlannerInfo *root, List *joinlist)
 
void reduce_unique_semijoins (PlannerInfo *root)
 
bool query_supports_distinctness (Query *query)
 
bool query_is_distinct_for (Query *query, List *colnos, List *opids)
 
bool innerrel_is_unique (PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, bool force_cache)
 
bool innerrel_is_unique_ext (PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, bool force_cache, List **extra_clauses)
 
Listremove_useless_self_joins (PlannerInfo *root, List *joinlist)
 
Planset_plan_references (PlannerInfo *root, Plan *plan)
 
bool trivial_subqueryscan (SubqueryScan *plan)
 
Paramfind_minmax_agg_replacement_param (PlannerInfo *root, Aggref *aggref)
 
void record_plan_function_dependency (PlannerInfo *root, Oid funcid)
 
void record_plan_type_dependency (PlannerInfo *root, Oid typid)
 
bool extract_query_dependencies_walker (Node *node, PlannerInfo *context)
 

Variables

PGDLLIMPORT double cursor_tuple_fraction
 
PGDLLIMPORT bool enable_self_join_elimination
 
PGDLLIMPORT int from_collapse_limit
 
PGDLLIMPORT int join_collapse_limit
 

Macro Definition Documentation

◆ DEFAULT_CURSOR_TUPLE_FRACTION

#define DEFAULT_CURSOR_TUPLE_FRACTION   0.1

Definition at line 21 of file planmain.h.

Typedef Documentation

◆ query_pathkeys_callback

typedef void(* query_pathkeys_callback) (PlannerInfo *root, void *extra)

Definition at line 26 of file planmain.h.

Function Documentation

◆ add_base_rels_to_query()

void add_base_rels_to_query ( PlannerInfo root,
Node jtnode 
)
extern

Definition at line 166 of file initsplan.c.

167{
168 if (jtnode == NULL)
169 return;
170 if (IsA(jtnode, RangeTblRef))
171 {
172 int varno = ((RangeTblRef *) jtnode)->rtindex;
173
174 (void) build_simple_rel(root, varno, NULL);
175 }
176 else if (IsA(jtnode, FromExpr))
177 {
178 FromExpr *f = (FromExpr *) jtnode;
179 ListCell *l;
180
181 foreach(l, f->fromlist)
183 }
184 else if (IsA(jtnode, JoinExpr))
185 {
186 JoinExpr *j = (JoinExpr *) jtnode;
187
190 }
191 else
192 elog(ERROR, "unrecognized node type: %d",
193 (int) nodeTag(jtnode));
194}
#define ERROR
Definition elog.h:40
#define elog(elevel,...)
Definition elog.h:228
void add_base_rels_to_query(PlannerInfo *root, Node *jtnode)
Definition initsplan.c:166
int j
Definition isn.c:78
#define IsA(nodeptr, _type_)
Definition nodes.h:164
#define nodeTag(nodeptr)
Definition nodes.h:139
#define lfirst(lc)
Definition pg_list.h:172
static int fb(int x)
tree ctl root
Definition radixtree.h:1857
RelOptInfo * build_simple_rel(PlannerInfo *root, int relid, RelOptInfo *parent)
Definition relnode.c:212
List * fromlist
Definition primnodes.h:2384

References add_base_rels_to_query(), build_simple_rel(), elog, ERROR, fb(), FromExpr::fromlist, IsA, j, JoinTreeItem::jtnode, lfirst, nodeTag, and root.

Referenced by add_base_rels_to_query(), and query_planner().

◆ add_other_rels_to_query()

void add_other_rels_to_query ( PlannerInfo root)
extern

Definition at line 204 of file initsplan.c.

205{
206 int rti;
207
208 for (rti = 1; rti < root->simple_rel_array_size; rti++)
209 {
210 RelOptInfo *rel = root->simple_rel_array[rti];
211 RangeTblEntry *rte = root->simple_rte_array[rti];
212
213 /* there may be empty slots corresponding to non-baserel RTEs */
214 if (rel == NULL)
215 continue;
216
217 /* Ignore any "otherrels" that were already added. */
218 if (rel->reloptkind != RELOPT_BASEREL)
219 continue;
220
221 /* If it's marked as inheritable, look for children. */
222 if (rte->inh)
224 }
225}
void expand_inherited_rtentry(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte, Index rti)
Definition inherit.c:88
@ RELOPT_BASEREL
Definition pathnodes.h:977
RelOptKind reloptkind
Definition pathnodes.h:1015

References expand_inherited_rtentry(), fb(), RELOPT_BASEREL, RelOptInfo::reloptkind, and root.

Referenced by query_planner().

◆ add_vars_to_attr_needed()

void add_vars_to_attr_needed ( PlannerInfo root,
List vars,
Relids  where_needed 
)
extern

Definition at line 361 of file initsplan.c.

363{
364 ListCell *temp;
365
367
368 foreach(temp, vars)
369 {
370 Node *node = (Node *) lfirst(temp);
371
372 if (IsA(node, Var))
373 {
374 Var *var = (Var *) node;
375 RelOptInfo *rel = find_base_rel(root, var->varno);
376 int attno = var->varattno;
377
379 continue;
380 Assert(attno >= rel->min_attr && attno <= rel->max_attr);
381 attno -= rel->min_attr;
382 rel->attr_needed[attno] = bms_add_members(rel->attr_needed[attno],
384 }
385 else if (IsA(node, PlaceHolderVar))
386 {
389
390 phinfo->ph_needed = bms_add_members(phinfo->ph_needed,
392 }
393 else
394 elog(ERROR, "unrecognized node type: %d", (int) nodeTag(node));
395 }
396}
bool bms_is_subset(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:412
Bitmapset * bms_add_members(Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:901
#define bms_is_empty(a)
Definition bitmapset.h:118
#define Assert(condition)
Definition c.h:943
PlaceHolderInfo * find_placeholder_info(PlannerInfo *root, PlaceHolderVar *phv)
Definition placeholder.c:85
RelOptInfo * find_base_rel(PlannerInfo *root, int relid)
Definition relnode.c:544
Definition nodes.h:135
Relids relids
Definition pathnodes.h:1021
AttrNumber min_attr
Definition pathnodes.h:1075
AttrNumber varattno
Definition primnodes.h:275
int varno
Definition primnodes.h:270

References Assert, bms_add_members(), bms_is_empty, bms_is_subset(), elog, ERROR, fb(), find_base_rel(), find_placeholder_info(), IsA, lfirst, RelOptInfo::min_attr, nodeTag, RelOptInfo::relids, root, Var::varattno, and Var::varno.

Referenced by rebuild_eclass_attr_needed(), rebuild_joinclause_attr_needed(), rebuild_lateral_attr_needed(), and rebuild_placeholder_attr_needed().

◆ add_vars_to_targetlist()

void add_vars_to_targetlist ( PlannerInfo root,
List vars,
Relids  where_needed 
)
extern

Definition at line 290 of file initsplan.c.

292{
293 ListCell *temp;
294
296
297 foreach(temp, vars)
298 {
299 Node *node = (Node *) lfirst(temp);
300
301 if (IsA(node, Var))
302 {
303 Var *var = (Var *) node;
304 RelOptInfo *rel = find_base_rel(root, var->varno);
305 int attno = var->varattno;
306
308 continue;
309 Assert(attno >= rel->min_attr && attno <= rel->max_attr);
310 attno -= rel->min_attr;
311 if (rel->attr_needed[attno] == NULL)
312 {
313 /*
314 * Variable not yet requested, so add to rel's targetlist.
315 *
316 * The value available at the rel's scan level has not been
317 * nulled by any outer join, so drop its varnullingrels.
318 * (We'll put those back as we climb up the join tree.)
319 */
320 var = copyObject(var);
321 var->varnullingrels = NULL;
322 rel->reltarget->exprs = lappend(rel->reltarget->exprs, var);
323 /* reltarget cost and width will be computed later */
324 }
325 rel->attr_needed[attno] = bms_add_members(rel->attr_needed[attno],
327 }
328 else if (IsA(node, PlaceHolderVar))
329 {
332
333 phinfo->ph_needed = bms_add_members(phinfo->ph_needed,
335 }
336 else
337 elog(ERROR, "unrecognized node type: %d", (int) nodeTag(node));
338 }
339}
List * lappend(List *list, void *datum)
Definition list.c:339
#define copyObject(obj)
Definition nodes.h:232
List * exprs
Definition pathnodes.h:1878
struct PathTarget * reltarget
Definition pathnodes.h:1045

References Assert, bms_add_members(), bms_is_empty, bms_is_subset(), copyObject, elog, ERROR, PathTarget::exprs, fb(), find_base_rel(), find_placeholder_info(), IsA, lappend(), lfirst, RelOptInfo::min_attr, nodeTag, RelOptInfo::relids, RelOptInfo::reltarget, root, Var::varattno, and Var::varno.

Referenced by build_base_rel_tlists(), distribute_qual_to_rels(), expand_inherited_rtentry(), extract_lateral_references(), fix_placeholder_input_needed_levels(), generate_base_implied_equalities_no_const(), and process_implied_equality().

◆ build_base_rel_tlists()

void build_base_rel_tlists ( PlannerInfo root,
List final_tlist 
)
extern

Definition at line 243 of file initsplan.c.

244{
245 List *tlist_vars = pull_var_clause((Node *) final_tlist,
249
250 if (tlist_vars != NIL)
251 {
253 list_free(tlist_vars);
254 }
255
256 /*
257 * If there's a HAVING clause, we'll need the Vars it uses, too. Note
258 * that HAVING can contain Aggrefs but not WindowFuncs.
259 */
260 if (root->parse->havingQual)
261 {
262 List *having_vars = pull_var_clause(root->parse->havingQual,
265
266 if (having_vars != NIL)
267 {
271 }
272 }
273}
Bitmapset * bms_make_singleton(int x)
Definition bitmapset.c:216
void add_vars_to_targetlist(PlannerInfo *root, List *vars, Relids where_needed)
Definition initsplan.c:290
void list_free(List *list)
Definition list.c:1546
#define PVC_RECURSE_AGGREGATES
Definition optimizer.h:198
#define PVC_RECURSE_WINDOWFUNCS
Definition optimizer.h:200
#define PVC_INCLUDE_PLACEHOLDERS
Definition optimizer.h:201
#define NIL
Definition pg_list.h:68
Definition pg_list.h:54
List * pull_var_clause(Node *node, int flags)
Definition var.c:653

References add_vars_to_targetlist(), bms_make_singleton(), fb(), list_free(), NIL, pull_var_clause(), PVC_INCLUDE_PLACEHOLDERS, PVC_RECURSE_AGGREGATES, PVC_RECURSE_WINDOWFUNCS, and root.

Referenced by distribute_row_identity_vars(), and query_planner().

◆ build_implied_join_equality()

RestrictInfo * build_implied_join_equality ( PlannerInfo root,
Oid  opno,
Oid  collation,
Expr item1,
Expr item2,
Relids  qualscope,
Index  security_level 
)
extern

Definition at line 3807 of file initsplan.c.

3814{
3816 Expr *clause;
3817
3818 /*
3819 * Build the new clause. Copy to ensure it shares no substructure with
3820 * original (this is necessary in case there are subselects in there...)
3821 */
3822 clause = make_opclause(opno,
3823 BOOLOID, /* opresulttype */
3824 false, /* opretset */
3827 InvalidOid,
3828 collation);
3829
3830 /*
3831 * Build the RestrictInfo node itself.
3832 */
3834 clause,
3835 true, /* is_pushed_down */
3836 false, /* !has_clone */
3837 false, /* !is_clone */
3838 false, /* pseudoconstant */
3839 security_level, /* security_level */
3840 qualscope, /* required_relids */
3841 NULL, /* incompatible_relids */
3842 NULL); /* outer_relids */
3843
3844 /* Set mergejoinability/hashjoinability flags */
3848
3849 return restrictinfo;
3850}
static void check_hashjoinable(RestrictInfo *restrictinfo)
Definition initsplan.c:4184
static void check_mergejoinable(RestrictInfo *restrictinfo)
Definition initsplan.c:4147
static void check_memoizable(RestrictInfo *restrictinfo)
Definition initsplan.c:4212
Expr * make_opclause(Oid opno, Oid opresulttype, bool opretset, Expr *leftop, Expr *rightop, Oid opcollid, Oid inputcollid)
Definition makefuncs.c:701
#define InvalidOid
RestrictInfo * make_restrictinfo(PlannerInfo *root, Expr *clause, bool is_pushed_down, bool has_clone, bool is_clone, bool pseudoconstant, Index security_level, Relids required_relids, Relids incompatible_relids, Relids outer_relids)

References check_hashjoinable(), check_memoizable(), check_mergejoinable(), copyObject, fb(), InvalidOid, make_opclause(), make_restrictinfo(), JoinTreeItem::qualscope, and root.

Referenced by create_join_clause(), reconsider_full_join_clause(), and reconsider_outer_join_clause().

◆ change_plan_targetlist()

Plan * change_plan_targetlist ( Plan subplan,
List tlist,
bool  tlist_parallel_safe 
)
extern

Definition at line 1992 of file createplan.c.

1993{
1994 /*
1995 * If the top plan node can't do projections and its existing target list
1996 * isn't already what we need, we need to add a Result node to help it
1997 * along.
1998 */
1999 if (!is_projection_capable_plan(subplan) &&
2000 !tlist_same_exprs(tlist, subplan->targetlist))
2001 subplan = inject_projection_plan(subplan, tlist,
2002 subplan->parallel_safe &&
2004 else
2005 {
2006 /* Else we can just replace the plan node's tlist */
2007 subplan->targetlist = tlist;
2009 }
2010 return subplan;
2011}
bool is_projection_capable_plan(Plan *plan)
static Plan * inject_projection_plan(Plan *subplan, List *tlist, bool parallel_safe)
bool parallel_safe
Definition plannodes.h:221
List * targetlist
Definition plannodes.h:235
bool tlist_same_exprs(List *tlist1, List *tlist2)
Definition tlist.c:227

References fb(), inject_projection_plan(), is_projection_capable_plan(), Plan::parallel_safe, Plan::targetlist, and tlist_same_exprs().

Referenced by create_nestloop_plan(), and postgresGetForeignPlan().

◆ create_lateral_join_info()

void create_lateral_join_info ( PlannerInfo root)
extern

Definition at line 1232 of file initsplan.c.

1233{
1234 bool found_laterals = false;
1235 Index rti;
1236 ListCell *lc;
1237
1238 /* We need do nothing if the query contains no LATERAL RTEs */
1239 if (!root->hasLateralRTEs)
1240 return;
1241
1242 /* We'll need to have the ph_eval_at values for PlaceHolderVars */
1243 Assert(root->placeholdersFrozen);
1244
1245 /*
1246 * Examine all baserels (the rel array has been set up by now).
1247 */
1248 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1249 {
1250 RelOptInfo *brel = root->simple_rel_array[rti];
1251 Relids lateral_relids;
1252
1253 /* there may be empty slots corresponding to non-baserel RTEs */
1254 if (brel == NULL)
1255 continue;
1256
1257 Assert(brel->relid == rti); /* sanity check on array */
1258
1259 /* ignore RTEs that are "other rels" */
1260 if (brel->reloptkind != RELOPT_BASEREL)
1261 continue;
1262
1263 lateral_relids = NULL;
1264
1265 /* consider each laterally-referenced Var or PHV */
1266 foreach(lc, brel->lateral_vars)
1267 {
1268 Node *node = (Node *) lfirst(lc);
1269
1270 if (IsA(node, Var))
1271 {
1272 Var *var = (Var *) node;
1273
1274 found_laterals = true;
1275 lateral_relids = bms_add_member(lateral_relids,
1276 var->varno);
1277 }
1278 else if (IsA(node, PlaceHolderVar))
1279 {
1280 PlaceHolderVar *phv = (PlaceHolderVar *) node;
1282
1283 found_laterals = true;
1284 lateral_relids = bms_add_members(lateral_relids,
1285 phinfo->ph_eval_at);
1286 }
1287 else
1288 Assert(false);
1289 }
1290
1291 /* We now have all the simple lateral refs from this rel */
1292 brel->direct_lateral_relids = lateral_relids;
1293 brel->lateral_relids = bms_copy(lateral_relids);
1294 }
1295
1296 /*
1297 * Now check for lateral references within PlaceHolderVars, and mark their
1298 * eval_at rels as having lateral references to the source rels.
1299 *
1300 * For a PHV that is due to be evaluated at a baserel, mark its source(s)
1301 * as direct lateral dependencies of the baserel (adding onto the ones
1302 * recorded above). If it's due to be evaluated at a join, mark its
1303 * source(s) as indirect lateral dependencies of each baserel in the join,
1304 * ie put them into lateral_relids but not direct_lateral_relids. This is
1305 * appropriate because we can't put any such baserel on the outside of a
1306 * join to one of the PHV's lateral dependencies, but on the other hand we
1307 * also can't yet join it directly to the dependency.
1308 */
1309 foreach(lc, root->placeholder_list)
1310 {
1312 Relids eval_at = phinfo->ph_eval_at;
1314 int varno;
1315
1316 if (phinfo->ph_lateral == NULL)
1317 continue; /* PHV is uninteresting if no lateral refs */
1318
1319 found_laterals = true;
1320
1321 /*
1322 * Include only baserels not outer joins in the evaluation sites'
1323 * lateral relids. This avoids problems when outer join order gets
1324 * rearranged, and it should still ensure that the lateral values are
1325 * available when needed.
1326 */
1327 lateral_refs = bms_intersect(phinfo->ph_lateral, root->all_baserels);
1329
1330 if (bms_get_singleton_member(eval_at, &varno))
1331 {
1332 /* Evaluation site is a baserel */
1333 RelOptInfo *brel = find_base_rel(root, varno);
1334
1335 brel->direct_lateral_relids =
1336 bms_add_members(brel->direct_lateral_relids,
1337 lateral_refs);
1338 brel->lateral_relids =
1339 bms_add_members(brel->lateral_relids,
1340 lateral_refs);
1341 }
1342 else
1343 {
1344 /* Evaluation site is a join */
1345 varno = -1;
1346 while ((varno = bms_next_member(eval_at, varno)) >= 0)
1347 {
1349
1350 if (brel == NULL)
1351 continue; /* ignore outer joins in eval_at */
1352 brel->lateral_relids = bms_add_members(brel->lateral_relids,
1353 lateral_refs);
1354 }
1355 }
1356 }
1357
1358 /*
1359 * If we found no actual lateral references, we're done; but reset the
1360 * hasLateralRTEs flag to avoid useless work later.
1361 */
1362 if (!found_laterals)
1363 {
1364 root->hasLateralRTEs = false;
1365 return;
1366 }
1367
1368 /*
1369 * Calculate the transitive closure of the lateral_relids sets, so that
1370 * they describe both direct and indirect lateral references. If relation
1371 * X references Y laterally, and Y references Z laterally, then we will
1372 * have to scan X on the inside of a nestloop with Z, so for all intents
1373 * and purposes X is laterally dependent on Z too.
1374 *
1375 * This code is essentially Warshall's algorithm for transitive closure.
1376 * The outer loop considers each baserel, and propagates its lateral
1377 * dependencies to those baserels that have a lateral dependency on it.
1378 */
1379 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1380 {
1381 RelOptInfo *brel = root->simple_rel_array[rti];
1383 Index rti2;
1384
1385 if (brel == NULL || brel->reloptkind != RELOPT_BASEREL)
1386 continue;
1387
1388 /* need not consider baserel further if it has no lateral refs */
1389 outer_lateral_relids = brel->lateral_relids;
1391 continue;
1392
1393 /* else scan all baserels */
1394 for (rti2 = 1; rti2 < root->simple_rel_array_size; rti2++)
1395 {
1396 RelOptInfo *brel2 = root->simple_rel_array[rti2];
1397
1398 if (brel2 == NULL || brel2->reloptkind != RELOPT_BASEREL)
1399 continue;
1400
1401 /* if brel2 has lateral ref to brel, propagate brel's refs */
1402 if (bms_is_member(rti, brel2->lateral_relids))
1403 brel2->lateral_relids = bms_add_members(brel2->lateral_relids,
1405 }
1406 }
1407
1408 /*
1409 * Now that we've identified all lateral references, mark each baserel
1410 * with the set of relids of rels that reference it laterally (possibly
1411 * indirectly) --- that is, the inverse mapping of lateral_relids.
1412 */
1413 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1414 {
1415 RelOptInfo *brel = root->simple_rel_array[rti];
1416 Relids lateral_relids;
1417 int rti2;
1418
1419 if (brel == NULL || brel->reloptkind != RELOPT_BASEREL)
1420 continue;
1421
1422 /* Nothing to do at rels with no lateral refs */
1423 lateral_relids = brel->lateral_relids;
1424 if (bms_is_empty(lateral_relids))
1425 continue;
1426
1427 /* No rel should have a lateral dependency on itself */
1428 Assert(!bms_is_member(rti, lateral_relids));
1429
1430 /* Mark this rel's referencees */
1431 rti2 = -1;
1432 while ((rti2 = bms_next_member(lateral_relids, rti2)) >= 0)
1433 {
1434 RelOptInfo *brel2 = root->simple_rel_array[rti2];
1435
1436 if (brel2 == NULL)
1437 continue; /* must be an OJ */
1438
1439 Assert(brel2->reloptkind == RELOPT_BASEREL);
1440 brel2->lateral_referencers =
1441 bms_add_member(brel2->lateral_referencers, rti);
1442 }
1443 }
1444}
Bitmapset * bms_intersect(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:292
int bms_next_member(const Bitmapset *a, int prevbit)
Definition bitmapset.c:1290
bool bms_is_member(int x, const Bitmapset *a)
Definition bitmapset.c:510
Bitmapset * bms_add_member(Bitmapset *a, int x)
Definition bitmapset.c:799
bool bms_get_singleton_member(const Bitmapset *a, int *member)
Definition bitmapset.c:708
Bitmapset * bms_copy(const Bitmapset *a)
Definition bitmapset.c:122
unsigned int Index
Definition c.h:698
RelOptInfo * find_base_rel_ignore_join(PlannerInfo *root, int relid)
Definition relnode.c:584
Relids lateral_relids
Definition pathnodes.h:1064

References Assert, bms_add_member(), bms_add_members(), bms_copy(), bms_get_singleton_member(), bms_intersect(), bms_is_empty, bms_is_member(), bms_next_member(), fb(), find_base_rel(), find_base_rel_ignore_join(), find_placeholder_info(), IsA, RelOptInfo::lateral_relids, lfirst, RELOPT_BASEREL, root, and Var::varno.

Referenced by query_planner().

◆ create_plan()

Plan * create_plan ( PlannerInfo root,
Path best_path 
)
extern

Definition at line 339 of file createplan.c.

340{
341 Plan *plan;
342
343 /* plan_params should not be in use in current query level */
344 Assert(root->plan_params == NIL);
345
346 /* Initialize this module's workspace in PlannerInfo */
347 root->curOuterRels = NULL;
348 root->curOuterParams = NIL;
349
350 /* Recursively process the path tree, demanding the correct tlist result */
352
353 /*
354 * Make sure the topmost plan node's targetlist exposes the original
355 * column names and other decorative info. Targetlists generated within
356 * the planner don't bother with that stuff, but we must have it on the
357 * top-level tlist seen at execution time. However, ModifyTable plan
358 * nodes don't have a tlist matching the querytree targetlist.
359 */
360 if (!IsA(plan, ModifyTable))
361 apply_tlist_labeling(plan->targetlist, root->processed_tlist);
362
363 /*
364 * Attach any initPlans created in this query level to the topmost plan
365 * node. (In principle the initplans could go in any plan node at or
366 * above where they're referenced, but there seems no reason to put them
367 * any lower than the topmost node for the query level. Also, see
368 * comments for SS_finalize_plan before you try to change this.)
369 */
371
372 /* Check we successfully assigned all NestLoopParams to plan nodes */
373 if (root->curOuterParams != NIL)
374 elog(ERROR, "failed to assign all NestLoopParams to plan nodes");
375
376 /*
377 * Reset plan_params to ensure param IDs used for nestloop params are not
378 * re-used later
379 */
380 root->plan_params = NIL;
381
382 return plan;
383}
static Plan * create_plan_recurse(PlannerInfo *root, Path *best_path, int flags)
Definition createplan.c:390
#define CP_EXACT_TLIST
Definition createplan.c:69
#define plan(x)
Definition pg_regress.c:164
void SS_attach_initplans(PlannerInfo *root, Plan *plan)
Definition subselect.c:2534
void apply_tlist_labeling(List *dest_tlist, List *src_tlist)
Definition tlist.c:327

References apply_tlist_labeling(), Assert, CP_EXACT_TLIST, create_plan_recurse(), elog, ERROR, fb(), IsA, NIL, plan, root, and SS_attach_initplans().

Referenced by create_minmaxagg_plan(), create_subqueryscan_plan(), make_subplan(), SS_process_ctes(), and standard_planner().

◆ deconstruct_jointree()

List * deconstruct_jointree ( PlannerInfo root)
extern

Definition at line 1471 of file initsplan.c.

1472{
1473 List *result;
1475 List *item_list = NIL;
1476 ListCell *lc;
1477
1478 /*
1479 * After this point, no more PlaceHolderInfos may be made, because
1480 * make_outerjoininfo requires all active placeholders to be present in
1481 * root->placeholder_list while we crawl up the join tree.
1482 */
1483 root->placeholdersFrozen = true;
1484
1485 /* Fetch the already-created top-level join domain for the query */
1486 top_jdomain = linitial_node(JoinDomain, root->join_domains);
1487 top_jdomain->jd_relids = NULL; /* filled during deconstruct_recurse */
1488
1489 /* Start recursion at top of jointree */
1490 Assert(root->parse->jointree != NULL &&
1491 IsA(root->parse->jointree, FromExpr));
1492
1493 /* These are filled as we scan the jointree */
1494 root->all_baserels = NULL;
1495 root->outer_join_rels = NULL;
1496
1497 /* Perform the initial scan of the jointree */
1498 result = deconstruct_recurse(root, (Node *) root->parse->jointree,
1500 &item_list);
1501
1502 /* Now we can form the value of all_query_rels, too */
1503 root->all_query_rels = bms_union(root->all_baserels, root->outer_join_rels);
1504
1505 /* ... which should match what we computed for the top join domain */
1506 Assert(bms_equal(root->all_query_rels, top_jdomain->jd_relids));
1507
1508 /* Now scan all the jointree nodes again, and distribute quals */
1509 foreach(lc, item_list)
1510 {
1512
1514 }
1515
1516 /*
1517 * If there were any special joins then we may have some postponed LEFT
1518 * JOIN clauses to deal with.
1519 */
1520 if (root->join_info_list)
1521 {
1522 foreach(lc, item_list)
1523 {
1525
1526 if (jtitem->oj_joinclauses != NIL)
1528 }
1529 }
1530
1531 /* Don't need the JoinTreeItems any more */
1533
1534 return result;
1535}
bool bms_equal(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:142
Bitmapset * bms_union(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:251
uint32 result
static List * deconstruct_recurse(PlannerInfo *root, Node *jtnode, JoinDomain *parent_domain, JoinTreeItem *parent_jtitem, List **item_list)
Definition initsplan.c:1553
static void deconstruct_distribute_oj_quals(PlannerInfo *root, List *jtitems, JoinTreeItem *jtitem)
Definition initsplan.c:2613
static void deconstruct_distribute(PlannerInfo *root, JoinTreeItem *jtitem)
Definition initsplan.c:1851
void list_free_deep(List *list)
Definition list.c:1560
#define linitial_node(type, l)
Definition pg_list.h:181

References Assert, bms_equal(), bms_union(), deconstruct_distribute(), deconstruct_distribute_oj_quals(), deconstruct_recurse(), fb(), IsA, lfirst, linitial_node, list_free_deep(), NIL, result, and root.

Referenced by query_planner().

◆ distribute_restrictinfo_to_rels()

void distribute_restrictinfo_to_rels ( PlannerInfo root,
RestrictInfo restrictinfo 
)
extern

Definition at line 3578 of file initsplan.c.

3580{
3581 Relids relids = restrictinfo->required_relids;
3582
3583 if (!bms_is_empty(relids))
3584 {
3585 int relid;
3586
3587 if (bms_get_singleton_member(relids, &relid))
3588 {
3589 /*
3590 * There is only one relation participating in the clause, so it
3591 * is a restriction clause for that relation.
3592 */
3594 }
3595 else
3596 {
3597 /*
3598 * The clause is a join clause, since there is more than one rel
3599 * in its relid set.
3600 */
3601
3602 /*
3603 * Check for hashjoinable operators. (We don't bother setting the
3604 * hashjoin info except in true join clauses.)
3605 */
3607
3608 /*
3609 * Likewise, check if the clause is suitable to be used with a
3610 * Memoize node to cache inner tuples during a parameterized
3611 * nested loop.
3612 */
3614
3615 /*
3616 * Add clause to the join lists of all the relevant relations.
3617 */
3619 }
3620 }
3621 else
3622 {
3623 /*
3624 * clause references no rels, and therefore we have no place to attach
3625 * it. Shouldn't get here if callers are working properly.
3626 */
3627 elog(ERROR, "cannot cope with variable-free clause");
3628 }
3629}
static void add_base_clause_to_rel(PlannerInfo *root, Index relid, RestrictInfo *restrictinfo)
Definition initsplan.c:3367
void add_join_clause_to_rels(PlannerInfo *root, RestrictInfo *restrictinfo, Relids join_relids)
Definition joininfo.c:98

References add_base_clause_to_rel(), add_join_clause_to_rels(), bms_get_singleton_member(), bms_is_empty, check_hashjoinable(), check_memoizable(), elog, ERROR, fb(), and root.

Referenced by add_non_redundant_clauses(), distribute_qual_to_rels(), generate_base_implied_equalities_broken(), generate_base_implied_equalities_const(), process_implied_equality(), reconsider_outer_join_clauses(), and remove_leftjoinrel_from_query().

◆ extract_query_dependencies_walker()

bool extract_query_dependencies_walker ( Node node,
PlannerInfo context 
)
extern

Definition at line 3749 of file setrefs.c.

3750{
3751 if (node == NULL)
3752 return false;
3753 Assert(!IsA(node, PlaceHolderVar));
3754 if (IsA(node, Query))
3755 {
3756 Query *query = (Query *) node;
3757 ListCell *lc;
3758
3759 if (query->commandType == CMD_UTILITY)
3760 {
3761 /*
3762 * This logic must handle any utility command for which parse
3763 * analysis was nontrivial (cf. stmt_requires_parse_analysis).
3764 *
3765 * Notably, CALL requires its own processing.
3766 */
3767 if (IsA(query->utilityStmt, CallStmt))
3768 {
3769 CallStmt *callstmt = (CallStmt *) query->utilityStmt;
3770
3771 /* We need not examine funccall, just the transformed exprs */
3773 context);
3775 context);
3776 return false;
3777 }
3778
3779 /*
3780 * Ignore other utility statements, except those (such as EXPLAIN)
3781 * that contain a parsed-but-not-planned query. For those, we
3782 * just need to transfer our attention to the contained query.
3783 */
3784 query = UtilityContainsQuery(query->utilityStmt);
3785 if (query == NULL)
3786 return false;
3787 }
3788
3789 /* Remember if any Query has RLS quals applied by rewriter */
3790 if (query->hasRowSecurity)
3791 context->glob->dependsOnRole = true;
3792
3793 /* Collect relation OIDs in this Query's rtable */
3794 foreach(lc, query->rtable)
3795 {
3797
3798 if (rte->rtekind == RTE_RELATION ||
3799 (rte->rtekind == RTE_SUBQUERY && OidIsValid(rte->relid)) ||
3800 (rte->rtekind == RTE_NAMEDTUPLESTORE && OidIsValid(rte->relid)))
3801 context->glob->relationOids =
3802 lappend_oid(context->glob->relationOids, rte->relid);
3803 }
3804
3805 /* And recurse into the query's subexpressions */
3807 context, 0);
3808 }
3809 /* Extract function dependencies and check for regclass Consts */
3810 fix_expr_common(context, node);
3812 context);
3813}
#define OidIsValid(objectId)
Definition c.h:858
List * lappend_oid(List *list, Oid datum)
Definition list.c:375
#define query_tree_walker(q, w, c, f)
Definition nodeFuncs.h:158
#define expression_tree_walker(n, w, c)
Definition nodeFuncs.h:153
@ CMD_UTILITY
Definition nodes.h:280
@ RTE_NAMEDTUPLESTORE
@ RTE_SUBQUERY
@ RTE_RELATION
static void fix_expr_common(PlannerInfo *root, Node *node)
Definition setrefs.c:2102
bool extract_query_dependencies_walker(Node *node, PlannerInfo *context)
Definition setrefs.c:3749
bool dependsOnRole
Definition pathnodes.h:251
List * relationOids
Definition pathnodes.h:227
PlannerGlobal * glob
Definition pathnodes.h:312
List * rtable
Definition parsenodes.h:178
CmdType commandType
Definition parsenodes.h:121
Node * utilityStmt
Definition parsenodes.h:141
Query * UtilityContainsQuery(Node *parsetree)
Definition utility.c:2199

References Assert, CMD_UTILITY, Query::commandType, PlannerGlobal::dependsOnRole, expression_tree_walker, extract_query_dependencies_walker(), fb(), fix_expr_common(), PlannerInfo::glob, IsA, lappend_oid(), lfirst, OidIsValid, query_tree_walker, PlannerGlobal::relationOids, Query::rtable, RTE_NAMEDTUPLESTORE, RTE_RELATION, RTE_SUBQUERY, UtilityContainsQuery(), and Query::utilityStmt.

Referenced by expression_planner_with_deps(), extract_query_dependencies(), and extract_query_dependencies_walker().

◆ find_lateral_references()

void find_lateral_references ( PlannerInfo root)
extern

Definition at line 1045 of file initsplan.c.

1046{
1047 Index rti;
1048
1049 /* We need do nothing if the query contains no LATERAL RTEs */
1050 if (!root->hasLateralRTEs)
1051 return;
1052
1053 /*
1054 * Examine all baserels (the rel array has been set up by now).
1055 */
1056 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1057 {
1058 RelOptInfo *brel = root->simple_rel_array[rti];
1059
1060 /* there may be empty slots corresponding to non-baserel RTEs */
1061 if (brel == NULL)
1062 continue;
1063
1064 Assert(brel->relid == rti); /* sanity check on array */
1065
1066 /*
1067 * This bit is less obvious than it might look. We ignore appendrel
1068 * otherrels and consider only their parent baserels. In a case where
1069 * a LATERAL-containing UNION ALL subquery was pulled up, it is the
1070 * otherrel that is actually going to be in the plan. However, we
1071 * want to mark all its lateral references as needed by the parent,
1072 * because it is the parent's relid that will be used for join
1073 * planning purposes. And the parent's RTE will contain all the
1074 * lateral references we need to know, since the pulled-up member is
1075 * nothing but a copy of parts of the original RTE's subquery. We
1076 * could visit the parent's children instead and transform their
1077 * references back to the parent's relid, but it would be much more
1078 * complicated for no real gain. (Important here is that the child
1079 * members have not yet received any processing beyond being pulled
1080 * up.) Similarly, in appendrels created by inheritance expansion,
1081 * it's sufficient to look at the parent relation.
1082 */
1083
1084 /* ignore RTEs that are "other rels" */
1085 if (brel->reloptkind != RELOPT_BASEREL)
1086 continue;
1087
1089 }
1090}
static void extract_lateral_references(PlannerInfo *root, RelOptInfo *brel, Index rtindex)
Definition initsplan.c:1093

References Assert, extract_lateral_references(), fb(), RELOPT_BASEREL, and root.

Referenced by query_planner().

◆ find_minmax_agg_replacement_param()

Param * find_minmax_agg_replacement_param ( PlannerInfo root,
Aggref aggref 
)
extern

Definition at line 3599 of file setrefs.c.

3600{
3601 if (root->minmax_aggs != NIL &&
3602 list_length(aggref->args) == 1)
3603 {
3605 ListCell *lc;
3606
3607 foreach(lc, root->minmax_aggs)
3608 {
3610
3611 if (mminfo->aggfnoid == aggref->aggfnoid &&
3612 equal(mminfo->target, curTarget->expr))
3613 return mminfo->param;
3614 }
3615 }
3616 return NULL;
3617}
bool equal(const void *a, const void *b)
Definition equalfuncs.c:223
static int list_length(const List *l)
Definition pg_list.h:152
#define linitial(l)
Definition pg_list.h:178
Oid aggfnoid
Definition primnodes.h:464
List * args
Definition primnodes.h:488
Param * param
Definition pathnodes.h:3465

References Aggref::aggfnoid, Aggref::args, equal(), fb(), lfirst, linitial, list_length(), NIL, MinMaxAggInfo::param, and root.

Referenced by finalize_primnode(), fix_scan_expr_mutator(), and fix_upper_expr_mutator().

◆ innerrel_is_unique()

bool innerrel_is_unique ( PlannerInfo root,
Relids  joinrelids,
Relids  outerrelids,
RelOptInfo innerrel,
JoinType  jointype,
List restrictlist,
bool  force_cache 
)
extern

Definition at line 1340 of file analyzejoins.c.

1347{
1348 return innerrel_is_unique_ext(root, joinrelids, outerrelids, innerrel,
1349 jointype, restrictlist, force_cache, NULL);
1350}
bool innerrel_is_unique_ext(PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, bool force_cache, List **extra_clauses)

References fb(), innerrel_is_unique_ext(), and root.

Referenced by add_paths_to_joinrel(), and reduce_unique_semijoins().

◆ innerrel_is_unique_ext()

bool innerrel_is_unique_ext ( PlannerInfo root,
Relids  joinrelids,
Relids  outerrelids,
RelOptInfo innerrel,
JoinType  jointype,
List restrictlist,
bool  force_cache,
List **  extra_clauses 
)
extern

Definition at line 1362 of file analyzejoins.c.

1370{
1372 ListCell *lc;
1374 List *outer_exprs = NIL;
1375 bool self_join = (extra_clauses != NULL);
1376
1377 /* Certainly can't prove uniqueness when there are no joinclauses */
1378 if (restrictlist == NIL)
1379 return false;
1380
1381 /*
1382 * Make a quick check to eliminate cases in which we will surely be unable
1383 * to prove uniqueness of the innerrel.
1384 */
1385 if (!rel_supports_distinctness(root, innerrel))
1386 return false;
1387
1388 /*
1389 * Query the cache to see if we've managed to prove that innerrel is
1390 * unique for any subset of this outerrel. For non-self-join search, we
1391 * don't need an exact match, as extra outerrels can't make the innerrel
1392 * any less unique (or more formally, the restrictlist for a join to a
1393 * superset outerrel must be a superset of the conditions we successfully
1394 * used before). For self-join search, we require an exact match of
1395 * outerrels because we need extra clauses to be valid for our case. Also,
1396 * for self-join checking we've filtered the clauses list. Thus, we can
1397 * match only the result cached for a self-join search for another
1398 * self-join check.
1399 */
1400 foreach(lc, innerrel->unique_for_rels)
1401 {
1403
1404 if ((!self_join && bms_is_subset(uniqueRelInfo->outerrelids, outerrelids)) ||
1405 (self_join && bms_equal(uniqueRelInfo->outerrelids, outerrelids) &&
1406 uniqueRelInfo->self_join))
1407 {
1408 if (extra_clauses)
1409 *extra_clauses = uniqueRelInfo->extra_clauses;
1410 return true; /* Success! */
1411 }
1412 }
1413
1414 /*
1415 * Conversely, we may have already determined that this outerrel, or some
1416 * superset thereof, cannot prove this innerrel to be unique.
1417 */
1418 foreach(lc, innerrel->non_unique_for_rels)
1419 {
1420 Relids unique_for_rels = (Relids) lfirst(lc);
1421
1422 if (bms_is_subset(outerrelids, unique_for_rels))
1423 return false;
1424 }
1425
1426 /* No cached information, so try to make the proof. */
1427 if (is_innerrel_unique_for(root, joinrelids, outerrelids, innerrel,
1428 jointype, restrictlist,
1429 self_join ? &outer_exprs : NULL))
1430 {
1431 /*
1432 * Cache the positive result for future probes, being sure to keep it
1433 * in the planner_cxt even if we are working in GEQO.
1434 *
1435 * Note: one might consider trying to isolate the minimal subset of
1436 * the outerrels that proved the innerrel unique. But it's not worth
1437 * the trouble, because the planner builds up joinrels incrementally
1438 * and so we'll see the minimally sufficient outerrels before any
1439 * supersets of them anyway.
1440 */
1441 old_context = MemoryContextSwitchTo(root->planner_cxt);
1443 uniqueRelInfo->outerrelids = bms_copy(outerrelids);
1444 uniqueRelInfo->self_join = self_join;
1445 uniqueRelInfo->extra_clauses = outer_exprs;
1446 innerrel->unique_for_rels = lappend(innerrel->unique_for_rels,
1449
1450 if (extra_clauses)
1451 *extra_clauses = outer_exprs;
1452 return true; /* Success! */
1453 }
1454 else
1455 {
1456 /*
1457 * None of the join conditions for outerrel proved innerrel unique, so
1458 * we can safely reject this outerrel or any subset of it in future
1459 * checks.
1460 *
1461 * However, in normal planning mode, caching this knowledge is totally
1462 * pointless; it won't be queried again, because we build up joinrels
1463 * from smaller to larger. It's only useful when using GEQO or
1464 * another planner extension that attempts planning multiple times.
1465 *
1466 * Also, allow callers to override that heuristic and force caching;
1467 * that's useful for reduce_unique_semijoins, which calls here before
1468 * the normal join search starts.
1469 */
1470 if (force_cache || root->assumeReplanning)
1471 {
1472 old_context = MemoryContextSwitchTo(root->planner_cxt);
1473 innerrel->non_unique_for_rels =
1474 lappend(innerrel->non_unique_for_rels,
1475 bms_copy(outerrelids));
1477 }
1478
1479 return false;
1480 }
1481}
static bool is_innerrel_unique_for(PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, List **extra_clauses)
static bool rel_supports_distinctness(PlannerInfo *root, RelOptInfo *rel)
#define makeNode(_type_)
Definition nodes.h:161
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
Definition palloc.h:124
Bitmapset * Relids
Definition pathnodes.h:103
List * unique_for_rels
Definition pathnodes.h:1124
List * non_unique_for_rels
Definition pathnodes.h:1126

References bms_copy(), bms_equal(), bms_is_subset(), fb(), is_innerrel_unique_for(), lappend(), lfirst, makeNode, MemoryContextSwitchTo(), NIL, RelOptInfo::non_unique_for_rels, rel_supports_distinctness(), root, and RelOptInfo::unique_for_rels.

Referenced by innerrel_is_unique(), and remove_self_joins_one_group().

◆ is_projection_capable_path()

bool is_projection_capable_path ( Path path)
extern

Definition at line 7231 of file createplan.c.

7232{
7233 /* Most plan types can project, so just list the ones that can't */
7234 switch (path->pathtype)
7235 {
7236 case T_Hash:
7237 case T_Material:
7238 case T_Memoize:
7239 case T_Sort:
7240 case T_IncrementalSort:
7241 case T_Unique:
7242 case T_SetOp:
7243 case T_LockRows:
7244 case T_Limit:
7245 case T_ModifyTable:
7246 case T_MergeAppend:
7247 case T_RecursiveUnion:
7248 return false;
7249 case T_CustomScan:
7251 return true;
7252 return false;
7253 case T_Append:
7254
7255 /*
7256 * Append can't project, but if an AppendPath is being used to
7257 * represent a dummy path, what will actually be generated is a
7258 * Result which can project.
7259 */
7260 return IS_DUMMY_APPEND(path);
7261 case T_ProjectSet:
7262
7263 /*
7264 * Although ProjectSet certainly projects, say "no" because we
7265 * don't want the planner to randomly replace its tlist with
7266 * something else; the SRFs have to stay at top level. This might
7267 * get relaxed later.
7268 */
7269 return false;
7270 default:
7271 break;
7272 }
7273 return true;
7274}
#define CUSTOMPATH_SUPPORT_PROJECTION
Definition extensible.h:86
#define castNode(_type_, nodeptr)
Definition nodes.h:182
#define IS_DUMMY_APPEND(p)
Definition pathnodes.h:2291
NodeTag pathtype
Definition pathnodes.h:1971

References castNode, CUSTOMPATH_SUPPORT_PROJECTION, fb(), IS_DUMMY_APPEND, and Path::pathtype.

Referenced by add_paths_with_pathkeys_for_rel(), apply_projection_to_path(), create_projection_path(), and create_projection_plan().

◆ is_projection_capable_plan()

bool is_projection_capable_plan ( Plan plan)
extern

Definition at line 7281 of file createplan.c.

7282{
7283 /* Most plan types can project, so just list the ones that can't */
7284 switch (nodeTag(plan))
7285 {
7286 case T_Hash:
7287 case T_Material:
7288 case T_Memoize:
7289 case T_Sort:
7290 case T_Unique:
7291 case T_SetOp:
7292 case T_LockRows:
7293 case T_Limit:
7294 case T_ModifyTable:
7295 case T_Append:
7296 case T_MergeAppend:
7297 case T_RecursiveUnion:
7298 return false;
7299 case T_CustomScan:
7301 return true;
7302 return false;
7303 case T_ProjectSet:
7304
7305 /*
7306 * Although ProjectSet certainly projects, say "no" because we
7307 * don't want the planner to randomly replace its tlist with
7308 * something else; the SRFs have to stay at top level. This might
7309 * get relaxed later.
7310 */
7311 return false;
7312 default:
7313 break;
7314 }
7315 return true;
7316}

References CUSTOMPATH_SUPPORT_PROJECTION, fb(), nodeTag, and plan.

Referenced by change_plan_targetlist(), create_projection_plan(), and prepare_sort_from_pathkeys().

◆ make_agg()

Agg * make_agg ( List tlist,
List qual,
AggStrategy  aggstrategy,
AggSplit  aggsplit,
int  numGroupCols,
AttrNumber grpColIdx,
Oid grpOperators,
Oid grpCollations,
List groupingSets,
List chain,
Cardinality  numGroups,
Size  transitionSpace,
Plan lefttree 
)
extern

Definition at line 6583 of file createplan.c.

6588{
6589 Agg *node = makeNode(Agg);
6590 Plan *plan = &node->plan;
6591
6592 node->aggstrategy = aggstrategy;
6593 node->aggsplit = aggsplit;
6594 node->numCols = numGroupCols;
6595 node->grpColIdx = grpColIdx;
6596 node->grpOperators = grpOperators;
6597 node->grpCollations = grpCollations;
6598 node->numGroups = numGroups;
6599 node->transitionSpace = transitionSpace;
6600 node->aggParams = NULL; /* SS_finalize_plan() will fill this */
6601 node->groupingSets = groupingSets;
6602 node->chain = chain;
6603
6604 plan->qual = qual;
6605 plan->targetlist = tlist;
6606 plan->lefttree = lefttree;
6607 plan->righttree = NULL;
6608
6609 return node;
6610}
AggSplit aggsplit
Definition plannodes.h:1217
List * chain
Definition plannodes.h:1244
List * groupingSets
Definition plannodes.h:1241
Bitmapset * aggParams
Definition plannodes.h:1236
Cardinality numGroups
Definition plannodes.h:1230
Plan plan
Definition plannodes.h:1211
int numCols
Definition plannodes.h:1220
uint64 transitionSpace
Definition plannodes.h:1233
AggStrategy aggstrategy
Definition plannodes.h:1214
struct Plan * righttree
Definition plannodes.h:240

References Agg::aggParams, Agg::aggsplit, Agg::aggstrategy, Agg::chain, fb(), Agg::groupingSets, makeNode, Agg::numCols, Agg::numGroups, Agg::plan, plan, Plan::righttree, and Agg::transitionSpace.

Referenced by create_agg_plan(), and create_groupingsets_plan().

◆ make_foreignscan()

ForeignScan * make_foreignscan ( List qptlist,
List qpqual,
Index  scanrelid,
List fdw_exprs,
List fdw_private,
List fdw_scan_tlist,
List fdw_recheck_quals,
Plan outer_plan 
)
extern

Definition at line 5803 of file createplan.c.

5811{
5813 Plan *plan = &node->scan.plan;
5814
5815 /* cost will be filled in by create_foreignscan_plan */
5816 plan->targetlist = qptlist;
5817 plan->qual = qpqual;
5818 plan->lefttree = outer_plan;
5819 plan->righttree = NULL;
5820 node->scan.scanrelid = scanrelid;
5821
5822 /* these may be overridden by the FDW's PlanDirectModify callback. */
5823 node->operation = CMD_SELECT;
5824 node->resultRelation = 0;
5825
5826 /* checkAsUser, fs_server will be filled in by create_foreignscan_plan */
5827 node->checkAsUser = InvalidOid;
5828 node->fs_server = InvalidOid;
5829 node->fdw_exprs = fdw_exprs;
5830 node->fdw_private = fdw_private;
5831 node->fdw_scan_tlist = fdw_scan_tlist;
5832 node->fdw_recheck_quals = fdw_recheck_quals;
5833 /* fs_relids, fs_base_relids will be filled by create_foreignscan_plan */
5834 node->fs_relids = NULL;
5835 node->fs_base_relids = NULL;
5836 /* fsSystemCol will be filled in by create_foreignscan_plan */
5837 node->fsSystemCol = false;
5838
5839 return node;
5840}
@ CMD_SELECT
Definition nodes.h:275
Oid checkAsUser
Definition plannodes.h:898
CmdType operation
Definition plannodes.h:894
List * fdw_exprs
Definition plannodes.h:902
bool fsSystemCol
Definition plannodes.h:914
Bitmapset * fs_relids
Definition plannodes.h:910
List * fdw_private
Definition plannodes.h:904
Bitmapset * fs_base_relids
Definition plannodes.h:912
Index resultRelation
Definition plannodes.h:896
List * fdw_recheck_quals
Definition plannodes.h:908
List * fdw_scan_tlist
Definition plannodes.h:906
Index scanrelid
Definition plannodes.h:544

References ForeignScan::checkAsUser, CMD_SELECT, fb(), ForeignScan::fdw_exprs, ForeignScan::fdw_private, ForeignScan::fdw_recheck_quals, ForeignScan::fdw_scan_tlist, ForeignScan::fs_base_relids, ForeignScan::fs_relids, ForeignScan::fs_server, ForeignScan::fsSystemCol, InvalidOid, makeNode, ForeignScan::operation, plan, ForeignScan::resultRelation, Plan::righttree, ForeignScan::scan, and Scan::scanrelid.

Referenced by fileGetForeignPlan(), and postgresGetForeignPlan().

◆ make_limit()

Limit * make_limit ( Plan lefttree,
Node limitOffset,
Node limitCount,
LimitOption  limitOption,
int  uniqNumCols,
AttrNumber uniqColIdx,
Oid uniqOperators,
Oid uniqCollations 
)
extern

Definition at line 6902 of file createplan.c.

6905{
6906 Limit *node = makeNode(Limit);
6907 Plan *plan = &node->plan;
6908
6909 plan->targetlist = lefttree->targetlist;
6910 plan->qual = NIL;
6911 plan->lefttree = lefttree;
6912 plan->righttree = NULL;
6913
6914 node->limitOffset = limitOffset;
6915 node->limitCount = limitCount;
6916 node->limitOption = limitOption;
6917 node->uniqNumCols = uniqNumCols;
6918 node->uniqColIdx = uniqColIdx;
6919 node->uniqOperators = uniqOperators;
6920 node->uniqCollations = uniqCollations;
6921
6922 return node;
6923}
LimitOption limitOption
Definition plannodes.h:1509
Plan plan
Definition plannodes.h:1500
Node * limitCount
Definition plannodes.h:1506
int uniqNumCols
Definition plannodes.h:1512
Node * limitOffset
Definition plannodes.h:1503

References fb(), Limit::limitCount, Limit::limitOffset, Limit::limitOption, makeNode, NIL, Limit::plan, plan, Plan::righttree, Plan::targetlist, and Limit::uniqNumCols.

Referenced by create_limit_plan(), and create_minmaxagg_plan().

◆ make_sort_from_sortclauses()

Sort * make_sort_from_sortclauses ( List sortcls,
Plan lefttree 
)
extern

Definition at line 6397 of file createplan.c.

6398{
6399 List *sub_tlist = lefttree->targetlist;
6400 ListCell *l;
6401 int numsortkeys;
6402 AttrNumber *sortColIdx;
6403 Oid *sortOperators;
6404 Oid *collations;
6405 bool *nullsFirst;
6406
6407 /* Convert list-ish representation to arrays wanted by executor */
6409 sortColIdx = (AttrNumber *) palloc(numsortkeys * sizeof(AttrNumber));
6410 sortOperators = (Oid *) palloc(numsortkeys * sizeof(Oid));
6411 collations = (Oid *) palloc(numsortkeys * sizeof(Oid));
6412 nullsFirst = (bool *) palloc(numsortkeys * sizeof(bool));
6413
6414 numsortkeys = 0;
6415 foreach(l, sortcls)
6416 {
6419
6420 sortColIdx[numsortkeys] = tle->resno;
6421 sortOperators[numsortkeys] = sortcl->sortop;
6422 collations[numsortkeys] = exprCollation((Node *) tle->expr);
6423 nullsFirst[numsortkeys] = sortcl->nulls_first;
6424 numsortkeys++;
6425 }
6426
6427 return make_sort(lefttree, numsortkeys,
6428 sortColIdx, sortOperators,
6429 collations, nullsFirst);
6430}
int16 AttrNumber
Definition attnum.h:21
static Sort * make_sort(Plan *lefttree, int numCols, AttrNumber *sortColIdx, Oid *sortOperators, Oid *collations, bool *nullsFirst)
void * palloc(Size size)
Definition mcxt.c:1387
Oid exprCollation(const Node *expr)
Definition nodeFuncs.c:826
unsigned int Oid
TargetEntry * get_sortgroupclause_tle(SortGroupClause *sgClause, List *targetList)
Definition tlist.c:376

References exprCollation(), fb(), get_sortgroupclause_tle(), lfirst, list_length(), make_sort(), palloc(), and Plan::targetlist.

◆ match_foreign_keys_to_quals()

void match_foreign_keys_to_quals ( PlannerInfo root)
extern

Definition at line 3982 of file initsplan.c.

3983{
3984 List *newlist = NIL;
3985 ListCell *lc;
3986
3987 foreach(lc, root->fkey_list)
3988 {
3992 int colno;
3993
3994 /*
3995 * Either relid might identify a rel that is in the query's rtable but
3996 * isn't referenced by the jointree, or has been removed by join
3997 * removal, so that it won't have a RelOptInfo. Hence don't use
3998 * find_base_rel() here. We can ignore such FKs.
3999 */
4000 if (fkinfo->con_relid >= root->simple_rel_array_size ||
4001 fkinfo->ref_relid >= root->simple_rel_array_size)
4002 continue; /* just paranoia */
4003 con_rel = root->simple_rel_array[fkinfo->con_relid];
4004 if (con_rel == NULL)
4005 continue;
4006 ref_rel = root->simple_rel_array[fkinfo->ref_relid];
4007 if (ref_rel == NULL)
4008 continue;
4009
4010 /*
4011 * Ignore FK unless both rels are baserels. This gets rid of FKs that
4012 * link to inheritance child rels (otherrels).
4013 */
4014 if (con_rel->reloptkind != RELOPT_BASEREL ||
4015 ref_rel->reloptkind != RELOPT_BASEREL)
4016 continue;
4017
4018 /*
4019 * Scan the columns and try to match them to eclasses and quals.
4020 *
4021 * Note: for simple inner joins, any match should be in an eclass.
4022 * "Loose" quals that syntactically match an FK equality must have
4023 * been rejected for EC status because they are outer-join quals or
4024 * similar. We can still consider them to match the FK.
4025 */
4026 for (colno = 0; colno < fkinfo->nkeys; colno++)
4027 {
4028 EquivalenceClass *ec;
4030 ref_attno;
4031 Oid fpeqop;
4032 ListCell *lc2;
4033
4035 /* Don't bother looking for loose quals if we got an EC match */
4036 if (ec != NULL)
4037 {
4038 fkinfo->nmatched_ec++;
4039 if (ec->ec_has_const)
4040 fkinfo->nconst_ec++;
4041 continue;
4042 }
4043
4044 /*
4045 * Scan joininfo list for relevant clauses. Either rel's joininfo
4046 * list would do equally well; we use con_rel's.
4047 */
4048 con_attno = fkinfo->conkey[colno];
4049 ref_attno = fkinfo->confkey[colno];
4050 fpeqop = InvalidOid; /* we'll look this up only if needed */
4051
4052 foreach(lc2, con_rel->joininfo)
4053 {
4054 RestrictInfo *rinfo = (RestrictInfo *) lfirst(lc2);
4055 OpExpr *clause = (OpExpr *) rinfo->clause;
4056 Var *leftvar;
4057 Var *rightvar;
4058
4059 /* Only binary OpExprs are useful for consideration */
4060 if (!IsA(clause, OpExpr) ||
4061 list_length(clause->args) != 2)
4062 continue;
4063 leftvar = (Var *) get_leftop((Expr *) clause);
4064 rightvar = (Var *) get_rightop((Expr *) clause);
4065
4066 /* Operands must be Vars, possibly with RelabelType */
4067 while (leftvar && IsA(leftvar, RelabelType))
4068 leftvar = (Var *) ((RelabelType *) leftvar)->arg;
4069 if (!(leftvar && IsA(leftvar, Var)))
4070 continue;
4071 while (rightvar && IsA(rightvar, RelabelType))
4072 rightvar = (Var *) ((RelabelType *) rightvar)->arg;
4073 if (!(rightvar && IsA(rightvar, Var)))
4074 continue;
4075
4076 /* Now try to match the vars to the current foreign key cols */
4077 if (fkinfo->ref_relid == leftvar->varno &&
4078 ref_attno == leftvar->varattno &&
4079 fkinfo->con_relid == rightvar->varno &&
4080 con_attno == rightvar->varattno)
4081 {
4082 /* Vars match, but is it the right operator? */
4083 if (clause->opno == fkinfo->conpfeqop[colno])
4084 {
4085 fkinfo->rinfos[colno] = lappend(fkinfo->rinfos[colno],
4086 rinfo);
4087 fkinfo->nmatched_ri++;
4088 }
4089 }
4090 else if (fkinfo->ref_relid == rightvar->varno &&
4091 ref_attno == rightvar->varattno &&
4092 fkinfo->con_relid == leftvar->varno &&
4093 con_attno == leftvar->varattno)
4094 {
4095 /*
4096 * Reverse match, must check commutator operator. Look it
4097 * up if we didn't already. (In the worst case we might
4098 * do multiple lookups here, but that would require an FK
4099 * equality operator without commutator, which is
4100 * unlikely.)
4101 */
4102 if (!OidIsValid(fpeqop))
4103 fpeqop = get_commutator(fkinfo->conpfeqop[colno]);
4104 if (clause->opno == fpeqop)
4105 {
4106 fkinfo->rinfos[colno] = lappend(fkinfo->rinfos[colno],
4107 rinfo);
4108 fkinfo->nmatched_ri++;
4109 }
4110 }
4111 }
4112 /* If we found any matching loose quals, count col as matched */
4113 if (fkinfo->rinfos[colno])
4114 fkinfo->nmatched_rcols++;
4115 }
4116
4117 /*
4118 * Currently, we drop multicolumn FKs that aren't fully matched to the
4119 * query. Later we might figure out how to derive some sort of
4120 * estimate from them, in which case this test should be weakened to
4121 * "if ((fkinfo->nmatched_ec + fkinfo->nmatched_rcols) > 0)".
4122 */
4123 if ((fkinfo->nmatched_ec + fkinfo->nmatched_rcols) == fkinfo->nkeys)
4125 }
4126 /* Replace fkey_list, thereby discarding any useless entries */
4127 root->fkey_list = newlist;
4128}
EquivalenceClass * match_eclasses_to_foreign_key_col(PlannerInfo *root, ForeignKeyOptInfo *fkinfo, int colno)
Oid get_commutator(Oid opno)
Definition lsyscache.c:1702
static Node * get_rightop(const void *clause)
Definition nodeFuncs.h:95
static Node * get_leftop(const void *clause)
Definition nodeFuncs.h:83
Expr * clause
Definition pathnodes.h:2901

References OpExpr::args, RestrictInfo::clause, EquivalenceClass::ec_has_const, fb(), get_commutator(), get_leftop(), get_rightop(), InvalidOid, IsA, lappend(), lfirst, list_length(), match_eclasses_to_foreign_key_col(), NIL, OidIsValid, OpExpr::opno, RELOPT_BASEREL, and root.

Referenced by query_planner().

◆ materialize_finished_plan()

Plan * materialize_finished_plan ( Plan subplan)
extern

Definition at line 6509 of file createplan.c.

6510{
6511 Plan *matplan;
6512 Path matpath; /* dummy for cost_material */
6514 bool unsafe_initplans;
6515
6516 matplan = (Plan *) make_material(subplan);
6517
6518 /*
6519 * XXX horrid kluge: if there are any initPlans attached to the subplan,
6520 * move them up to the Material node, which is now effectively the top
6521 * plan node in its query level. This prevents failure in
6522 * SS_finalize_plan(), which see for comments.
6523 */
6524 matplan->initPlan = subplan->initPlan;
6525 subplan->initPlan = NIL;
6526
6527 /* Move the initplans' cost delta, as well */
6530 subplan->startup_cost -= initplan_cost;
6531 subplan->total_cost -= initplan_cost;
6532
6533 /* Set cost data */
6536 subplan->disabled_nodes,
6537 subplan->startup_cost,
6538 subplan->total_cost,
6539 subplan->plan_rows,
6540 subplan->plan_width);
6541 matplan->disabled_nodes = subplan->disabled_nodes;
6542 matplan->startup_cost = matpath.startup_cost + initplan_cost;
6543 matplan->total_cost = matpath.total_cost + initplan_cost;
6544 matplan->plan_rows = subplan->plan_rows;
6545 matplan->plan_width = subplan->plan_width;
6546 matplan->parallel_aware = false;
6547 matplan->parallel_safe = subplan->parallel_safe;
6548
6549 return matplan;
6550}
void cost_material(Path *path, bool enabled, int input_disabled_nodes, Cost input_startup_cost, Cost input_total_cost, double tuples, int width)
Definition costsize.c:2583
bool enable_material
Definition costsize.c:155
static Material * make_material(Plan *lefttree)
double Cost
Definition nodes.h:261
Cost total_cost
Definition plannodes.h:205
Cost startup_cost
Definition plannodes.h:203
int plan_width
Definition plannodes.h:213
Cardinality plan_rows
Definition plannodes.h:211
int disabled_nodes
Definition plannodes.h:201
List * initPlan
Definition plannodes.h:242
void SS_compute_initplan_cost(List *init_plans, Cost *initplan_cost_p, bool *unsafe_initplans_p)
Definition subselect.c:2493

References cost_material(), Plan::disabled_nodes, enable_material, fb(), Plan::initPlan, make_material(), NIL, Plan::parallel_safe, Plan::plan_rows, Plan::plan_width, SS_compute_initplan_cost(), Plan::startup_cost, and Plan::total_cost.

Referenced by build_subplan(), and standard_planner().

◆ preprocess_minmax_aggregates()

void preprocess_minmax_aggregates ( PlannerInfo root)
extern

Definition at line 74 of file planagg.c.

75{
76 Query *parse = root->parse;
77 FromExpr *jtnode;
81 RelOptInfo *grouped_rel;
82 ListCell *lc;
83
84 /* minmax_aggs list should be empty at this point */
85 Assert(root->minmax_aggs == NIL);
86
87 /* Nothing to do if query has no aggregates */
88 if (!parse->hasAggs)
89 return;
90
91 Assert(!parse->setOperations); /* shouldn't get here if a setop */
92 Assert(parse->rowMarks == NIL); /* nor if FOR UPDATE */
93
94 /*
95 * Reject unoptimizable cases.
96 *
97 * We don't handle GROUP BY or windowing, because our current
98 * implementations of grouping require looking at all the rows anyway, and
99 * so there's not much point in optimizing MIN/MAX.
100 */
101 if (parse->groupClause || list_length(parse->groupingSets) > 1 ||
102 parse->hasWindowFuncs)
103 return;
104
105 /*
106 * Reject if query contains any CTEs; there's no way to build an indexscan
107 * on one so we couldn't succeed here. (If the CTEs are unreferenced,
108 * that's not true, but it doesn't seem worth expending cycles to check.)
109 */
110 if (parse->cteList)
111 return;
112
113 /*
114 * We also restrict the query to reference exactly one table, since join
115 * conditions can't be handled reasonably. (We could perhaps handle a
116 * query containing cartesian-product joins, but it hardly seems worth the
117 * trouble.) However, the single table could be buried in several levels
118 * of FromExpr due to subqueries. Note the "single" table could be an
119 * inheritance parent, too, including the case of a UNION ALL subquery
120 * that's been flattened to an appendrel.
121 */
122 jtnode = parse->jointree;
123 while (IsA(jtnode, FromExpr))
124 {
125 if (list_length(jtnode->fromlist) != 1)
126 return;
127 jtnode = linitial(jtnode->fromlist);
128 }
129 if (!IsA(jtnode, RangeTblRef))
130 return;
131 rtr = (RangeTblRef *) jtnode;
132 rte = planner_rt_fetch(rtr->rtindex, root);
133 if (rte->rtekind == RTE_RELATION)
134 /* ordinary relation, ok */ ;
135 else if (rte->rtekind == RTE_SUBQUERY && rte->inh)
136 /* flattened UNION ALL subquery, ok */ ;
137 else
138 return;
139
140 /*
141 * Examine all the aggregates and verify all are MIN/MAX aggregates. Stop
142 * as soon as we find one that isn't.
143 */
144 aggs_list = NIL;
146 return;
147
148 /*
149 * OK, there is at least the possibility of performing the optimization.
150 * Build an access path for each aggregate. If any of the aggregates
151 * prove to be non-indexable, give up; there is no point in optimizing
152 * just some of them.
153 */
154 foreach(lc, aggs_list)
155 {
157 Oid eqop;
158 bool reverse;
159
160 /*
161 * We'll need the equality operator that goes with the aggregate's
162 * ordering operator.
163 */
164 eqop = get_equality_op_for_ordering_op(mminfo->aggsortop, &reverse);
165 if (!OidIsValid(eqop)) /* shouldn't happen */
166 elog(ERROR, "could not find equality operator for ordering operator %u",
167 mminfo->aggsortop);
168
169 /*
170 * We can use either an ordering that gives NULLS FIRST or one that
171 * gives NULLS LAST; furthermore there's unlikely to be much
172 * performance difference between them, so it doesn't seem worth
173 * costing out both ways if we get a hit on the first one. NULLS
174 * FIRST is more likely to be available if the operator is a
175 * reverse-sort operator, so try that first if reverse.
176 */
177 if (build_minmax_path(root, mminfo, eqop, mminfo->aggsortop, reverse, reverse))
178 continue;
179 if (build_minmax_path(root, mminfo, eqop, mminfo->aggsortop, reverse, !reverse))
180 continue;
181
182 /* No indexable path for this aggregate, so fail */
183 return;
184 }
185
186 /*
187 * OK, we can do the query this way. Prepare to create a MinMaxAggPath
188 * node.
189 *
190 * First, create an output Param node for each agg. (If we end up not
191 * using the MinMaxAggPath, we'll waste a PARAM_EXEC slot for each agg,
192 * which is not worth worrying about. We can't wait till create_plan time
193 * to decide whether to make the Param, unfortunately.)
194 */
195 foreach(lc, aggs_list)
196 {
198
199 mminfo->param =
201 exprType((Node *) mminfo->target),
202 -1,
203 exprCollation((Node *) mminfo->target));
204 }
205
206 /*
207 * Create a MinMaxAggPath node with the appropriate estimated costs and
208 * other needed data, and add it to the UPPERREL_GROUP_AGG upperrel, where
209 * it will compete against the standard aggregate implementation. (It
210 * will likely always win, but we need not assume that here.)
211 *
212 * Note: grouping_planner won't have created this upperrel yet, but it's
213 * fine for us to create it first. We will not have inserted the correct
214 * consider_parallel value in it, but MinMaxAggPath paths are currently
215 * never parallel-safe anyway, so that doesn't matter. Likewise, it
216 * doesn't matter that we haven't filled FDW-related fields in the rel.
217 * Also, because there are no rowmarks, we know that the processed_tlist
218 * doesn't need to change anymore, so making the pathtarget now is safe.
219 */
221 add_path(grouped_rel, (Path *)
222 create_minmaxagg_path(root, grouped_rel,
224 root->processed_tlist),
225 aggs_list,
226 (List *) parse->havingQual));
227}
void parse(int)
Definition parse.c:49
Oid get_equality_op_for_ordering_op(Oid opno, bool *reverse)
Definition lsyscache.c:326
Oid exprType(const Node *expr)
Definition nodeFuncs.c:42
MinMaxAggPath * create_minmaxagg_path(PlannerInfo *root, RelOptInfo *rel, PathTarget *target, List *mmaggregates, List *quals)
Definition pathnode.c:3302
void add_path(RelOptInfo *parent_rel, Path *new_path)
Definition pathnode.c:459
#define planner_rt_fetch(rti, root)
Definition pathnodes.h:704
@ UPPERREL_GROUP_AGG
Definition pathnodes.h:147
static bool can_minmax_aggs(PlannerInfo *root, List **context)
Definition planagg.c:238
static bool build_minmax_path(PlannerInfo *root, MinMaxAggInfo *mminfo, Oid eqop, Oid sortop, bool reverse_sort, bool nulls_first)
Definition planagg.c:318
RelOptInfo * fetch_upper_rel(PlannerInfo *root, UpperRelationKind kind, Relids relids)
Definition relnode.c:1617
Param * SS_make_initplan_output_param(PlannerInfo *root, Oid resulttype, int32 resulttypmod, Oid resultcollation)
Definition subselect.c:3294
#define create_pathtarget(root, tlist)
Definition tlist.h:58

References add_path(), Assert, build_minmax_path(), can_minmax_aggs(), create_minmaxagg_path(), create_pathtarget, elog, ERROR, exprCollation(), exprType(), fb(), fetch_upper_rel(), FromExpr::fromlist, get_equality_op_for_ordering_op(), IsA, lfirst, linitial, list_length(), NIL, OidIsValid, MinMaxAggInfo::param, parse(), planner_rt_fetch, root, RTE_RELATION, RTE_SUBQUERY, SS_make_initplan_output_param(), and UPPERREL_GROUP_AGG.

Referenced by grouping_planner().

◆ process_implied_equality()

RestrictInfo * process_implied_equality ( PlannerInfo root,
Oid  opno,
Oid  collation,
Expr item1,
Expr item2,
Relids  qualscope,
Index  security_level,
bool  both_const 
)
extern

Definition at line 3663 of file initsplan.c.

3671{
3673 Node *clause;
3674 Relids relids;
3675 bool pseudoconstant = false;
3676
3677 /*
3678 * Build the new clause. Copy to ensure it shares no substructure with
3679 * original (this is necessary in case there are subselects in there...)
3680 */
3681 clause = (Node *) make_opclause(opno,
3682 BOOLOID, /* opresulttype */
3683 false, /* opretset */
3686 InvalidOid,
3687 collation);
3688
3689 /* If both constant, try to reduce to a boolean constant. */
3690 if (both_const)
3691 {
3692 clause = eval_const_expressions(root, clause);
3693
3694 /* If we produced const TRUE, just drop the clause */
3695 if (clause && IsA(clause, Const))
3696 {
3697 Const *cclause = (Const *) clause;
3698
3699 Assert(cclause->consttype == BOOLOID);
3700 if (!cclause->constisnull && DatumGetBool(cclause->constvalue))
3701 return NULL;
3702 }
3703 }
3704
3705 /*
3706 * The rest of this is a very cut-down version of distribute_qual_to_rels.
3707 * We can skip most of the work therein, but there are a couple of special
3708 * cases we still have to handle.
3709 *
3710 * Retrieve all relids mentioned within the possibly-simplified clause.
3711 */
3712 relids = pull_varnos(root, clause);
3713 Assert(bms_is_subset(relids, qualscope));
3714
3715 /*
3716 * If the clause is variable-free, our normal heuristic for pushing it
3717 * down to just the mentioned rels doesn't work, because there are none.
3718 * Apply it as a gating qual at the appropriate level (see comments for
3719 * get_join_domain_min_rels).
3720 */
3721 if (bms_is_empty(relids))
3722 {
3723 /* eval at join domain's safe level */
3724 relids = get_join_domain_min_rels(root, qualscope);
3725 /* mark as gating qual */
3726 pseudoconstant = true;
3727 /* tell createplan.c to check for gating quals */
3728 root->hasPseudoConstantQuals = true;
3729 }
3730
3731 /*
3732 * Build the RestrictInfo node itself.
3733 */
3735 (Expr *) clause,
3736 true, /* is_pushed_down */
3737 false, /* !has_clone */
3738 false, /* !is_clone */
3740 security_level,
3741 relids,
3742 NULL, /* incompatible_relids */
3743 NULL); /* outer_relids */
3744
3745 /*
3746 * If it's a join clause, add vars used in the clause to targetlists of
3747 * their relations, so that they will be emitted by the plan nodes that
3748 * scan those relations (else they won't be available at the join node!).
3749 *
3750 * Typically, we'd have already done this when the component expressions
3751 * were first seen by distribute_qual_to_rels; but it is possible that
3752 * some of the Vars could have missed having that done because they only
3753 * appeared in single-relation clauses originally. So do it here for
3754 * safety.
3755 *
3756 * See also rebuild_joinclause_attr_needed, which has to partially repeat
3757 * this work after removal of an outer join. (Since we will put this
3758 * clause into the joininfo lists, that function needn't do any extra work
3759 * to find it.)
3760 */
3761 if (bms_membership(relids) == BMS_MULTIPLE)
3762 {
3763 List *vars = pull_var_clause(clause,
3767
3769 list_free(vars);
3770 }
3771
3772 /*
3773 * Check mergejoinability. This will usually succeed, since the op came
3774 * from an EquivalenceClass; but we could have reduced the original clause
3775 * to a constant.
3776 */
3778
3779 /*
3780 * Note we don't do initialize_mergeclause_eclasses(); the caller can
3781 * handle that much more cheaply than we can. It's okay to call
3782 * distribute_restrictinfo_to_rels() before that happens.
3783 */
3784
3785 /*
3786 * Push the new clause into all the appropriate restrictinfo lists.
3787 */
3789
3790 return restrictinfo;
3791}
BMS_Membership bms_membership(const Bitmapset *a)
Definition bitmapset.c:765
@ BMS_MULTIPLE
Definition bitmapset.h:73
Node * eval_const_expressions(PlannerInfo *root, Node *node)
Definition clauses.c:2500
void distribute_restrictinfo_to_rels(PlannerInfo *root, RestrictInfo *restrictinfo)
Definition initsplan.c:3578
static Relids get_join_domain_min_rels(PlannerInfo *root, Relids domain_relids)
Definition initsplan.c:3876
static bool DatumGetBool(Datum X)
Definition postgres.h:100
Relids pull_varnos(PlannerInfo *root, Node *node)
Definition var.c:114

References add_vars_to_targetlist(), Assert, bms_is_empty, bms_is_subset(), bms_membership(), BMS_MULTIPLE, check_mergejoinable(), copyObject, DatumGetBool(), distribute_restrictinfo_to_rels(), eval_const_expressions(), fb(), get_join_domain_min_rels(), InvalidOid, IsA, list_free(), make_opclause(), make_restrictinfo(), pull_var_clause(), pull_varnos(), PVC_INCLUDE_PLACEHOLDERS, PVC_RECURSE_AGGREGATES, PVC_RECURSE_WINDOWFUNCS, JoinTreeItem::qualscope, and root.

Referenced by generate_base_implied_equalities_const(), and generate_base_implied_equalities_no_const().

◆ query_is_distinct_for()

bool query_is_distinct_for ( Query query,
List colnos,
List opids 
)
extern

Definition at line 1147 of file analyzejoins.c.

1148{
1149 ListCell *l;
1150 Oid opid;
1151
1153
1154 /*
1155 * DISTINCT (including DISTINCT ON) guarantees uniqueness if all the
1156 * columns in the DISTINCT clause appear in colnos and operator semantics
1157 * match. This is true even if there are SRFs in the DISTINCT columns or
1158 * elsewhere in the tlist.
1159 */
1160 if (query->distinctClause)
1161 {
1162 foreach(l, query->distinctClause)
1163 {
1166 query->targetList);
1167
1169 if (!OidIsValid(opid) ||
1171 break; /* exit early if no match */
1172 }
1173 if (l == NULL) /* had matches for all? */
1174 return true;
1175 }
1176
1177 /*
1178 * Otherwise, a set-returning function in the query's targetlist can
1179 * result in returning duplicate rows, despite any grouping that might
1180 * occur before tlist evaluation. (If all tlist SRFs are within GROUP BY
1181 * columns, it would be safe because they'd be expanded before grouping.
1182 * But it doesn't currently seem worth the effort to check for that.)
1183 */
1184 if (query->hasTargetSRFs)
1185 return false;
1186
1187 /*
1188 * Similarly, GROUP BY without GROUPING SETS guarantees uniqueness if all
1189 * the grouped columns appear in colnos and operator semantics match.
1190 */
1191 if (query->groupClause && !query->groupingSets)
1192 {
1193 foreach(l, query->groupClause)
1194 {
1197 query->targetList);
1198
1200 if (!OidIsValid(opid) ||
1202 break; /* exit early if no match */
1203 }
1204 if (l == NULL) /* had matches for all? */
1205 return true;
1206 }
1207 else if (query->groupingSets)
1208 {
1209 List *gsets;
1210
1211 /*
1212 * If we have grouping sets with expressions, we probably don't have
1213 * uniqueness and analysis would be hard. Punt.
1214 */
1215 if (query->groupClause)
1216 return false;
1217
1218 /*
1219 * If we have no groupClause (therefore no grouping expressions), we
1220 * might have one or many empty grouping sets. If there's just one,
1221 * or if the DISTINCT clause is used on the GROUP BY, then we're
1222 * returning only one row and are certainly unique. But otherwise, we
1223 * know we're certainly not unique.
1224 */
1225 if (query->groupDistinct)
1226 return true;
1227
1228 gsets = expand_grouping_sets(query->groupingSets, false, -1);
1229
1230 return (list_length(gsets) == 1);
1231 }
1232 else
1233 {
1234 /*
1235 * If we have no GROUP BY, but do have aggregates or HAVING, then the
1236 * result is at most one row so it's surely unique, for any operators.
1237 */
1238 if (query->hasAggs || query->havingQual)
1239 return true;
1240 }
1241
1242 /*
1243 * UNION, INTERSECT, EXCEPT guarantee uniqueness of the whole output row,
1244 * except with ALL.
1245 */
1246 if (query->setOperations)
1247 {
1249
1250 Assert(topop->op != SETOP_NONE);
1251
1252 if (!topop->all)
1253 {
1254 ListCell *lg;
1255
1256 /* We're good if all the nonjunk output columns are in colnos */
1257 lg = list_head(topop->groupClauses);
1258 foreach(l, query->targetList)
1259 {
1262
1263 if (tle->resjunk)
1264 continue; /* ignore resjunk columns */
1265
1266 /* non-resjunk columns should have grouping clauses */
1267 Assert(lg != NULL);
1269 lg = lnext(topop->groupClauses, lg);
1270
1272 if (!OidIsValid(opid) ||
1274 break; /* exit early if no match */
1275 }
1276 if (l == NULL) /* had matches for all? */
1277 return true;
1278 }
1279 }
1280
1281 /*
1282 * XXX Are there any other cases in which we can easily see the result
1283 * must be distinct?
1284 *
1285 * If you do add more smarts to this function, be sure to update
1286 * query_supports_distinctness() to match.
1287 */
1288
1289 return false;
1290}
static Oid distinct_col_search(int colno, List *colnos, List *opids)
bool equality_ops_are_compatible(Oid opno1, Oid opno2)
Definition lsyscache.c:773
List * expand_grouping_sets(List *groupingSets, bool groupDistinct, int limit)
Definition parse_agg.c:2019
@ SETOP_NONE
static ListCell * list_head(const List *l)
Definition pg_list.h:128
static ListCell * lnext(const List *l, const ListCell *c)
Definition pg_list.h:375
bool groupDistinct
Definition parsenodes.h:220
Node * setOperations
Definition parsenodes.h:239
List * groupClause
Definition parsenodes.h:219
Node * havingQual
Definition parsenodes.h:225
List * targetList
Definition parsenodes.h:201
List * groupingSets
Definition parsenodes.h:223
List * distinctClause
Definition parsenodes.h:229

References Assert, castNode, distinct_col_search(), Query::distinctClause, equality_ops_are_compatible(), expand_grouping_sets(), fb(), get_sortgroupclause_tle(), Query::groupClause, Query::groupDistinct, Query::groupingSets, Query::havingQual, lfirst, list_head(), list_length(), lnext(), OidIsValid, SETOP_NONE, Query::setOperations, and Query::targetList.

Referenced by rel_is_distinct_for().

◆ query_planner()

RelOptInfo * query_planner ( PlannerInfo root,
query_pathkeys_callback  qp_callback,
void qp_extra 
)
extern

Definition at line 54 of file planmain.c.

56{
57 Query *parse = root->parse;
60
61 /*
62 * Init planner lists to empty.
63 *
64 * NOTE: append_rel_list was set up by subquery_planner, so do not touch
65 * here.
66 */
67 root->join_rel_list = NIL;
68 root->join_rel_hash = NULL;
69 root->join_rel_level = NULL;
70 root->join_cur_level = 0;
71 root->canon_pathkeys = NIL;
72 root->left_join_clauses = NIL;
73 root->right_join_clauses = NIL;
74 root->full_join_clauses = NIL;
75 root->join_info_list = NIL;
76 root->placeholder_list = NIL;
77 root->placeholder_array = NULL;
78 root->placeholder_array_size = 0;
79 root->agg_clause_list = NIL;
80 root->group_expr_list = NIL;
81 root->tlist_vars = NIL;
82 root->fkey_list = NIL;
83 root->initial_rels = NIL;
84
85 /*
86 * Set up arrays for accessing base relations and AppendRelInfos.
87 */
89
90 /*
91 * In the trivial case where the jointree is a single RTE_RESULT relation,
92 * bypass all the rest of this function and just make a RelOptInfo and its
93 * one access path. This is worth optimizing because it applies for
94 * common cases like "SELECT expression" and "INSERT ... VALUES()".
95 */
96 Assert(parse->jointree->fromlist != NIL);
97 if (list_length(parse->jointree->fromlist) == 1)
98 {
99 Node *jtnode = (Node *) linitial(parse->jointree->fromlist);
100
101 if (IsA(jtnode, RangeTblRef))
102 {
103 int varno = ((RangeTblRef *) jtnode)->rtindex;
104 RangeTblEntry *rte = root->simple_rte_array[varno];
105
106 Assert(rte != NULL);
107 if (rte->rtekind == RTE_RESULT)
108 {
109 /* Make the RelOptInfo for it directly */
111
112 /*
113 * If query allows parallelism in general, check whether the
114 * quals are parallel-restricted. (We need not check
115 * final_rel->reltarget because it's empty at this point.
116 * Anything parallel-restricted in the query tlist will be
117 * dealt with later.) We should always do this in a subquery,
118 * since it might be useful to use the subquery in parallel
119 * paths in the parent level. At top level this is normally
120 * not worth the cycles, because a Result-only plan would
121 * never be interesting to parallelize. However, if
122 * debug_parallel_query is on, then we want to execute the
123 * Result in a parallel worker if possible, so we must check.
124 */
125 if (root->glob->parallelModeOK &&
126 (root->query_level > 1 ||
128 final_rel->consider_parallel =
129 is_parallel_safe(root, parse->jointree->quals);
130
131 /*
132 * The only path for it is a trivial Result path. We cheat a
133 * bit here by using a GroupResultPath, because that way we
134 * can just jam the quals into it without preprocessing them.
135 * (But, if you hold your head at the right angle, a FROM-less
136 * SELECT is a kind of degenerate-grouping case, so it's not
137 * that much of a cheat.)
138 */
141 final_rel->reltarget,
142 (List *) parse->jointree->quals));
143
144 /* Select cheapest path (pretty easy in this case...) */
146
147 /*
148 * We don't need to run generate_base_implied_equalities, but
149 * we do need to pretend that EC merging is complete.
150 */
151 root->ec_merging_done = true;
152
153 /*
154 * We still are required to call qp_callback, in case it's
155 * something like "SELECT 2+2 ORDER BY 1".
156 */
157 (*qp_callback) (root, qp_extra);
158
159 return final_rel;
160 }
161 }
162 }
163
164 /*
165 * Construct RelOptInfo nodes for all base relations used in the query.
166 * Appendrel member relations ("other rels") will be added later.
167 *
168 * Note: the reason we find the baserels by searching the jointree, rather
169 * than scanning the rangetable, is that the rangetable may contain RTEs
170 * for rels not actively part of the query, for example views. We don't
171 * want to make RelOptInfos for them.
172 */
173 add_base_rels_to_query(root, (Node *) parse->jointree);
174
175 /* Remove any redundant GROUP BY columns */
177
178 /*
179 * Examine the targetlist and join tree, adding entries to baserel
180 * targetlists for all referenced Vars, and generating PlaceHolderInfo
181 * entries for all referenced PlaceHolderVars. Restrict and join clauses
182 * are added to appropriate lists belonging to the mentioned relations. We
183 * also build EquivalenceClasses for provably equivalent expressions. The
184 * SpecialJoinInfo list is also built to hold information about join order
185 * restrictions. Finally, we form a target joinlist for make_one_rel() to
186 * work from.
187 */
188 build_base_rel_tlists(root, root->processed_tlist);
189
191
193
195
196 /*
197 * Reconsider any postponed outer-join quals now that we have built up
198 * equivalence classes. (This could result in further additions or
199 * mergings of classes.)
200 */
202
203 /*
204 * If we formed any equivalence classes, generate additional restriction
205 * clauses as appropriate. (Implied join clauses are formed on-the-fly
206 * later.)
207 */
209
210 /*
211 * We have completed merging equivalence sets, so it's now possible to
212 * generate pathkeys in canonical form; so compute query_pathkeys and
213 * other pathkeys fields in PlannerInfo.
214 */
215 (*qp_callback) (root, qp_extra);
216
217 /*
218 * Examine any "placeholder" expressions generated during subquery pullup.
219 * Make sure that the Vars they need are marked as needed at the relevant
220 * join level. This must be done before join removal because it might
221 * cause Vars or placeholders to be needed above a join when they weren't
222 * so marked before.
223 */
225
226 /*
227 * Remove any useless outer joins. Ideally this would be done during
228 * jointree preprocessing, but the necessary information isn't available
229 * until we've built baserel data structures and classified qual clauses.
230 */
232
233 /*
234 * Also, reduce any semijoins with unique inner rels to plain inner joins.
235 * Likewise, this can't be done until now for lack of needed info.
236 */
238
239 /*
240 * Remove self joins on a unique column.
241 */
243
244 /*
245 * Now distribute "placeholders" to base rels as needed. This has to be
246 * done after join removal because removal could change whether a
247 * placeholder is evaluable at a base rel.
248 */
250
251 /*
252 * Construct the lateral reference sets now that we have finalized
253 * PlaceHolderVar eval levels.
254 */
256
257 /*
258 * Match foreign keys to equivalence classes and join quals. This must be
259 * done after finalizing equivalence classes, and it's useful to wait till
260 * after join removal so that we can skip processing foreign keys
261 * involving removed relations.
262 */
264
265 /*
266 * Look for join OR clauses that we can extract single-relation
267 * restriction OR clauses from.
268 */
270
271 /*
272 * Check if eager aggregation is applicable, and if so, set up
273 * root->agg_clause_list and root->group_expr_list.
274 */
276
277 /*
278 * Now expand appendrels by adding "otherrels" for their children. We
279 * delay this to the end so that we have as much information as possible
280 * available for each baserel, including all restriction clauses. That
281 * let us prune away partitions that don't satisfy a restriction clause.
282 * Also note that some information such as lateral_relids is propagated
283 * from baserels to otherrels here, so we must have computed it already.
284 */
286
287 /*
288 * Distribute any UPDATE/DELETE/MERGE row identity variables to the target
289 * relations. This can't be done till we've finished expansion of
290 * appendrels.
291 */
293
294 /*
295 * Ready to do the primary planning.
296 */
298
299 /* Check that we got at least one usable path */
300 if (!final_rel || !final_rel->cheapest_total_path ||
301 final_rel->cheapest_total_path->param_info != NULL)
302 elog(ERROR, "failed to construct the join relation");
303
304 return final_rel;
305}
RelOptInfo * make_one_rel(PlannerInfo *root, List *joinlist)
Definition allpaths.c:179
List * remove_useless_joins(PlannerInfo *root, List *joinlist)
List * remove_useless_self_joins(PlannerInfo *root, List *joinlist)
void reduce_unique_semijoins(PlannerInfo *root)
void distribute_row_identity_vars(PlannerInfo *root)
bool is_parallel_safe(PlannerInfo *root, Node *node)
Definition clauses.c:766
void generate_base_implied_equalities(PlannerInfo *root)
void reconsider_outer_join_clauses(PlannerInfo *root)
void match_foreign_keys_to_quals(PlannerInfo *root)
Definition initsplan.c:3982
void find_lateral_references(PlannerInfo *root)
Definition initsplan.c:1045
void remove_useless_groupby_columns(PlannerInfo *root)
Definition initsplan.c:420
void build_base_rel_tlists(PlannerInfo *root, List *final_tlist)
Definition initsplan.c:243
List * deconstruct_jointree(PlannerInfo *root)
Definition initsplan.c:1471
void setup_eager_aggregation(PlannerInfo *root)
Definition initsplan.c:643
void add_other_rels_to_query(PlannerInfo *root)
Definition initsplan.c:204
void create_lateral_join_info(PlannerInfo *root)
Definition initsplan.c:1232
@ DEBUG_PARALLEL_OFF
Definition optimizer.h:96
void extract_restriction_or_clauses(PlannerInfo *root)
Definition orclauses.c:75
@ RTE_RESULT
void set_cheapest(RelOptInfo *parent_rel)
Definition pathnode.c:268
GroupResultPath * create_group_result_path(PlannerInfo *root, RelOptInfo *rel, PathTarget *target, List *havingqual)
Definition pathnode.c:1664
void add_placeholders_to_base_rels(PlannerInfo *root)
void fix_placeholder_input_needed_levels(PlannerInfo *root)
void find_placeholders_in_jointree(PlannerInfo *root)
int debug_parallel_query
Definition planner.c:69
void setup_simple_rel_arrays(PlannerInfo *root)
Definition relnode.c:114

References add_base_rels_to_query(), add_other_rels_to_query(), add_path(), add_placeholders_to_base_rels(), Assert, build_base_rel_tlists(), build_simple_rel(), create_group_result_path(), create_lateral_join_info(), DEBUG_PARALLEL_OFF, debug_parallel_query, deconstruct_jointree(), distribute_row_identity_vars(), elog, ERROR, extract_restriction_or_clauses(), fb(), find_lateral_references(), find_placeholders_in_jointree(), fix_placeholder_input_needed_levels(), generate_base_implied_equalities(), is_parallel_safe(), IsA, linitial, list_length(), make_one_rel(), match_foreign_keys_to_quals(), NIL, parse(), reconsider_outer_join_clauses(), reduce_unique_semijoins(), remove_useless_groupby_columns(), remove_useless_joins(), remove_useless_self_joins(), root, RTE_RESULT, set_cheapest(), setup_eager_aggregation(), and setup_simple_rel_arrays().

Referenced by build_minmax_path(), and grouping_planner().

◆ query_supports_distinctness()

bool query_supports_distinctness ( Query query)
extern

Definition at line 1109 of file analyzejoins.c.

1110{
1111 /* SRFs break distinctness except with DISTINCT, see below */
1112 if (query->hasTargetSRFs && query->distinctClause == NIL)
1113 return false;
1114
1115 /* check for features we can prove distinctness with */
1116 if (query->distinctClause != NIL ||
1117 query->groupClause != NIL ||
1118 query->groupingSets != NIL ||
1119 query->hasAggs ||
1120 query->havingQual ||
1121 query->setOperations)
1122 return true;
1123
1124 return false;
1125}

References Query::distinctClause, Query::groupClause, Query::groupingSets, Query::havingQual, NIL, and Query::setOperations.

Referenced by rel_supports_distinctness().

◆ rebuild_joinclause_attr_needed()

void rebuild_joinclause_attr_needed ( PlannerInfo root)
extern

Definition at line 3910 of file initsplan.c.

3911{
3912 /*
3913 * We must examine all join clauses, but there's no value in processing
3914 * any join clause more than once. So it's slightly annoying that we have
3915 * to find them via the per-base-relation joininfo lists. Avoid duplicate
3916 * processing by tracking the rinfo_serial numbers of join clauses we've
3917 * already seen. (This doesn't work for is_clone clauses, so we must
3918 * waste effort on them.)
3919 */
3921 Index rti;
3922
3923 /* Scan all baserels for join clauses */
3924 for (rti = 1; rti < root->simple_rel_array_size; rti++)
3925 {
3926 RelOptInfo *brel = root->simple_rel_array[rti];
3927 ListCell *lc;
3928
3929 if (brel == NULL)
3930 continue;
3931 if (brel->reloptkind != RELOPT_BASEREL)
3932 continue;
3933
3934 foreach(lc, brel->joininfo)
3935 {
3936 RestrictInfo *rinfo = (RestrictInfo *) lfirst(lc);
3937 Relids relids = rinfo->required_relids;
3938
3939 if (!rinfo->is_clone) /* else serial number is not unique */
3940 {
3942 continue; /* saw it already */
3944 rinfo->rinfo_serial);
3945 }
3946
3947 if (bms_membership(relids) == BMS_MULTIPLE)
3948 {
3949 List *vars = pull_var_clause((Node *) rinfo->clause,
3954
3955 if (rinfo->is_clone)
3956 where_needed = bms_intersect(relids, root->all_baserels);
3957 else
3958 where_needed = relids;
3960 list_free(vars);
3961 }
3962 }
3963 }
3964}
void add_vars_to_attr_needed(PlannerInfo *root, List *vars, Relids where_needed)
Definition initsplan.c:361
Relids required_relids
Definition pathnodes.h:2932

References add_vars_to_attr_needed(), bms_add_member(), bms_intersect(), bms_is_member(), bms_membership(), BMS_MULTIPLE, RestrictInfo::clause, fb(), RestrictInfo::is_clone, lfirst, list_free(), pull_var_clause(), PVC_INCLUDE_PLACEHOLDERS, PVC_RECURSE_AGGREGATES, PVC_RECURSE_WINDOWFUNCS, RELOPT_BASEREL, RestrictInfo::required_relids, RestrictInfo::rinfo_serial, and root.

Referenced by remove_leftjoinrel_from_query(), and remove_self_join_rel().

◆ rebuild_lateral_attr_needed()

void rebuild_lateral_attr_needed ( PlannerInfo root)
extern

Definition at line 1194 of file initsplan.c.

1195{
1196 Index rti;
1197
1198 /* We need do nothing if the query contains no LATERAL RTEs */
1199 if (!root->hasLateralRTEs)
1200 return;
1201
1202 /* Examine the same baserels that find_lateral_references did */
1203 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1204 {
1205 RelOptInfo *brel = root->simple_rel_array[rti];
1207
1208 if (brel == NULL)
1209 continue;
1210 if (brel->reloptkind != RELOPT_BASEREL)
1211 continue;
1212
1213 /*
1214 * We don't need to repeat all of extract_lateral_references, since it
1215 * kindly saved the extracted Vars/PHVs in lateral_vars.
1216 */
1217 if (brel->lateral_vars == NIL)
1218 continue;
1219
1221
1223 }
1224}

References add_vars_to_attr_needed(), bms_make_singleton(), fb(), NIL, RELOPT_BASEREL, and root.

Referenced by remove_leftjoinrel_from_query(), and remove_self_join_rel().

◆ record_plan_function_dependency()

void record_plan_function_dependency ( PlannerInfo root,
Oid  funcid 
)
extern

Definition at line 3632 of file setrefs.c.

3633{
3634 /*
3635 * For performance reasons, we don't bother to track built-in functions;
3636 * we just assume they'll never change (or at least not in ways that'd
3637 * invalidate plans using them). For this purpose we can consider a
3638 * built-in function to be one with OID less than FirstUnpinnedObjectId.
3639 * Note that the OID generator guarantees never to generate such an OID
3640 * after startup, even at OID wraparound.
3641 */
3642 if (funcid >= (Oid) FirstUnpinnedObjectId)
3643 {
3645
3646 /*
3647 * It would work to use any syscache on pg_proc, but the easiest is
3648 * PROCOID since we already have the function's OID at hand. Note
3649 * that plancache.c knows we use PROCOID.
3650 */
3651 inval_item->cacheId = PROCOID;
3653 ObjectIdGetDatum(funcid));
3654
3655 root->glob->invalItems = lappend(root->glob->invalItems, inval_item);
3656 }
3657}
static Datum ObjectIdGetDatum(Oid X)
Definition postgres.h:252
#define GetSysCacheHashValue1(cacheId, key1)
Definition syscache.h:118
#define FirstUnpinnedObjectId
Definition transam.h:196

References fb(), FirstUnpinnedObjectId, GetSysCacheHashValue1, lappend(), makeNode, ObjectIdGetDatum(), and root.

Referenced by fix_expr_common(), inline_function(), and inline_function_in_from().

◆ record_plan_type_dependency()

void record_plan_type_dependency ( PlannerInfo root,
Oid  typid 
)
extern

Definition at line 3672 of file setrefs.c.

3673{
3674 /*
3675 * As in record_plan_function_dependency, ignore the possibility that
3676 * someone would change a built-in domain.
3677 */
3678 if (typid >= (Oid) FirstUnpinnedObjectId)
3679 {
3681
3682 /*
3683 * It would work to use any syscache on pg_type, but the easiest is
3684 * TYPEOID since we already have the type's OID at hand. Note that
3685 * plancache.c knows we use TYPEOID.
3686 */
3687 inval_item->cacheId = TYPEOID;
3689 ObjectIdGetDatum(typid));
3690
3691 root->glob->invalItems = lappend(root->glob->invalItems, inval_item);
3692 }
3693}

References fb(), FirstUnpinnedObjectId, GetSysCacheHashValue1, lappend(), makeNode, ObjectIdGetDatum(), and root.

Referenced by eval_const_expressions_mutator().

◆ reduce_unique_semijoins()

void reduce_unique_semijoins ( PlannerInfo root)
extern

Definition at line 875 of file analyzejoins.c.

876{
877 ListCell *lc;
878
879 /*
880 * Scan the join_info_list to find semijoins.
881 */
882 foreach(lc, root->join_info_list)
883 {
885 int innerrelid;
886 RelOptInfo *innerrel;
888 List *restrictlist;
889
890 /*
891 * Must be a semijoin to a single baserel, else we aren't going to be
892 * able to do anything with it.
893 */
894 if (sjinfo->jointype != JOIN_SEMI)
895 continue;
896
898 continue;
899
900 innerrel = find_base_rel(root, innerrelid);
901
902 /*
903 * Before we trouble to run generate_join_implied_equalities, make a
904 * quick check to eliminate cases in which we will surely be unable to
905 * prove uniqueness of the innerrel.
906 */
907 if (!rel_supports_distinctness(root, innerrel))
908 continue;
909
910 /* Compute the relid set for the join we are considering */
912 Assert(sjinfo->ojrelid == 0); /* SEMI joins don't have RT indexes */
913
914 /*
915 * Since we're only considering a single-rel RHS, any join clauses it
916 * has must be clauses linking it to the semijoin's min_lefthand. We
917 * can also consider EC-derived join clauses.
918 */
919 restrictlist =
922 sjinfo->min_lefthand,
923 innerrel,
924 NULL),
925 innerrel->joininfo);
926
927 /* Test whether the innerrel is unique for those clauses. */
929 joinrelids, sjinfo->min_lefthand, innerrel,
930 JOIN_SEMI, restrictlist, true))
931 continue;
932
933 /* OK, remove the SpecialJoinInfo from the list. */
934 root->join_info_list = foreach_delete_current(root->join_info_list, lc);
935 }
936}
bool innerrel_is_unique(PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, bool force_cache)
List * generate_join_implied_equalities(PlannerInfo *root, Relids join_relids, Relids outer_relids, RelOptInfo *inner_rel, SpecialJoinInfo *sjinfo)
List * list_concat(List *list1, const List *list2)
Definition list.c:561
@ JOIN_SEMI
Definition nodes.h:317
#define foreach_delete_current(lst, var_or_cell)
Definition pg_list.h:423
List * joininfo
Definition pathnodes.h:1148
Relids min_righthand
Definition pathnodes.h:3227
JoinType jointype
Definition pathnodes.h:3230
Relids min_lefthand
Definition pathnodes.h:3226

References Assert, bms_get_singleton_member(), bms_union(), fb(), find_base_rel(), foreach_delete_current, generate_join_implied_equalities(), innerrel_is_unique(), JOIN_SEMI, RelOptInfo::joininfo, SpecialJoinInfo::jointype, lfirst, list_concat(), SpecialJoinInfo::min_lefthand, SpecialJoinInfo::min_righthand, SpecialJoinInfo::ojrelid, rel_supports_distinctness(), and root.

Referenced by query_planner().

◆ remove_useless_groupby_columns()

void remove_useless_groupby_columns ( PlannerInfo root)
extern

Definition at line 420 of file initsplan.c.

421{
422 Query *parse = root->parse;
425 bool tryremove = false;
426 ListCell *lc;
427 int relid;
428
429 /* No chance to do anything if there are less than two GROUP BY items */
430 if (list_length(root->processed_groupClause) < 2)
431 return;
432
433 /* Don't fiddle with the GROUP BY clause if the query has grouping sets */
434 if (parse->groupingSets)
435 return;
436
437 /*
438 * Scan the GROUP BY clause to find GROUP BY items that are simple Vars.
439 * Fill groupbyattnos[k] with a bitmapset of the column attnos of RTE k
440 * that are GROUP BY items.
441 */
443 foreach(lc, root->processed_groupClause)
444 {
447 Var *var = (Var *) tle->expr;
448
449 /*
450 * Ignore non-Vars and Vars from other query levels.
451 *
452 * XXX in principle, stable expressions containing Vars could also be
453 * removed, if all the Vars are functionally dependent on other GROUP
454 * BY items. But it's not clear that such cases occur often enough to
455 * be worth troubling over.
456 */
457 if (!IsA(var, Var) ||
458 var->varlevelsup > 0)
459 continue;
460
461 /* OK, remember we have this Var */
462 relid = var->varno;
463 Assert(relid <= list_length(parse->rtable));
464
465 /*
466 * If this isn't the first column for this relation then we now have
467 * multiple columns. That means there might be some that can be
468 * removed.
469 */
472 var->varattno - FirstLowInvalidHeapAttributeNumber);
473 }
474
475 /*
476 * No Vars or didn't find multiple Vars for any relation in the GROUP BY?
477 * If so, nothing can be removed, so don't waste more effort trying.
478 */
479 if (!tryremove)
480 return;
481
482 /*
483 * Consider each relation and see if it is possible to remove some of its
484 * Vars from GROUP BY. For simplicity and speed, we do the actual removal
485 * in a separate pass. Here, we just fill surplusvars[k] with a bitmapset
486 * of the column attnos of RTE k that are removable GROUP BY items.
487 */
488 surplusvars = NULL; /* don't allocate array unless required */
489 relid = 0;
490 foreach(lc, parse->rtable)
491 {
493 RelOptInfo *rel;
497
498 relid++;
499
500 /* Only plain relations could have primary-key constraints */
501 if (rte->rtekind != RTE_RELATION)
502 continue;
503
504 /*
505 * We must skip inheritance parent tables as some of the child rels
506 * may cause duplicate rows. This cannot happen with partitioned
507 * tables, however.
508 */
509 if (rte->inh && rte->relkind != RELKIND_PARTITIONED_TABLE)
510 continue;
511
512 /* Nothing to do unless this rel has multiple Vars in GROUP BY */
513 relattnos = groupbyattnos[relid];
515 continue;
516
517 rel = root->simple_rel_array[relid];
518
519 /*
520 * Now check each index for this relation to see if there are any with
521 * columns which are a proper subset of the grouping columns for this
522 * relation.
523 */
525 {
527 bool nulls_check_ok;
528
529 /*
530 * Skip any non-unique and deferrable indexes. Predicate indexes
531 * have not been checked yet, so we must skip those too as the
532 * predOK check that's done later might fail.
533 */
534 if (!index->unique || !index->immediate || index->indpred != NIL)
535 continue;
536
537 /* For simplicity, we currently don't support expression indexes */
538 if (index->indexprs != NIL)
539 continue;
540
542 nulls_check_ok = true;
543 for (int i = 0; i < index->nkeycolumns; i++)
544 {
545 /*
546 * We must insist that the index columns are all defined NOT
547 * NULL otherwise duplicate NULLs could exist. However, we
548 * can relax this check when the index is defined with NULLS
549 * NOT DISTINCT as there can only be 1 NULL row, therefore
550 * functional dependency on the unique columns is maintained,
551 * despite the NULL.
552 */
553 if (!index->nullsnotdistinct &&
554 !bms_is_member(index->indexkeys[i],
555 rel->notnullattnums))
556 {
557 nulls_check_ok = false;
558 break;
559 }
560
561 ind_attnos =
563 index->indexkeys[i] -
565 }
566
567 if (!nulls_check_ok)
568 continue;
569
570 /*
571 * Skip any indexes where the indexed columns aren't a proper
572 * subset of the GROUP BY.
573 */
575 continue;
576
577 /*
578 * Record the attribute numbers from the index with the fewest
579 * columns. This allows the largest number of columns to be
580 * removed from the GROUP BY clause. In the future, we may wish
581 * to consider using the narrowest set of columns and looking at
582 * pg_statistic.stawidth as it might be better to use an index
583 * with, say two INT4s, rather than, say, one long varlena column.
584 */
585 if (index->nkeycolumns < best_nkeycolumns)
586 {
588 best_nkeycolumns = index->nkeycolumns;
589 }
590 }
591
592 /* Did we find a suitable index? */
594 {
595 /*
596 * To easily remember whether we've found anything to do, we don't
597 * allocate the surplusvars[] array until we find something.
598 */
599 if (surplusvars == NULL)
601
602 /* Remember the attnos of the removable columns */
604 }
605 }
606
607 /*
608 * If we found any surplus Vars, build a new GROUP BY clause without them.
609 * (Note: this may leave some TLEs with unreferenced ressortgroupref
610 * markings, but that's harmless.)
611 */
612 if (surplusvars != NULL)
613 {
615
616 foreach(lc, root->processed_groupClause)
617 {
620 Var *var = (Var *) tle->expr;
621
622 /*
623 * New list must include non-Vars, outer Vars, and anything not
624 * marked as surplus.
625 */
626 if (!IsA(var, Var) ||
627 var->varlevelsup > 0 ||
629 surplusvars[var->varno]))
631 }
632
633 root->processed_groupClause = new_groupby;
634 }
635}
Bitmapset * bms_difference(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:346
BMS_Comparison bms_subset_compare(const Bitmapset *a, const Bitmapset *b)
Definition bitmapset.c:445
@ BMS_SUBSET1
Definition bitmapset.h:63
#define PG_INT32_MAX
Definition c.h:673
int32_t int32
Definition c.h:620
#define palloc0_array(type, count)
Definition fe_memutils.h:77
int i
Definition isn.c:77
#define lfirst_node(type, lc)
Definition pg_list.h:176
#define foreach_node(type, var, lst)
Definition pg_list.h:528
Bitmapset * notnullattnums
Definition pathnodes.h:1083
List * indexlist
Definition pathnodes.h:1091
Definition type.h:96
#define FirstLowInvalidHeapAttributeNumber
Definition sysattr.h:27

References Assert, bms_add_member(), bms_difference(), bms_is_empty, bms_is_member(), bms_membership(), BMS_MULTIPLE, BMS_SUBSET1, bms_subset_compare(), fb(), FirstLowInvalidHeapAttributeNumber, foreach_node, get_sortgroupclause_tle(), i, RelOptInfo::indexlist, IsA, lappend(), lfirst_node, list_length(), NIL, RelOptInfo::notnullattnums, palloc0_array, parse(), PG_INT32_MAX, root, RTE_RELATION, Var::varattno, Var::varlevelsup, and Var::varno.

Referenced by query_planner().

◆ remove_useless_joins()

List * remove_useless_joins ( PlannerInfo root,
List joinlist 
)
extern

Definition at line 93 of file analyzejoins.c.

94{
95 ListCell *lc;
96
97 /*
98 * We are only interested in relations that are left-joined to, so we can
99 * scan the join_info_list to find them easily.
100 */
101restart:
102 foreach(lc, root->join_info_list)
103 {
105 int innerrelid;
106 int nremoved;
107
108 /* Skip if not removable */
109 if (!join_is_removable(root, sjinfo))
110 continue;
111
112 /*
113 * Currently, join_is_removable can only succeed when the sjinfo's
114 * righthand is a single baserel. Remove that rel from the query and
115 * joinlist.
116 */
118
120
121 /* We verify that exactly one reference gets removed from joinlist */
122 nremoved = 0;
124 if (nremoved != 1)
125 elog(ERROR, "failed to find relation %d in joinlist", innerrelid);
126
127 /*
128 * We can delete this SpecialJoinInfo from the list too, since it's no
129 * longer of interest. (Since we'll restart the foreach loop
130 * immediately, we don't bother with foreach_delete_current.)
131 */
132 root->join_info_list = list_delete_cell(root->join_info_list, lc);
133
134 /*
135 * Restart the scan. This is necessary to ensure we find all
136 * removable joins independently of ordering of the join_info_list
137 * (note that removal of attr_needed bits may make a join appear
138 * removable that did not before).
139 */
140 goto restart;
141 }
142
143 return joinlist;
144}
static List * remove_rel_from_joinlist(List *joinlist, int relid, int *nremoved)
static void remove_leftjoinrel_from_query(PlannerInfo *root, int relid, SpecialJoinInfo *sjinfo)
static bool join_is_removable(PlannerInfo *root, SpecialJoinInfo *sjinfo)
int bms_singleton_member(const Bitmapset *a)
Definition bitmapset.c:665
List * list_delete_cell(List *list, ListCell *cell)
Definition list.c:841

References bms_singleton_member(), elog, ERROR, fb(), join_is_removable(), lfirst, list_delete_cell(), SpecialJoinInfo::min_righthand, remove_leftjoinrel_from_query(), remove_rel_from_joinlist(), and root.

Referenced by query_planner().

◆ remove_useless_self_joins()

List * remove_useless_self_joins ( PlannerInfo root,
List joinlist 
)
extern

Definition at line 2515 of file analyzejoins.c.

2516{
2518 int relid = -1;
2519
2522 return joinlist;
2523
2524 /*
2525 * Merge pairs of relations participated in self-join. Remove unnecessary
2526 * range table entries.
2527 */
2529
2530 if (unlikely(toRemove != NULL))
2531 {
2532 /* At the end, remove orphaned relation links */
2533 while ((relid = bms_next_member(toRemove, relid)) >= 0)
2534 {
2535 int nremoved = 0;
2536
2538 if (nremoved != 1)
2539 elog(ERROR, "failed to find relation %d in joinlist", relid);
2540 }
2541 }
2542
2543 return joinlist;
2544}
static Relids remove_self_joins_recurse(PlannerInfo *root, List *joinlist, Relids toRemove)
bool enable_self_join_elimination
#define unlikely(x)
Definition c.h:438

References bms_next_member(), elog, enable_self_join_elimination, ERROR, fb(), IsA, linitial, list_length(), NIL, remove_rel_from_joinlist(), remove_self_joins_recurse(), root, and unlikely.

Referenced by query_planner().

◆ restriction_is_always_false()

bool restriction_is_always_false ( PlannerInfo root,
RestrictInfo restrictinfo 
)
extern

Definition at line 3507 of file initsplan.c.

3509{
3510 /*
3511 * For a clone clause, we don't have a reliable way to determine if the
3512 * input expression of a NullTest is non-nullable: nullingrel bits in
3513 * clone clauses may not reflect reality, so we dare not draw conclusions
3514 * from clones about whether Vars are guaranteed not-null.
3515 */
3516 if (restrictinfo->has_clone || restrictinfo->is_clone)
3517 return false;
3518
3519 /* Check for NullTest qual */
3520 if (IsA(restrictinfo->clause, NullTest))
3521 {
3522 NullTest *nulltest = (NullTest *) restrictinfo->clause;
3523
3524 /* is this NullTest an IS_NULL qual? */
3525 if (nulltest->nulltesttype != IS_NULL)
3526 return false;
3527
3528 /*
3529 * Empty rows can appear NULL in some contexts and NOT NULL in others,
3530 * so avoid this optimization for row expressions.
3531 */
3532 if (nulltest->argisrow)
3533 return false;
3534
3536 }
3537
3538 /* If it's an OR, check its sub-clauses */
3540 {
3541 ListCell *lc;
3542
3543 Assert(is_orclause(restrictinfo->orclause));
3544
3545 /*
3546 * Currently, when processing OR expressions, we only return true when
3547 * all of the OR branches are always false. This could perhaps be
3548 * expanded to remove OR branches that are provably false. This may
3549 * be a useful thing to do as it could result in the OR being left
3550 * with a single arg. That's useful as it would allow the OR
3551 * condition to be replaced with its single argument which may allow
3552 * use of an index for faster filtering on the remaining condition.
3553 */
3554 foreach(lc, ((BoolExpr *) restrictinfo->orclause)->args)
3555 {
3556 Node *orarg = (Node *) lfirst(lc);
3557
3558 if (!IsA(orarg, RestrictInfo) ||
3560 return false;
3561 }
3562 return true;
3563 }
3564
3565 return false;
3566}
bool expr_is_nonnullable(PlannerInfo *root, Expr *expr, NotNullSource source)
Definition clauses.c:4772
bool restriction_is_always_false(PlannerInfo *root, RestrictInfo *restrictinfo)
Definition initsplan.c:3507
static bool is_orclause(const void *clause)
Definition nodeFuncs.h:116
@ NOTNULL_SOURCE_RELOPT
Definition optimizer.h:136
@ IS_NULL
Definition primnodes.h:1980
bool restriction_is_or_clause(RestrictInfo *restrictinfo)

References Assert, expr_is_nonnullable(), fb(), IS_NULL, is_orclause(), IsA, lfirst, NOTNULL_SOURCE_RELOPT, restriction_is_always_false(), restriction_is_or_clause(), and root.

Referenced by add_base_clause_to_rel(), add_join_clause_to_rels(), and restriction_is_always_false().

◆ restriction_is_always_true()

bool restriction_is_always_true ( PlannerInfo root,
RestrictInfo restrictinfo 
)
extern

Definition at line 3442 of file initsplan.c.

3444{
3445 /*
3446 * For a clone clause, we don't have a reliable way to determine if the
3447 * input expression of a NullTest is non-nullable: nullingrel bits in
3448 * clone clauses may not reflect reality, so we dare not draw conclusions
3449 * from clones about whether Vars are guaranteed not-null.
3450 */
3451 if (restrictinfo->has_clone || restrictinfo->is_clone)
3452 return false;
3453
3454 /* Check for NullTest qual */
3455 if (IsA(restrictinfo->clause, NullTest))
3456 {
3457 NullTest *nulltest = (NullTest *) restrictinfo->clause;
3458
3459 /* is this NullTest an IS_NOT_NULL qual? */
3460 if (nulltest->nulltesttype != IS_NOT_NULL)
3461 return false;
3462
3463 /*
3464 * Empty rows can appear NULL in some contexts and NOT NULL in others,
3465 * so avoid this optimization for row expressions.
3466 */
3467 if (nulltest->argisrow)
3468 return false;
3469
3471 }
3472
3473 /* If it's an OR, check its sub-clauses */
3475 {
3476 ListCell *lc;
3477
3478 Assert(is_orclause(restrictinfo->orclause));
3479
3480 /*
3481 * if any of the given OR branches is provably always true then the
3482 * entire condition is true.
3483 */
3484 foreach(lc, ((BoolExpr *) restrictinfo->orclause)->args)
3485 {
3486 Node *orarg = (Node *) lfirst(lc);
3487
3488 if (!IsA(orarg, RestrictInfo))
3489 continue;
3490
3492 return true;
3493 }
3494 }
3495
3496 return false;
3497}
bool restriction_is_always_true(PlannerInfo *root, RestrictInfo *restrictinfo)
Definition initsplan.c:3442
@ IS_NOT_NULL
Definition primnodes.h:1980

References Assert, expr_is_nonnullable(), fb(), IS_NOT_NULL, is_orclause(), IsA, lfirst, NOTNULL_SOURCE_RELOPT, restriction_is_always_true(), restriction_is_or_clause(), and root.

Referenced by add_base_clause_to_rel(), add_join_clause_to_rels(), and restriction_is_always_true().

◆ set_plan_references()

Plan * set_plan_references ( PlannerInfo root,
Plan plan 
)
extern

Definition at line 291 of file setrefs.c.

292{
293 Plan *result;
294 PlannerGlobal *glob = root->glob;
295 int rtoffset = list_length(glob->finalrtable);
296 ListCell *lc;
297
298 /*
299 * Add all the query's RTEs to the flattened rangetable. The live ones
300 * will have their rangetable indexes increased by rtoffset. (Additional
301 * RTEs, not referenced by the Plan tree, might get added after those.)
302 */
304
305 /*
306 * Adjust RT indexes of PlanRowMarks and add to final rowmarks list
307 */
308 foreach(lc, root->rowMarks)
309 {
312
313 /* sanity check on existing row marks */
314 Assert(root->simple_rel_array[rc->rti] != NULL &&
315 root->simple_rte_array[rc->rti] != NULL);
316
317 /* flat copy is enough since all fields are scalars */
319 memcpy(newrc, rc, sizeof(PlanRowMark));
320
321 /* adjust indexes ... but *not* the rowmarkId */
322 newrc->rti += rtoffset;
323 newrc->prti += rtoffset;
324
326 }
327
328 /*
329 * Adjust RT indexes of AppendRelInfos and add to final appendrels list.
330 * We assume the AppendRelInfos were built during planning and don't need
331 * to be copied.
332 */
333 foreach(lc, root->append_rel_list)
334 {
336
337 /* adjust RT indexes */
338 appinfo->parent_relid += rtoffset;
339 appinfo->child_relid += rtoffset;
340
341 /*
342 * Rather than adjust the translated_vars entries, just drop 'em.
343 * Neither the executor nor EXPLAIN currently need that data.
344 */
345 appinfo->translated_vars = NIL;
346
348 }
349
350 /* If needed, create workspace for processing AlternativeSubPlans */
351 if (root->hasAlternativeSubPlans)
352 {
353 root->isAltSubplan = (bool *)
354 palloc0(list_length(glob->subplans) * sizeof(bool));
355 root->isUsedSubplan = (bool *)
356 palloc0(list_length(glob->subplans) * sizeof(bool));
357 }
358
359 /* Now fix the Plan tree */
360 result = set_plan_refs(root, plan, rtoffset);
361
362 /*
363 * If we have AlternativeSubPlans, it is likely that we now have some
364 * unreferenced subplans in glob->subplans. To avoid expending cycles on
365 * those subplans later, get rid of them by setting those list entries to
366 * NULL. (Note: we can't do this immediately upon processing an
367 * AlternativeSubPlan, because there may be multiple copies of the
368 * AlternativeSubPlan, and they can get resolved differently.)
369 */
370 if (root->hasAlternativeSubPlans)
371 {
372 foreach(lc, glob->subplans)
373 {
375
376 /*
377 * If it was used by some AlternativeSubPlan in this query level,
378 * but wasn't selected as best by any AlternativeSubPlan, then we
379 * don't need it. Do not touch subplans that aren't parts of
380 * AlternativeSubPlans.
381 */
382 if (root->isAltSubplan[ndx] && !root->isUsedSubplan[ndx])
383 lfirst(lc) = NULL;
384 }
385 }
386
387 return result;
388}
memcpy(sums, checksumBaseOffsets, sizeof(checksumBaseOffsets))
#define palloc_object(type)
Definition fe_memutils.h:74
void * palloc0(Size size)
Definition mcxt.c:1417
#define foreach_current_index(var_or_cell)
Definition pg_list.h:435
static void add_rtes_to_flat_rtable(PlannerInfo *root, bool recursing)
Definition setrefs.c:399
static Plan * set_plan_refs(PlannerInfo *root, Plan *plan, int rtoffset)
Definition setrefs.c:642
List * subplans
Definition pathnodes.h:178
List * appendRelations
Definition pathnodes.h:221
List * finalrowmarks
Definition pathnodes.h:215
List * finalrtable
Definition pathnodes.h:193

References add_rtes_to_flat_rtable(), PlannerGlobal::appendRelations, Assert, fb(), PlannerGlobal::finalrowmarks, PlannerGlobal::finalrtable, foreach_current_index, lappend(), lfirst, lfirst_node, list_length(), memcpy(), NIL, palloc0(), palloc_object, plan, result, root, PlanRowMark::rti, set_plan_refs(), and PlannerGlobal::subplans.

Referenced by set_subqueryscan_references(), and standard_planner().

◆ setup_eager_aggregation()

void setup_eager_aggregation ( PlannerInfo root)
extern

Definition at line 643 of file initsplan.c.

644{
645 /*
646 * Don't apply eager aggregation if disabled by user.
647 */
649 return;
650
651 /*
652 * Don't apply eager aggregation if there are no available GROUP BY
653 * clauses.
654 */
655 if (!root->processed_groupClause)
656 return;
657
658 /*
659 * For now we don't try to support grouping sets.
660 */
661 if (root->parse->groupingSets)
662 return;
663
664 /*
665 * For now we don't try to support DISTINCT or ORDER BY aggregates.
666 */
667 if (root->numOrderedAggs > 0)
668 return;
669
670 /*
671 * If there are any aggregates that do not support partial mode, or any
672 * partial aggregates that are non-serializable, do not apply eager
673 * aggregation.
674 */
675 if (root->hasNonPartialAggs || root->hasNonSerialAggs)
676 return;
677
678 /*
679 * We don't try to apply eager aggregation if there are set-returning
680 * functions in targetlist.
681 */
682 if (root->parse->hasTargetSRFs)
683 return;
684
685 /*
686 * Eager aggregation only makes sense if there are multiple base rels in
687 * the query.
688 */
689 if (bms_membership(root->all_baserels) != BMS_MULTIPLE)
690 return;
691
692 /*
693 * Don't apply eager aggregation if any aggregate poses a risk of
694 * excessive memory usage during partial aggregation.
695 */
697 return;
698
699 /*
700 * Collect aggregate expressions and plain Vars that appear in the
701 * targetlist and havingQual.
702 */
704
705 /*
706 * If there are no suitable aggregate expressions, we cannot apply eager
707 * aggregation.
708 */
709 if (root->agg_clause_list == NIL)
710 return;
711
712 /*
713 * Collect grouping expressions that appear in grouping clauses.
714 */
716}
bool enable_eager_aggregate
Definition allpaths.c:82
static void create_grouping_expr_infos(PlannerInfo *root)
Definition initsplan.c:880
static bool is_partial_agg_memory_risky(PlannerInfo *root)
Definition initsplan.c:730
static void create_agg_clause_infos(PlannerInfo *root)
Definition initsplan.c:751

References bms_membership(), BMS_MULTIPLE, create_agg_clause_infos(), create_grouping_expr_infos(), enable_eager_aggregate, is_partial_agg_memory_risky(), NIL, and root.

Referenced by query_planner().

◆ trivial_subqueryscan()

bool trivial_subqueryscan ( SubqueryScan plan)
extern

Definition at line 1528 of file setrefs.c.

1529{
1530 int attrno;
1531 ListCell *lp,
1532 *lc;
1533
1534 /* We might have detected this already; in which case reuse the result */
1535 if (plan->scanstatus == SUBQUERY_SCAN_TRIVIAL)
1536 return true;
1537 if (plan->scanstatus == SUBQUERY_SCAN_NONTRIVIAL)
1538 return false;
1539 Assert(plan->scanstatus == SUBQUERY_SCAN_UNKNOWN);
1540 /* Initially, mark the SubqueryScan as non-deletable from the plan tree */
1541 plan->scanstatus = SUBQUERY_SCAN_NONTRIVIAL;
1542
1543 if (plan->scan.plan.qual != NIL)
1544 return false;
1545
1546 if (list_length(plan->scan.plan.targetlist) !=
1547 list_length(plan->subplan->targetlist))
1548 return false; /* tlists not same length */
1549
1550 attrno = 1;
1551 forboth(lp, plan->scan.plan.targetlist, lc, plan->subplan->targetlist)
1552 {
1555
1556 if (ptle->resjunk != ctle->resjunk)
1557 return false; /* tlist doesn't match junk status */
1558
1559 /*
1560 * We accept either a Var referencing the corresponding element of the
1561 * subplan tlist, or a Const equaling the subplan element. See
1562 * generate_setop_tlist() for motivation.
1563 */
1564 if (ptle->expr && IsA(ptle->expr, Var))
1565 {
1566 Var *var = (Var *) ptle->expr;
1567
1568 Assert(var->varno == plan->scan.scanrelid);
1569 Assert(var->varlevelsup == 0);
1570 if (var->varattno != attrno)
1571 return false; /* out of order */
1572 }
1573 else if (ptle->expr && IsA(ptle->expr, Const))
1574 {
1575 if (!equal(ptle->expr, ctle->expr))
1576 return false;
1577 }
1578 else
1579 return false;
1580
1581 attrno++;
1582 }
1583
1584 /* Re-mark the SubqueryScan as deletable from the plan tree */
1585 plan->scanstatus = SUBQUERY_SCAN_TRIVIAL;
1586
1587 return true;
1588}
#define forboth(cell1, list1, cell2, list2)
Definition pg_list.h:550
@ SUBQUERY_SCAN_NONTRIVIAL
Definition plannodes.h:770
@ SUBQUERY_SCAN_UNKNOWN
Definition plannodes.h:768
@ SUBQUERY_SCAN_TRIVIAL
Definition plannodes.h:769

References Assert, equal(), fb(), forboth, IsA, lfirst, list_length(), NIL, plan, SUBQUERY_SCAN_NONTRIVIAL, SUBQUERY_SCAN_TRIVIAL, SUBQUERY_SCAN_UNKNOWN, Var::varattno, Var::varlevelsup, and Var::varno.

Referenced by mark_async_capable_plan(), and set_subqueryscan_references().

Variable Documentation

◆ cursor_tuple_fraction

PGDLLIMPORT double cursor_tuple_fraction
extern

Definition at line 68 of file planner.c.

Referenced by standard_planner().

◆ enable_self_join_elimination

PGDLLIMPORT bool enable_self_join_elimination
extern

Definition at line 54 of file analyzejoins.c.

Referenced by remove_useless_self_joins().

◆ from_collapse_limit

PGDLLIMPORT int from_collapse_limit
extern

Definition at line 41 of file initsplan.c.

Referenced by deconstruct_recurse().

◆ join_collapse_limit

PGDLLIMPORT int join_collapse_limit
extern

Definition at line 42 of file initsplan.c.

Referenced by deconstruct_recurse().