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:39
#define elog(elevel,...)
Definition elog.h:227
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:83
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 3788 of file initsplan.c.

3795{
3797 Expr *clause;
3798
3799 /*
3800 * Build the new clause. Copy to ensure it shares no substructure with
3801 * original (this is necessary in case there are subselects in there...)
3802 */
3803 clause = make_opclause(opno,
3804 BOOLOID, /* opresulttype */
3805 false, /* opretset */
3808 InvalidOid,
3809 collation);
3810
3811 /*
3812 * Build the RestrictInfo node itself.
3813 */
3815 clause,
3816 true, /* is_pushed_down */
3817 false, /* !has_clone */
3818 false, /* !is_clone */
3819 false, /* pseudoconstant */
3820 security_level, /* security_level */
3821 qualscope, /* required_relids */
3822 NULL, /* incompatible_relids */
3823 NULL); /* outer_relids */
3824
3825 /* Set mergejoinability/hashjoinability flags */
3829
3830 return restrictinfo;
3831}
static void check_hashjoinable(RestrictInfo *restrictinfo)
Definition initsplan.c:4165
static void check_mergejoinable(RestrictInfo *restrictinfo)
Definition initsplan.c:4128
static void check_memoizable(RestrictInfo *restrictinfo)
Definition initsplan.c:4193
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 1213 of file initsplan.c.

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

1453{
1454 List *result;
1456 List *item_list = NIL;
1457 ListCell *lc;
1458
1459 /*
1460 * After this point, no more PlaceHolderInfos may be made, because
1461 * make_outerjoininfo requires all active placeholders to be present in
1462 * root->placeholder_list while we crawl up the join tree.
1463 */
1464 root->placeholdersFrozen = true;
1465
1466 /* Fetch the already-created top-level join domain for the query */
1467 top_jdomain = linitial_node(JoinDomain, root->join_domains);
1468 top_jdomain->jd_relids = NULL; /* filled during deconstruct_recurse */
1469
1470 /* Start recursion at top of jointree */
1471 Assert(root->parse->jointree != NULL &&
1472 IsA(root->parse->jointree, FromExpr));
1473
1474 /* These are filled as we scan the jointree */
1475 root->all_baserels = NULL;
1476 root->outer_join_rels = NULL;
1477
1478 /* Perform the initial scan of the jointree */
1479 result = deconstruct_recurse(root, (Node *) root->parse->jointree,
1481 &item_list);
1482
1483 /* Now we can form the value of all_query_rels, too */
1484 root->all_query_rels = bms_union(root->all_baserels, root->outer_join_rels);
1485
1486 /* ... which should match what we computed for the top join domain */
1487 Assert(bms_equal(root->all_query_rels, top_jdomain->jd_relids));
1488
1489 /* Now scan all the jointree nodes again, and distribute quals */
1490 foreach(lc, item_list)
1491 {
1493
1495 }
1496
1497 /*
1498 * If there were any special joins then we may have some postponed LEFT
1499 * JOIN clauses to deal with.
1500 */
1501 if (root->join_info_list)
1502 {
1503 foreach(lc, item_list)
1504 {
1506
1507 if (jtitem->oj_joinclauses != NIL)
1509 }
1510 }
1511
1512 /* Don't need the JoinTreeItems any more */
1514
1515 return result;
1516}
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
static List * deconstruct_recurse(PlannerInfo *root, Node *jtnode, JoinDomain *parent_domain, JoinTreeItem *parent_jtitem, List **item_list)
Definition initsplan.c:1534
static void deconstruct_distribute_oj_quals(PlannerInfo *root, List *jtitems, JoinTreeItem *jtitem)
Definition initsplan.c:2594
static void deconstruct_distribute(PlannerInfo *root, JoinTreeItem *jtitem)
Definition initsplan.c:1832
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, and root.

Referenced by query_planner().

◆ distribute_restrictinfo_to_rels()

void distribute_restrictinfo_to_rels ( PlannerInfo root,
RestrictInfo restrictinfo 
)
extern

Definition at line 3559 of file initsplan.c.

3561{
3562 Relids relids = restrictinfo->required_relids;
3563
3564 if (!bms_is_empty(relids))
3565 {
3566 int relid;
3567
3568 if (bms_get_singleton_member(relids, &relid))
3569 {
3570 /*
3571 * There is only one relation participating in the clause, so it
3572 * is a restriction clause for that relation.
3573 */
3575 }
3576 else
3577 {
3578 /*
3579 * The clause is a join clause, since there is more than one rel
3580 * in its relid set.
3581 */
3582
3583 /*
3584 * Check for hashjoinable operators. (We don't bother setting the
3585 * hashjoin info except in true join clauses.)
3586 */
3588
3589 /*
3590 * Likewise, check if the clause is suitable to be used with a
3591 * Memoize node to cache inner tuples during a parameterized
3592 * nested loop.
3593 */
3595
3596 /*
3597 * Add clause to the join lists of all the relevant relations.
3598 */
3600 }
3601 }
3602 else
3603 {
3604 /*
3605 * clause references no rels, and therefore we have no place to attach
3606 * it. Shouldn't get here if callers are working properly.
3607 */
3608 elog(ERROR, "cannot cope with variable-free clause");
3609 }
3610}
static void add_base_clause_to_rel(PlannerInfo *root, Index relid, RestrictInfo *restrictinfo)
Definition initsplan.c:3348
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:2198

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 1026 of file initsplan.c.

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

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 1310 of file analyzejoins.c.

1317{
1318 return innerrel_is_unique_ext(root, joinrelids, outerrelids, innerrel,
1319 jointype, restrictlist, force_cache, NULL);
1320}
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 1332 of file analyzejoins.c.

1340{
1342 ListCell *lc;
1344 List *outer_exprs = NIL;
1345 bool self_join = (extra_clauses != NULL);
1346
1347 /* Certainly can't prove uniqueness when there are no joinclauses */
1348 if (restrictlist == NIL)
1349 return false;
1350
1351 /*
1352 * Make a quick check to eliminate cases in which we will surely be unable
1353 * to prove uniqueness of the innerrel.
1354 */
1355 if (!rel_supports_distinctness(root, innerrel))
1356 return false;
1357
1358 /*
1359 * Query the cache to see if we've managed to prove that innerrel is
1360 * unique for any subset of this outerrel. For non-self-join search, we
1361 * don't need an exact match, as extra outerrels can't make the innerrel
1362 * any less unique (or more formally, the restrictlist for a join to a
1363 * superset outerrel must be a superset of the conditions we successfully
1364 * used before). For self-join search, we require an exact match of
1365 * outerrels because we need extra clauses to be valid for our case. Also,
1366 * for self-join checking we've filtered the clauses list. Thus, we can
1367 * match only the result cached for a self-join search for another
1368 * self-join check.
1369 */
1370 foreach(lc, innerrel->unique_for_rels)
1371 {
1373
1374 if ((!self_join && bms_is_subset(uniqueRelInfo->outerrelids, outerrelids)) ||
1375 (self_join && bms_equal(uniqueRelInfo->outerrelids, outerrelids) &&
1376 uniqueRelInfo->self_join))
1377 {
1378 if (extra_clauses)
1379 *extra_clauses = uniqueRelInfo->extra_clauses;
1380 return true; /* Success! */
1381 }
1382 }
1383
1384 /*
1385 * Conversely, we may have already determined that this outerrel, or some
1386 * superset thereof, cannot prove this innerrel to be unique.
1387 */
1388 foreach(lc, innerrel->non_unique_for_rels)
1389 {
1390 Relids unique_for_rels = (Relids) lfirst(lc);
1391
1392 if (bms_is_subset(outerrelids, unique_for_rels))
1393 return false;
1394 }
1395
1396 /* No cached information, so try to make the proof. */
1397 if (is_innerrel_unique_for(root, joinrelids, outerrelids, innerrel,
1398 jointype, restrictlist,
1399 self_join ? &outer_exprs : NULL))
1400 {
1401 /*
1402 * Cache the positive result for future probes, being sure to keep it
1403 * in the planner_cxt even if we are working in GEQO.
1404 *
1405 * Note: one might consider trying to isolate the minimal subset of
1406 * the outerrels that proved the innerrel unique. But it's not worth
1407 * the trouble, because the planner builds up joinrels incrementally
1408 * and so we'll see the minimally sufficient outerrels before any
1409 * supersets of them anyway.
1410 */
1411 old_context = MemoryContextSwitchTo(root->planner_cxt);
1413 uniqueRelInfo->outerrelids = bms_copy(outerrelids);
1414 uniqueRelInfo->self_join = self_join;
1415 uniqueRelInfo->extra_clauses = outer_exprs;
1416 innerrel->unique_for_rels = lappend(innerrel->unique_for_rels,
1419
1420 if (extra_clauses)
1421 *extra_clauses = outer_exprs;
1422 return true; /* Success! */
1423 }
1424 else
1425 {
1426 /*
1427 * None of the join conditions for outerrel proved innerrel unique, so
1428 * we can safely reject this outerrel or any subset of it in future
1429 * checks.
1430 *
1431 * However, in normal planning mode, caching this knowledge is totally
1432 * pointless; it won't be queried again, because we build up joinrels
1433 * from smaller to larger. It's only useful when using GEQO or
1434 * another planner extension that attempts planning multiple times.
1435 *
1436 * Also, allow callers to override that heuristic and force caching;
1437 * that's useful for reduce_unique_semijoins, which calls here before
1438 * the normal join search starts.
1439 */
1440 if (force_cache || root->assumeReplanning)
1441 {
1442 old_context = MemoryContextSwitchTo(root->planner_cxt);
1443 innerrel->non_unique_for_rels =
1444 lappend(innerrel->non_unique_for_rels,
1445 bms_copy(outerrelids));
1447 }
1448
1449 return false;
1450 }
1451}
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 3963 of file initsplan.c.

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

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

1118{
1119 ListCell *l;
1120 Oid opid;
1121
1123
1124 /*
1125 * DISTINCT (including DISTINCT ON) guarantees uniqueness if all the
1126 * columns in the DISTINCT clause appear in colnos and operator semantics
1127 * match. This is true even if there are SRFs in the DISTINCT columns or
1128 * elsewhere in the tlist.
1129 */
1130 if (query->distinctClause)
1131 {
1132 foreach(l, query->distinctClause)
1133 {
1136 query->targetList);
1137
1139 if (!OidIsValid(opid) ||
1141 break; /* exit early if no match */
1142 }
1143 if (l == NULL) /* had matches for all? */
1144 return true;
1145 }
1146
1147 /*
1148 * Otherwise, a set-returning function in the query's targetlist can
1149 * result in returning duplicate rows, despite any grouping that might
1150 * occur before tlist evaluation. (If all tlist SRFs are within GROUP BY
1151 * columns, it would be safe because they'd be expanded before grouping.
1152 * But it doesn't currently seem worth the effort to check for that.)
1153 */
1154 if (query->hasTargetSRFs)
1155 return false;
1156
1157 /*
1158 * Similarly, GROUP BY without GROUPING SETS guarantees uniqueness if all
1159 * the grouped columns appear in colnos and operator semantics match.
1160 */
1161 if (query->groupClause && !query->groupingSets)
1162 {
1163 foreach(l, query->groupClause)
1164 {
1167 query->targetList);
1168
1170 if (!OidIsValid(opid) ||
1172 break; /* exit early if no match */
1173 }
1174 if (l == NULL) /* had matches for all? */
1175 return true;
1176 }
1177 else if (query->groupingSets)
1178 {
1179 List *gsets;
1180
1181 /*
1182 * If we have grouping sets with expressions, we probably don't have
1183 * uniqueness and analysis would be hard. Punt.
1184 */
1185 if (query->groupClause)
1186 return false;
1187
1188 /*
1189 * If we have no groupClause (therefore no grouping expressions), we
1190 * might have one or many empty grouping sets. If there's just one,
1191 * or if the DISTINCT clause is used on the GROUP BY, then we're
1192 * returning only one row and are certainly unique. But otherwise, we
1193 * know we're certainly not unique.
1194 */
1195 if (query->groupDistinct)
1196 return true;
1197
1198 gsets = expand_grouping_sets(query->groupingSets, false, -1);
1199
1200 return (list_length(gsets) == 1);
1201 }
1202 else
1203 {
1204 /*
1205 * If we have no GROUP BY, but do have aggregates or HAVING, then the
1206 * result is at most one row so it's surely unique, for any operators.
1207 */
1208 if (query->hasAggs || query->havingQual)
1209 return true;
1210 }
1211
1212 /*
1213 * UNION, INTERSECT, EXCEPT guarantee uniqueness of the whole output row,
1214 * except with ALL.
1215 */
1216 if (query->setOperations)
1217 {
1219
1220 Assert(topop->op != SETOP_NONE);
1221
1222 if (!topop->all)
1223 {
1224 ListCell *lg;
1225
1226 /* We're good if all the nonjunk output columns are in colnos */
1227 lg = list_head(topop->groupClauses);
1228 foreach(l, query->targetList)
1229 {
1232
1233 if (tle->resjunk)
1234 continue; /* ignore resjunk columns */
1235
1236 /* non-resjunk columns should have grouping clauses */
1237 Assert(lg != NULL);
1239 lg = lnext(topop->groupClauses, lg);
1240
1242 if (!OidIsValid(opid) ||
1244 break; /* exit early if no match */
1245 }
1246 if (l == NULL) /* had matches for all? */
1247 return true;
1248 }
1249 }
1250
1251 /*
1252 * XXX Are there any other cases in which we can easily see the result
1253 * must be distinct?
1254 *
1255 * If you do add more smarts to this function, be sure to update
1256 * query_supports_distinctness() to match.
1257 */
1258
1259 return false;
1260}
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:764
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:3963
void find_lateral_references(PlannerInfo *root)
Definition initsplan.c:1026
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:1452
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:1213
@ 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 1079 of file analyzejoins.c.

1080{
1081 /* SRFs break distinctness except with DISTINCT, see below */
1082 if (query->hasTargetSRFs && query->distinctClause == NIL)
1083 return false;
1084
1085 /* check for features we can prove distinctness with */
1086 if (query->distinctClause != NIL ||
1087 query->groupClause != NIL ||
1088 query->groupingSets != NIL ||
1089 query->hasAggs ||
1090 query->havingQual ||
1091 query->setOperations)
1092 return true;
1093
1094 return false;
1095}

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 3891 of file initsplan.c.

3892{
3893 /*
3894 * We must examine all join clauses, but there's no value in processing
3895 * any join clause more than once. So it's slightly annoying that we have
3896 * to find them via the per-base-relation joininfo lists. Avoid duplicate
3897 * processing by tracking the rinfo_serial numbers of join clauses we've
3898 * already seen. (This doesn't work for is_clone clauses, so we must
3899 * waste effort on them.)
3900 */
3902 Index rti;
3903
3904 /* Scan all baserels for join clauses */
3905 for (rti = 1; rti < root->simple_rel_array_size; rti++)
3906 {
3907 RelOptInfo *brel = root->simple_rel_array[rti];
3908 ListCell *lc;
3909
3910 if (brel == NULL)
3911 continue;
3912 if (brel->reloptkind != RELOPT_BASEREL)
3913 continue;
3914
3915 foreach(lc, brel->joininfo)
3916 {
3917 RestrictInfo *rinfo = (RestrictInfo *) lfirst(lc);
3918 Relids relids = rinfo->required_relids;
3919
3920 if (!rinfo->is_clone) /* else serial number is not unique */
3921 {
3923 continue; /* saw it already */
3925 rinfo->rinfo_serial);
3926 }
3927
3928 if (bms_membership(relids) == BMS_MULTIPLE)
3929 {
3930 List *vars = pull_var_clause((Node *) rinfo->clause,
3935
3936 if (rinfo->is_clone)
3937 where_needed = bms_intersect(relids, root->all_baserels);
3938 else
3939 where_needed = relids;
3941 list_free(vars);
3942 }
3943 }
3944 }
3945}
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 1175 of file initsplan.c.

1176{
1177 Index rti;
1178
1179 /* We need do nothing if the query contains no LATERAL RTEs */
1180 if (!root->hasLateralRTEs)
1181 return;
1182
1183 /* Examine the same baserels that find_lateral_references did */
1184 for (rti = 1; rti < root->simple_rel_array_size; rti++)
1185 {
1186 RelOptInfo *brel = root->simple_rel_array[rti];
1188
1189 if (brel == NULL)
1190 continue;
1191 if (brel->reloptkind != RELOPT_BASEREL)
1192 continue;
1193
1194 /*
1195 * We don't need to repeat all of extract_lateral_references, since it
1196 * kindly saved the extracted Vars/PHVs in lateral_vars.
1197 */
1198 if (brel->lateral_vars == NIL)
1199 continue;
1200
1202
1204 }
1205}

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 845 of file analyzejoins.c.

846{
847 ListCell *lc;
848
849 /*
850 * Scan the join_info_list to find semijoins.
851 */
852 foreach(lc, root->join_info_list)
853 {
855 int innerrelid;
856 RelOptInfo *innerrel;
858 List *restrictlist;
859
860 /*
861 * Must be a semijoin to a single baserel, else we aren't going to be
862 * able to do anything with it.
863 */
864 if (sjinfo->jointype != JOIN_SEMI)
865 continue;
866
868 continue;
869
870 innerrel = find_base_rel(root, innerrelid);
871
872 /*
873 * Before we trouble to run generate_join_implied_equalities, make a
874 * quick check to eliminate cases in which we will surely be unable to
875 * prove uniqueness of the innerrel.
876 */
877 if (!rel_supports_distinctness(root, innerrel))
878 continue;
879
880 /* Compute the relid set for the join we are considering */
882 Assert(sjinfo->ojrelid == 0); /* SEMI joins don't have RT indexes */
883
884 /*
885 * Since we're only considering a single-rel RHS, any join clauses it
886 * has must be clauses linking it to the semijoin's min_lefthand. We
887 * can also consider EC-derived join clauses.
888 */
889 restrictlist =
892 sjinfo->min_lefthand,
893 innerrel,
894 NULL),
895 innerrel->joininfo);
896
897 /* Test whether the innerrel is unique for those clauses. */
899 joinrelids, sjinfo->min_lefthand, innerrel,
900 JOIN_SEMI, restrictlist, true))
901 continue;
902
903 /* OK, remove the SpecialJoinInfo from the list. */
904 root->join_info_list = foreach_delete_current(root->join_info_list, lc);
905 }
906}
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 91 of file analyzejoins.c.

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

2489{
2491 int relid = -1;
2492
2495 return joinlist;
2496
2497 /*
2498 * Merge pairs of relations participated in self-join. Remove unnecessary
2499 * range table entries.
2500 */
2502
2503 if (unlikely(toRemove != NULL))
2504 {
2505 /* At the end, remove orphaned relation links */
2506 while ((relid = bms_next_member(toRemove, relid)) >= 0)
2507 {
2508 int nremoved = 0;
2509
2511 if (nremoved != 1)
2512 elog(ERROR, "failed to find relation %d in joinlist", relid);
2513 }
2514 }
2515
2516 return joinlist;
2517}
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 3488 of file initsplan.c.

3490{
3491 /*
3492 * For a clone clause, we don't have a reliable way to determine if the
3493 * input expression of a NullTest is non-nullable: nullingrel bits in
3494 * clone clauses may not reflect reality, so we dare not draw conclusions
3495 * from clones about whether Vars are guaranteed not-null.
3496 */
3497 if (restrictinfo->has_clone || restrictinfo->is_clone)
3498 return false;
3499
3500 /* Check for NullTest qual */
3501 if (IsA(restrictinfo->clause, NullTest))
3502 {
3503 NullTest *nulltest = (NullTest *) restrictinfo->clause;
3504
3505 /* is this NullTest an IS_NULL qual? */
3506 if (nulltest->nulltesttype != IS_NULL)
3507 return false;
3508
3509 /*
3510 * Empty rows can appear NULL in some contexts and NOT NULL in others,
3511 * so avoid this optimization for row expressions.
3512 */
3513 if (nulltest->argisrow)
3514 return false;
3515
3517 }
3518
3519 /* If it's an OR, check its sub-clauses */
3521 {
3522 ListCell *lc;
3523
3524 Assert(is_orclause(restrictinfo->orclause));
3525
3526 /*
3527 * Currently, when processing OR expressions, we only return true when
3528 * all of the OR branches are always false. This could perhaps be
3529 * expanded to remove OR branches that are provably false. This may
3530 * be a useful thing to do as it could result in the OR being left
3531 * with a single arg. That's useful as it would allow the OR
3532 * condition to be replaced with its single argument which may allow
3533 * use of an index for faster filtering on the remaining condition.
3534 */
3535 foreach(lc, ((BoolExpr *) restrictinfo->orclause)->args)
3536 {
3537 Node *orarg = (Node *) lfirst(lc);
3538
3539 if (!IsA(orarg, RestrictInfo) ||
3541 return false;
3542 }
3543 return true;
3544 }
3545
3546 return false;
3547}
bool expr_is_nonnullable(PlannerInfo *root, Expr *expr, NotNullSource source)
Definition clauses.c:4749
bool restriction_is_always_false(PlannerInfo *root, RestrictInfo *restrictinfo)
Definition initsplan.c:3488
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 3423 of file initsplan.c.

3425{
3426 /*
3427 * For a clone clause, we don't have a reliable way to determine if the
3428 * input expression of a NullTest is non-nullable: nullingrel bits in
3429 * clone clauses may not reflect reality, so we dare not draw conclusions
3430 * from clones about whether Vars are guaranteed not-null.
3431 */
3432 if (restrictinfo->has_clone || restrictinfo->is_clone)
3433 return false;
3434
3435 /* Check for NullTest qual */
3436 if (IsA(restrictinfo->clause, NullTest))
3437 {
3438 NullTest *nulltest = (NullTest *) restrictinfo->clause;
3439
3440 /* is this NullTest an IS_NOT_NULL qual? */
3441 if (nulltest->nulltesttype != IS_NOT_NULL)
3442 return false;
3443
3444 /*
3445 * Empty rows can appear NULL in some contexts and NOT NULL in others,
3446 * so avoid this optimization for row expressions.
3447 */
3448 if (nulltest->argisrow)
3449 return false;
3450
3452 }
3453
3454 /* If it's an OR, check its sub-clauses */
3456 {
3457 ListCell *lc;
3458
3459 Assert(is_orclause(restrictinfo->orclause));
3460
3461 /*
3462 * if any of the given OR branches is provably always true then the
3463 * entire condition is true.
3464 */
3465 foreach(lc, ((BoolExpr *) restrictinfo->orclause)->args)
3466 {
3467 Node *orarg = (Node *) lfirst(lc);
3468
3469 if (!IsA(orarg, RestrictInfo))
3470 continue;
3471
3473 return true;
3474 }
3475 }
3476
3477 return false;
3478}
bool restriction_is_always_true(PlannerInfo *root, RestrictInfo *restrictinfo)
Definition initsplan.c:3423
@ 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}
#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(), NIL, palloc0(), palloc_object, plan, 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:869
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().