Thanks to visit codestin.com
Credit goes to doxygen.postgresql.org

PostgreSQL Source Code git master
optimizer.h File Reference
#include "nodes/parsenodes.h"
Include dependency graph for optimizer.h:
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Macros

#define PVC_INCLUDE_AGGREGATES   0x0001 /* include Aggrefs in output list */
 
#define PVC_RECURSE_AGGREGATES   0x0002 /* recurse into Aggref arguments */
 
#define PVC_INCLUDE_WINDOWFUNCS   0x0004 /* include WindowFuncs in output list */
 
#define PVC_RECURSE_WINDOWFUNCS   0x0008 /* recurse into WindowFunc arguments */
 
#define PVC_INCLUDE_PLACEHOLDERS
 
#define PVC_RECURSE_PLACEHOLDERS
 
#define PVC_INCLUDE_CONVERTROWTYPES
 

Typedefs

typedef struct PlannerInfo PlannerInfo
 
typedef struct IndexOptInfo IndexOptInfo
 
typedef struct SpecialJoinInfo SpecialJoinInfo
 
typedef struct PlannedStmt PlannedStmt
 
typedef struct ParamListInfoDataParamListInfo
 
typedef struct HeapTupleDataHeapTuple
 

Enumerations

enum  DebugParallelMode { DEBUG_PARALLEL_OFF , DEBUG_PARALLEL_ON , DEBUG_PARALLEL_REGRESS }
 

Functions

Selectivity clause_selectivity (PlannerInfo *root, Node *clause, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
 
Selectivity clause_selectivity_ext (PlannerInfo *root, Node *clause, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, bool use_extended_stats)
 
Selectivity clauselist_selectivity (PlannerInfo *root, List *clauses, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
 
Selectivity clauselist_selectivity_ext (PlannerInfo *root, List *clauses, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, bool use_extended_stats)
 
double clamp_row_est (double nrows)
 
int32 clamp_width_est (int64 tuple_width)
 
long clamp_cardinality_to_long (Cardinality x)
 
bool is_pseudo_constant_for_index (PlannerInfo *root, Node *expr, IndexOptInfo *index)
 
PlannedStmtplanner (Query *parse, const char *query_string, int cursorOptions, ParamListInfo boundParams)
 
Exprexpression_planner (Expr *expr)
 
Exprexpression_planner_with_deps (Expr *expr, List **relationOids, List **invalItems)
 
bool plan_cluster_use_sort (Oid tableOid, Oid indexOid)
 
int plan_create_index_workers (Oid tableOid, Oid indexOid)
 
void extract_query_dependencies (Node *query, List **relationOids, List **invalItems, bool *hasRowSecurity)
 
Nodenegate_clause (Node *node)
 
Exprcanonicalize_qual (Expr *qual, bool is_check)
 
bool contain_mutable_functions (Node *clause)
 
bool contain_mutable_functions_after_planning (Expr *expr)
 
bool contain_volatile_functions (Node *clause)
 
bool contain_volatile_functions_after_planning (Expr *expr)
 
bool contain_volatile_functions_not_nextval (Node *clause)
 
Nodeeval_const_expressions (PlannerInfo *root, Node *node)
 
void convert_saop_to_hashed_saop (Node *node)
 
Nodeestimate_expression_value (PlannerInfo *root, Node *node)
 
Exprevaluate_expr (Expr *expr, Oid result_type, int32 result_typmod, Oid result_collation)
 
bool var_is_nonnullable (PlannerInfo *root, Var *var, bool use_rel_info)
 
Listexpand_function_arguments (List *args, bool include_out_arguments, Oid result_type, HeapTuple func_tuple)
 
ScalarArrayOpExprmake_SAOP_expr (Oid oper, Node *leftexpr, Oid coltype, Oid arraycollid, Oid inputcollid, List *exprs, bool haveNonConst)
 
bool predicate_implied_by (List *predicate_list, List *clause_list, bool weak)
 
bool predicate_refuted_by (List *predicate_list, List *clause_list, bool weak)
 
int count_nonjunk_tlist_entries (List *tlist)
 
TargetEntryget_sortgroupref_tle (Index sortref, List *targetList)
 
TargetEntryget_sortgroupclause_tle (SortGroupClause *sgClause, List *targetList)
 
Nodeget_sortgroupclause_expr (SortGroupClause *sgClause, List *targetList)
 
Listget_sortgrouplist_exprs (List *sgClauses, List *targetList)
 
SortGroupClauseget_sortgroupref_clause (Index sortref, List *clauses)
 
SortGroupClauseget_sortgroupref_clause_noerr (Index sortref, List *clauses)
 
Bitmapsetpull_varnos (PlannerInfo *root, Node *node)
 
Bitmapsetpull_varnos_of_level (PlannerInfo *root, Node *node, int levelsup)
 
void pull_varattnos (Node *node, Index varno, Bitmapset **varattnos)
 
Listpull_vars_of_level (Node *node, int levelsup)
 
bool contain_var_clause (Node *node)
 
bool contain_vars_of_level (Node *node, int levelsup)
 
bool contain_vars_returning_old_or_new (Node *node)
 
int locate_var_of_level (Node *node, int levelsup)
 
Listpull_var_clause (Node *node, int flags)
 
Nodeflatten_join_alias_vars (PlannerInfo *root, Query *query, Node *node)
 
Nodeflatten_group_exprs (PlannerInfo *root, Query *query, Node *node)
 

Variables

PGDLLIMPORT double seq_page_cost
 
PGDLLIMPORT double random_page_cost
 
PGDLLIMPORT double cpu_tuple_cost
 
PGDLLIMPORT double cpu_index_tuple_cost
 
PGDLLIMPORT double cpu_operator_cost
 
PGDLLIMPORT double parallel_tuple_cost
 
PGDLLIMPORT double parallel_setup_cost
 
PGDLLIMPORT double recursive_worktable_factor
 
PGDLLIMPORT int effective_cache_size
 
PGDLLIMPORT int debug_parallel_query
 
PGDLLIMPORT bool parallel_leader_participation
 
PGDLLIMPORT bool enable_distinct_reordering
 

Macro Definition Documentation

◆ PVC_INCLUDE_AGGREGATES

#define PVC_INCLUDE_AGGREGATES   0x0001 /* include Aggrefs in output list */

Definition at line 183 of file optimizer.h.

◆ PVC_INCLUDE_CONVERTROWTYPES

#define PVC_INCLUDE_CONVERTROWTYPES
Value:
0x0040 /* include ConvertRowtypeExprs in
* output list */

Definition at line 189 of file optimizer.h.

◆ PVC_INCLUDE_PLACEHOLDERS

#define PVC_INCLUDE_PLACEHOLDERS
Value:
0x0010 /* include PlaceHolderVars in
* output list */

Definition at line 187 of file optimizer.h.

◆ PVC_INCLUDE_WINDOWFUNCS

#define PVC_INCLUDE_WINDOWFUNCS   0x0004 /* include WindowFuncs in output list */

Definition at line 185 of file optimizer.h.

◆ PVC_RECURSE_AGGREGATES

#define PVC_RECURSE_AGGREGATES   0x0002 /* recurse into Aggref arguments */

Definition at line 184 of file optimizer.h.

◆ PVC_RECURSE_PLACEHOLDERS

#define PVC_RECURSE_PLACEHOLDERS
Value:
0x0020 /* recurse into PlaceHolderVar
* arguments */

Definition at line 188 of file optimizer.h.

◆ PVC_RECURSE_WINDOWFUNCS

#define PVC_RECURSE_WINDOWFUNCS   0x0008 /* recurse into WindowFunc arguments */

Definition at line 186 of file optimizer.h.

Typedef Documentation

◆ HeapTuple

typedef struct HeapTupleData* HeapTuple

Definition at line 40 of file optimizer.h.

◆ IndexOptInfo

typedef struct IndexOptInfo IndexOptInfo

Definition at line 34 of file optimizer.h.

◆ ParamListInfo

Definition at line 39 of file optimizer.h.

◆ PlannedStmt

typedef struct PlannedStmt PlannedStmt

Definition at line 38 of file optimizer.h.

◆ PlannerInfo

typedef struct PlannerInfo PlannerInfo

Definition at line 33 of file optimizer.h.

◆ SpecialJoinInfo

Definition at line 35 of file optimizer.h.

Enumeration Type Documentation

◆ DebugParallelMode

Enumerator
DEBUG_PARALLEL_OFF 
DEBUG_PARALLEL_ON 
DEBUG_PARALLEL_REGRESS 

Definition at line 93 of file optimizer.h.

94{
DebugParallelMode
Definition: optimizer.h:94
@ DEBUG_PARALLEL_REGRESS
Definition: optimizer.h:97
@ DEBUG_PARALLEL_ON
Definition: optimizer.h:96
@ DEBUG_PARALLEL_OFF
Definition: optimizer.h:95

Function Documentation

◆ canonicalize_qual()

Expr * canonicalize_qual ( Expr qual,
bool  is_check 
)

Definition at line 293 of file prepqual.c.

294{
295 Expr *newqual;
296
297 /* Quick exit for empty qual */
298 if (qual == NULL)
299 return NULL;
300
301 /* This should not be invoked on quals in implicit-AND format */
302 Assert(!IsA(qual, List));
303
304 /*
305 * Pull up redundant subclauses in OR-of-AND trees. We do this only
306 * within the top-level AND/OR structure; there's no point in looking
307 * deeper. Also remove any NULL constants in the top-level structure.
308 */
309 newqual = find_duplicate_ors(qual, is_check);
310
311 return newqual;
312}
Assert(PointerIsAligned(start, uint64))
#define IsA(nodeptr, _type_)
Definition: nodes.h:164
static Expr * find_duplicate_ors(Expr *qual, bool is_check)
Definition: prepqual.c:406
Definition: pg_list.h:54

References Assert(), find_duplicate_ors(), and IsA.

Referenced by ConstraintImpliedByRelConstraint(), convert_EXISTS_to_ANY(), DoCopy(), get_proposed_default_constraint(), get_relation_constraints(), preprocess_expression(), and RelationGetIndexPredicate().

◆ clamp_cardinality_to_long()

long clamp_cardinality_to_long ( Cardinality  x)

Definition at line 265 of file costsize.c.

266{
267 /*
268 * Just for paranoia's sake, ensure we do something sane with negative or
269 * NaN values.
270 */
271 if (isnan(x))
272 return LONG_MAX;
273 if (x <= 0)
274 return 0;
275
276 /*
277 * If "long" is 64 bits, then LONG_MAX cannot be represented exactly as a
278 * double. Casting it to double and back may well result in overflow due
279 * to rounding, so avoid doing that. We trust that any double value that
280 * compares strictly less than "(double) LONG_MAX" will cast to a
281 * representable "long" value.
282 */
283 return (x < (double) LONG_MAX) ? (long) x : LONG_MAX;
284}
int x
Definition: isn.c:75

References x.

Referenced by buildSubPlanHash(), create_recursiveunion_plan(), create_setop_plan(), and make_agg().

◆ clamp_row_est()

double clamp_row_est ( double  nrows)

Definition at line 213 of file costsize.c.

214{
215 /*
216 * Avoid infinite and NaN row estimates. Costs derived from such values
217 * are going to be useless. Also force the estimate to be at least one
218 * row, to make explain output look better and to avoid possible
219 * divide-by-zero when interpolating costs. Make it an integer, too.
220 */
221 if (nrows > MAXIMUM_ROWCOUNT || isnan(nrows))
222 nrows = MAXIMUM_ROWCOUNT;
223 else if (nrows <= 1.0)
224 nrows = 1.0;
225 else
226 nrows = rint(nrows);
227
228 return nrows;
229}
#define MAXIMUM_ROWCOUNT
Definition: costsize.c:128

References MAXIMUM_ROWCOUNT.

Referenced by adjust_limit_rows_costs(), approx_tuple_count(), bernoulli_samplescangetsamplesize(), calc_joinrel_size_estimate(), compute_bitmap_pages(), compute_gather_rows(), cost_agg(), cost_append(), cost_bitmap_heap_scan(), cost_group(), cost_index(), cost_seqscan(), cost_subplan(), cost_subqueryscan(), create_bitmap_subplan(), create_memoize_path(), estimate_array_length(), estimate_hash_bucket_stats(), estimate_num_groups(), estimate_path_cost_size(), estimate_size(), expression_returns_set_rows(), final_cost_hashjoin(), final_cost_mergejoin(), final_cost_nestloop(), get_parameterized_baserel_size(), get_variable_numdistinct(), get_windowclause_startup_tuples(), initial_cost_mergejoin(), set_baserel_size_estimates(), set_cte_size_estimates(), set_foreign_size(), system_rows_samplescangetsamplesize(), system_samplescangetsamplesize(), system_time_samplescangetsamplesize(), and table_block_relation_estimate_size().

◆ clamp_width_est()

int32 clamp_width_est ( int64  tuple_width)

Definition at line 242 of file costsize.c.

243{
244 /*
245 * Anything more than MaxAllocSize is clearly bogus, since we could not
246 * create a tuple that large.
247 */
248 if (tuple_width > MaxAllocSize)
249 return (int32) MaxAllocSize;
250
251 /*
252 * Unlike clamp_row_est, we just Assert that the value isn't negative,
253 * rather than masking such errors.
254 */
255 Assert(tuple_width >= 0);
256
257 return (int32) tuple_width;
258}
int32_t int32
Definition: c.h:535
#define MaxAllocSize
Definition: fe_memutils.h:22

References Assert(), and MaxAllocSize.

Referenced by add_placeholders_to_joinrel(), build_joinrel_tlist(), create_one_window_path(), get_rel_data_width(), set_pathtarget_cost_width(), and set_rel_width().

◆ clause_selectivity()

Selectivity clause_selectivity ( PlannerInfo root,
Node clause,
int  varRelid,
JoinType  jointype,
SpecialJoinInfo sjinfo 
)

Definition at line 667 of file clausesel.c.

672{
673 return clause_selectivity_ext(root, clause, varRelid,
674 jointype, sjinfo, true);
675}
Selectivity clause_selectivity_ext(PlannerInfo *root, Node *clause, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, bool use_extended_stats)
Definition: clausesel.c:684
tree ctl root
Definition: radixtree.h:1857

References clause_selectivity_ext(), and root.

Referenced by approx_tuple_count(), booltestsel(), consider_new_or_clause(), and get_foreign_key_join_selectivity().

◆ clause_selectivity_ext()

Selectivity clause_selectivity_ext ( PlannerInfo root,
Node clause,
int  varRelid,
JoinType  jointype,
SpecialJoinInfo sjinfo,
bool  use_extended_stats 
)

Definition at line 684 of file clausesel.c.

690{
691 Selectivity s1 = 0.5; /* default for any unhandled clause type */
692 RestrictInfo *rinfo = NULL;
693 bool cacheable = false;
694
695 if (clause == NULL) /* can this still happen? */
696 return s1;
697
698 if (IsA(clause, RestrictInfo))
699 {
700 rinfo = (RestrictInfo *) clause;
701
702 /*
703 * If the clause is marked pseudoconstant, then it will be used as a
704 * gating qual and should not affect selectivity estimates; hence
705 * return 1.0. The only exception is that a constant FALSE may be
706 * taken as having selectivity 0.0, since it will surely mean no rows
707 * out of the plan. This case is simple enough that we need not
708 * bother caching the result.
709 */
710 if (rinfo->pseudoconstant)
711 {
712 if (!IsA(rinfo->clause, Const))
713 return (Selectivity) 1.0;
714 }
715
716 /*
717 * If possible, cache the result of the selectivity calculation for
718 * the clause. We can cache if varRelid is zero or the clause
719 * contains only vars of that relid --- otherwise varRelid will affect
720 * the result, so mustn't cache. Outer join quals might be examined
721 * with either their join's actual jointype or JOIN_INNER, so we need
722 * two cache variables to remember both cases. Note: we assume the
723 * result won't change if we are switching the input relations or
724 * considering a unique-ified case, so we only need one cache variable
725 * for all non-JOIN_INNER cases.
726 */
727 if (varRelid == 0 ||
728 rinfo->num_base_rels == 0 ||
729 (rinfo->num_base_rels == 1 &&
730 bms_is_member(varRelid, rinfo->clause_relids)))
731 {
732 /* Cacheable --- do we already have the result? */
733 if (jointype == JOIN_INNER)
734 {
735 if (rinfo->norm_selec >= 0)
736 return rinfo->norm_selec;
737 }
738 else
739 {
740 if (rinfo->outer_selec >= 0)
741 return rinfo->outer_selec;
742 }
743 cacheable = true;
744 }
745
746 /*
747 * Proceed with examination of contained clause. If the clause is an
748 * OR-clause, we want to look at the variant with sub-RestrictInfos,
749 * so that per-subclause selectivities can be cached.
750 */
751 if (rinfo->orclause)
752 clause = (Node *) rinfo->orclause;
753 else
754 clause = (Node *) rinfo->clause;
755 }
756
757 if (IsA(clause, Var))
758 {
759 Var *var = (Var *) clause;
760
761 /*
762 * We probably shouldn't ever see an uplevel Var here, but if we do,
763 * return the default selectivity...
764 */
765 if (var->varlevelsup == 0 &&
766 (varRelid == 0 || varRelid == (int) var->varno))
767 {
768 /* Use the restriction selectivity function for a bool Var */
769 s1 = boolvarsel(root, (Node *) var, varRelid);
770 }
771 }
772 else if (IsA(clause, Const))
773 {
774 /* bool constant is pretty easy... */
775 Const *con = (Const *) clause;
776
777 s1 = con->constisnull ? 0.0 :
778 DatumGetBool(con->constvalue) ? 1.0 : 0.0;
779 }
780 else if (IsA(clause, Param))
781 {
782 /* see if we can replace the Param */
783 Node *subst = estimate_expression_value(root, clause);
784
785 if (IsA(subst, Const))
786 {
787 /* bool constant is pretty easy... */
788 Const *con = (Const *) subst;
789
790 s1 = con->constisnull ? 0.0 :
791 DatumGetBool(con->constvalue) ? 1.0 : 0.0;
792 }
793 else
794 {
795 /* XXX any way to do better than default? */
796 }
797 }
798 else if (is_notclause(clause))
799 {
800 /* inverse of the selectivity of the underlying clause */
802 (Node *) get_notclausearg((Expr *) clause),
803 varRelid,
804 jointype,
805 sjinfo,
806 use_extended_stats);
807 }
808 else if (is_andclause(clause))
809 {
810 /* share code with clauselist_selectivity() */
812 ((BoolExpr *) clause)->args,
813 varRelid,
814 jointype,
815 sjinfo,
816 use_extended_stats);
817 }
818 else if (is_orclause(clause))
819 {
820 /*
821 * Almost the same thing as clauselist_selectivity, but with the
822 * clauses connected by OR.
823 */
825 ((BoolExpr *) clause)->args,
826 varRelid,
827 jointype,
828 sjinfo,
829 use_extended_stats);
830 }
831 else if (is_opclause(clause) || IsA(clause, DistinctExpr))
832 {
833 OpExpr *opclause = (OpExpr *) clause;
834 Oid opno = opclause->opno;
835
836 if (treat_as_join_clause(root, clause, rinfo, varRelid, sjinfo))
837 {
838 /* Estimate selectivity for a join clause. */
839 s1 = join_selectivity(root, opno,
840 opclause->args,
841 opclause->inputcollid,
842 jointype,
843 sjinfo);
844 }
845 else
846 {
847 /* Estimate selectivity for a restriction clause. */
849 opclause->args,
850 opclause->inputcollid,
851 varRelid);
852 }
853
854 /*
855 * DistinctExpr has the same representation as OpExpr, but the
856 * contained operator is "=" not "<>", so we must negate the result.
857 * This estimation method doesn't give the right behavior for nulls,
858 * but it's better than doing nothing.
859 */
860 if (IsA(clause, DistinctExpr))
861 s1 = 1.0 - s1;
862 }
863 else if (is_funcclause(clause))
864 {
865 FuncExpr *funcclause = (FuncExpr *) clause;
866
867 /* Try to get an estimate from the support function, if any */
869 funcclause->funcid,
870 funcclause->args,
871 funcclause->inputcollid,
872 treat_as_join_clause(root, clause, rinfo,
873 varRelid, sjinfo),
874 varRelid,
875 jointype,
876 sjinfo);
877
878 /* If no support, fall back on boolvarsel */
879 if (s1 < 0)
880 s1 = boolvarsel(root, clause, varRelid);
881 }
882 else if (IsA(clause, ScalarArrayOpExpr))
883 {
884 /* Use node specific selectivity calculation function */
886 (ScalarArrayOpExpr *) clause,
887 treat_as_join_clause(root, clause, rinfo,
888 varRelid, sjinfo),
889 varRelid,
890 jointype,
891 sjinfo);
892 }
893 else if (IsA(clause, RowCompareExpr))
894 {
895 /* Use node specific selectivity calculation function */
897 (RowCompareExpr *) clause,
898 varRelid,
899 jointype,
900 sjinfo);
901 }
902 else if (IsA(clause, NullTest))
903 {
904 /* Use node specific selectivity calculation function */
906 ((NullTest *) clause)->nulltesttype,
907 (Node *) ((NullTest *) clause)->arg,
908 varRelid,
909 jointype,
910 sjinfo);
911 }
912 else if (IsA(clause, BooleanTest))
913 {
914 /* Use node specific selectivity calculation function */
916 ((BooleanTest *) clause)->booltesttype,
917 (Node *) ((BooleanTest *) clause)->arg,
918 varRelid,
919 jointype,
920 sjinfo);
921 }
922 else if (IsA(clause, CurrentOfExpr))
923 {
924 /* CURRENT OF selects at most one row of its table */
925 CurrentOfExpr *cexpr = (CurrentOfExpr *) clause;
926 RelOptInfo *crel = find_base_rel(root, cexpr->cvarno);
927
928 if (crel->tuples > 0)
929 s1 = 1.0 / crel->tuples;
930 }
931 else if (IsA(clause, RelabelType))
932 {
933 /* Not sure this case is needed, but it can't hurt */
935 (Node *) ((RelabelType *) clause)->arg,
936 varRelid,
937 jointype,
938 sjinfo,
939 use_extended_stats);
940 }
941 else if (IsA(clause, CoerceToDomain))
942 {
943 /* Not sure this case is needed, but it can't hurt */
945 (Node *) ((CoerceToDomain *) clause)->arg,
946 varRelid,
947 jointype,
948 sjinfo,
949 use_extended_stats);
950 }
951 else
952 {
953 /*
954 * For anything else, see if we can consider it as a boolean variable.
955 * This only works if it's an immutable expression in Vars of a single
956 * relation; but there's no point in us checking that here because
957 * boolvarsel() will do it internally, and return a suitable default
958 * selectivity if not.
959 */
960 s1 = boolvarsel(root, clause, varRelid);
961 }
962
963 /* Cache the result if possible */
964 if (cacheable)
965 {
966 if (jointype == JOIN_INNER)
967 rinfo->norm_selec = s1;
968 else
969 rinfo->outer_selec = s1;
970 }
971
972#ifdef SELECTIVITY_DEBUG
973 elog(DEBUG4, "clause_selectivity: s1 %f", s1);
974#endif /* SELECTIVITY_DEBUG */
975
976 return s1;
977}
bool bms_is_member(int x, const Bitmapset *a)
Definition: bitmapset.c:510
Node * estimate_expression_value(PlannerInfo *root, Node *node)
Definition: clauses.c:2403
Selectivity clauselist_selectivity_ext(PlannerInfo *root, List *clauses, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, bool use_extended_stats)
Definition: clausesel.c:117
static bool treat_as_join_clause(PlannerInfo *root, Node *clause, RestrictInfo *rinfo, int varRelid, SpecialJoinInfo *sjinfo)
Definition: clausesel.c:586
static Selectivity clauselist_selectivity_or(PlannerInfo *root, List *clauses, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, bool use_extended_stats)
Definition: clausesel.c:359
#define elog(elevel,...)
Definition: elog.h:226
#define DEBUG4
Definition: elog.h:27
static bool is_andclause(const void *clause)
Definition: nodeFuncs.h:107
static bool is_orclause(const void *clause)
Definition: nodeFuncs.h:116
static bool is_opclause(const void *clause)
Definition: nodeFuncs.h:76
static bool is_funcclause(const void *clause)
Definition: nodeFuncs.h:69
static bool is_notclause(const void *clause)
Definition: nodeFuncs.h:125
static Expr * get_notclausearg(const void *notclause)
Definition: nodeFuncs.h:134
double Selectivity
Definition: nodes.h:260
@ JOIN_INNER
Definition: nodes.h:303
void * arg
Selectivity restriction_selectivity(PlannerInfo *root, Oid operatorid, List *args, Oid inputcollid, int varRelid)
Definition: plancat.c:2073
Selectivity join_selectivity(PlannerInfo *root, Oid operatorid, List *args, Oid inputcollid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: plancat.c:2112
Selectivity function_selectivity(PlannerInfo *root, Oid funcid, List *args, Oid inputcollid, bool is_join, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: plancat.c:2152
static bool DatumGetBool(Datum X)
Definition: postgres.h:100
unsigned int Oid
Definition: postgres_ext.h:32
char * s1
RelOptInfo * find_base_rel(PlannerInfo *root, int relid)
Definition: relnode.c:416
Selectivity booltestsel(PlannerInfo *root, BoolTestType booltesttype, Node *arg, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: selfuncs.c:1555
Selectivity nulltestsel(PlannerInfo *root, NullTestType nulltesttype, Node *arg, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: selfuncs.c:1713
Selectivity boolvarsel(PlannerInfo *root, Node *arg, int varRelid)
Definition: selfuncs.c:1516
Selectivity scalararraysel(PlannerInfo *root, ScalarArrayOpExpr *clause, bool is_join_clause, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: selfuncs.c:1831
Selectivity rowcomparesel(PlannerInfo *root, RowCompareExpr *clause, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo)
Definition: selfuncs.c:2220
Oid funcid
Definition: primnodes.h:769
List * args
Definition: primnodes.h:787
Definition: nodes.h:135
Oid opno
Definition: primnodes.h:837
List * args
Definition: primnodes.h:855
Cardinality tuples
Definition: pathnodes.h:981
Expr * clause
Definition: pathnodes.h:2704
Definition: primnodes.h:262
int varno
Definition: primnodes.h:269
Index varlevelsup
Definition: primnodes.h:294

References arg, generate_unaccent_rules::args, FuncExpr::args, OpExpr::args, bms_is_member(), booltestsel(), boolvarsel(), RestrictInfo::clause, clause_selectivity_ext(), clauselist_selectivity_ext(), clauselist_selectivity_or(), CurrentOfExpr::cvarno, DatumGetBool(), DEBUG4, elog, estimate_expression_value(), find_base_rel(), FuncExpr::funcid, function_selectivity(), get_notclausearg(), is_andclause(), is_funcclause(), is_notclause(), is_opclause(), is_orclause(), IsA, JOIN_INNER, join_selectivity(), nulltestsel(), OpExpr::opno, restriction_selectivity(), root, rowcomparesel(), s1, scalararraysel(), treat_as_join_clause(), RelOptInfo::tuples, and RangeQueryClause::var.

Referenced by clause_selectivity(), clause_selectivity_ext(), clauselist_selectivity_ext(), clauselist_selectivity_or(), and statext_mcv_clauselist_selectivity().

◆ clauselist_selectivity()

◆ clauselist_selectivity_ext()

Selectivity clauselist_selectivity_ext ( PlannerInfo root,
List clauses,
int  varRelid,
JoinType  jointype,
SpecialJoinInfo sjinfo,
bool  use_extended_stats 
)

Definition at line 117 of file clausesel.c.

123{
124 Selectivity s1 = 1.0;
125 RelOptInfo *rel;
126 Bitmapset *estimatedclauses = NULL;
127 RangeQueryClause *rqlist = NULL;
128 ListCell *l;
129 int listidx;
130
131 /*
132 * If there's exactly one clause, just go directly to
133 * clause_selectivity_ext(). None of what we might do below is relevant.
134 */
135 if (list_length(clauses) == 1)
136 return clause_selectivity_ext(root, (Node *) linitial(clauses),
137 varRelid, jointype, sjinfo,
138 use_extended_stats);
139
140 /*
141 * Determine if these clauses reference a single relation. If so, and if
142 * it has extended statistics, try to apply those.
143 */
144 rel = find_single_rel_for_clauses(root, clauses);
145 if (use_extended_stats && rel && rel->rtekind == RTE_RELATION && rel->statlist != NIL)
146 {
147 /*
148 * Estimate as many clauses as possible using extended statistics.
149 *
150 * 'estimatedclauses' is populated with the 0-based list position
151 * index of clauses estimated here, and that should be ignored below.
152 */
153 s1 = statext_clauselist_selectivity(root, clauses, varRelid,
154 jointype, sjinfo, rel,
155 &estimatedclauses, false);
156 }
157
158 /*
159 * Apply normal selectivity estimates for remaining clauses. We'll be
160 * careful to skip any clauses which were already estimated above.
161 *
162 * Anything that doesn't look like a potential rangequery clause gets
163 * multiplied into s1 and forgotten. Anything that does gets inserted into
164 * an rqlist entry.
165 */
166 listidx = -1;
167 foreach(l, clauses)
168 {
169 Node *clause = (Node *) lfirst(l);
170 RestrictInfo *rinfo;
172
173 listidx++;
174
175 /*
176 * Skip this clause if it's already been estimated by some other
177 * statistics above.
178 */
179 if (bms_is_member(listidx, estimatedclauses))
180 continue;
181
182 /* Compute the selectivity of this clause in isolation */
183 s2 = clause_selectivity_ext(root, clause, varRelid, jointype, sjinfo,
184 use_extended_stats);
185
186 /*
187 * Check for being passed a RestrictInfo.
188 *
189 * If it's a pseudoconstant RestrictInfo, then s2 is either 1.0 or
190 * 0.0; just use that rather than looking for range pairs.
191 */
192 if (IsA(clause, RestrictInfo))
193 {
194 rinfo = (RestrictInfo *) clause;
195 if (rinfo->pseudoconstant)
196 {
197 s1 = s1 * s2;
198 continue;
199 }
200 clause = (Node *) rinfo->clause;
201 }
202 else
203 rinfo = NULL;
204
205 /*
206 * See if it looks like a restriction clause with a pseudoconstant on
207 * one side. (Anything more complicated than that might not behave in
208 * the simple way we are expecting.) Most of the tests here can be
209 * done more efficiently with rinfo than without.
210 */
211 if (is_opclause(clause) && list_length(((OpExpr *) clause)->args) == 2)
212 {
213 OpExpr *expr = (OpExpr *) clause;
214 bool varonleft = true;
215 bool ok;
216
217 if (rinfo)
218 {
219 ok = (rinfo->num_base_rels == 1) &&
221 rinfo->right_relids) ||
222 (varonleft = false,
224 rinfo->left_relids)));
225 }
226 else
227 {
228 ok = (NumRelids(root, clause) == 1) &&
230 (varonleft = false,
232 }
233
234 if (ok)
235 {
236 /*
237 * If it's not a "<"/"<="/">"/">=" operator, just merge the
238 * selectivity in generically. But if it's the right oprrest,
239 * add the clause to rqlist for later processing.
240 */
241 switch (get_oprrest(expr->opno))
242 {
243 case F_SCALARLTSEL:
244 case F_SCALARLESEL:
245 addRangeClause(&rqlist, clause,
246 varonleft, true, s2);
247 break;
248 case F_SCALARGTSEL:
249 case F_SCALARGESEL:
250 addRangeClause(&rqlist, clause,
251 varonleft, false, s2);
252 break;
253 default:
254 /* Just merge the selectivity in generically */
255 s1 = s1 * s2;
256 break;
257 }
258 continue; /* drop to loop bottom */
259 }
260 }
261
262 /* Not the right form, so treat it generically. */
263 s1 = s1 * s2;
264 }
265
266 /*
267 * Now scan the rangequery pair list.
268 */
269 while (rqlist != NULL)
270 {
271 RangeQueryClause *rqnext;
272
273 if (rqlist->have_lobound && rqlist->have_hibound)
274 {
275 /* Successfully matched a pair of range clauses */
277
278 /*
279 * Exact equality to the default value probably means the
280 * selectivity function punted. This is not airtight but should
281 * be good enough.
282 */
283 if (rqlist->hibound == DEFAULT_INEQ_SEL ||
284 rqlist->lobound == DEFAULT_INEQ_SEL)
285 {
287 }
288 else
289 {
290 s2 = rqlist->hibound + rqlist->lobound - 1.0;
291
292 /* Adjust for double-exclusion of NULLs */
293 s2 += nulltestsel(root, IS_NULL, rqlist->var,
294 varRelid, jointype, sjinfo);
295
296 /*
297 * A zero or slightly negative s2 should be converted into a
298 * small positive value; we probably are dealing with a very
299 * tight range and got a bogus result due to roundoff errors.
300 * However, if s2 is very negative, then we probably have
301 * default selectivity estimates on one or both sides of the
302 * range that we failed to recognize above for some reason.
303 */
304 if (s2 <= 0.0)
305 {
306 if (s2 < -0.01)
307 {
308 /*
309 * No data available --- use a default estimate that
310 * is small, but not real small.
311 */
313 }
314 else
315 {
316 /*
317 * It's just roundoff error; use a small positive
318 * value
319 */
320 s2 = 1.0e-10;
321 }
322 }
323 }
324 /* Merge in the selectivity of the pair of clauses */
325 s1 *= s2;
326 }
327 else
328 {
329 /* Only found one of a pair, merge it in generically */
330 if (rqlist->have_lobound)
331 s1 *= rqlist->lobound;
332 else
333 s1 *= rqlist->hibound;
334 }
335 /* release storage and advance */
336 rqnext = rqlist->next;
337 pfree(rqlist);
338 rqlist = rqnext;
339 }
340
341 return s1;
342}
int NumRelids(PlannerInfo *root, Node *clause)
Definition: clauses.c:2137
bool is_pseudo_constant_clause(Node *clause)
Definition: clauses.c:2095
bool is_pseudo_constant_clause_relids(Node *clause, Relids relids)
Definition: clauses.c:2115
static void addRangeClause(RangeQueryClause **rqlist, Node *clause, bool varonleft, bool isLTsel, Selectivity s2)
Definition: clausesel.c:427
static RelOptInfo * find_single_rel_for_clauses(PlannerInfo *root, List *clauses)
Definition: clausesel.c:523
Selectivity statext_clauselist_selectivity(PlannerInfo *root, List *clauses, int varRelid, JoinType jointype, SpecialJoinInfo *sjinfo, RelOptInfo *rel, Bitmapset **estimatedclauses, bool is_or)
RegProcedure get_oprrest(Oid opno)
Definition: lsyscache.c:1724
void pfree(void *pointer)
Definition: mcxt.c:1594
@ RTE_RELATION
Definition: parsenodes.h:1042
#define lfirst(lc)
Definition: pg_list.h:172
static int list_length(const List *l)
Definition: pg_list.h:152
#define NIL
Definition: pg_list.h:68
#define linitial(l)
Definition: pg_list.h:178
#define lsecond(l)
Definition: pg_list.h:183
char * s2
@ IS_NULL
Definition: primnodes.h:1963
#define DEFAULT_RANGE_INEQ_SEL
Definition: selfuncs.h:40
#define DEFAULT_INEQ_SEL
Definition: selfuncs.h:37
Selectivity hibound
Definition: clausesel.c:38
Selectivity lobound
Definition: clausesel.c:37
struct RangeQueryClause * next
Definition: clausesel.c:33
List * statlist
Definition: pathnodes.h:978
RTEKind rtekind
Definition: pathnodes.h:958

References addRangeClause(), generate_unaccent_rules::args, OpExpr::args, bms_is_member(), RestrictInfo::clause, clause_selectivity_ext(), DEFAULT_INEQ_SEL, DEFAULT_RANGE_INEQ_SEL, find_single_rel_for_clauses(), get_oprrest(), RangeQueryClause::have_hibound, RangeQueryClause::have_lobound, RangeQueryClause::hibound, IS_NULL, is_opclause(), is_pseudo_constant_clause(), is_pseudo_constant_clause_relids(), IsA, lfirst, linitial, list_length(), RangeQueryClause::lobound, lsecond, RangeQueryClause::next, NIL, nulltestsel(), NumRelids(), OpExpr::opno, pfree(), root, RTE_RELATION, RelOptInfo::rtekind, s1, s2, statext_clauselist_selectivity(), RelOptInfo::statlist, and RangeQueryClause::var.

Referenced by clause_selectivity_ext(), clauselist_apply_dependencies(), clauselist_selectivity(), and statext_mcv_clauselist_selectivity().

◆ contain_mutable_functions()

◆ contain_mutable_functions_after_planning()

bool contain_mutable_functions_after_planning ( Expr expr)

Definition at line 494 of file clauses.c.

495{
496 /* We assume here that expression_planner() won't scribble on its input */
497 expr = expression_planner(expr);
498
499 /* Now we can search for non-immutable functions */
500 return contain_mutable_functions((Node *) expr);
501}
bool contain_mutable_functions(Node *clause)
Definition: clauses.c:374
Expr * expression_planner(Expr *expr)
Definition: planner.c:6719

References contain_mutable_functions(), and expression_planner().

Referenced by CheckPredicate(), ComputeIndexAttrs(), and cookDefault().

◆ contain_var_clause()

◆ contain_vars_of_level()

bool contain_vars_of_level ( Node node,
int  levelsup 
)

◆ contain_vars_returning_old_or_new()

bool contain_vars_returning_old_or_new ( Node node)

Definition at line 511 of file var.c.

512{
514}
static bool contain_vars_returning_old_or_new_walker(Node *node, void *context)
Definition: var.c:517

References contain_vars_returning_old_or_new_walker().

Referenced by make_modifytable().

◆ contain_volatile_functions()

bool contain_volatile_functions ( Node clause)

Definition at line 542 of file clauses.c.

543{
544 return contain_volatile_functions_walker(clause, NULL);
545}
static bool contain_volatile_functions_walker(Node *node, void *context)
Definition: clauses.c:554

References contain_volatile_functions_walker().

Referenced by apply_child_basequals(), ATExecAddColumn(), check_hashjoinable(), check_mergejoinable(), check_output_expressions(), compute_semijoin_info(), contain_volatile_functions_after_planning(), convert_ANY_sublink_to_join(), convert_EXISTS_sublink_to_join(), convert_EXISTS_to_ANY(), convert_VALUES_to_ANY(), CopyFrom(), distribute_qual_to_rels(), estimate_num_groups(), ExecInitWindowAgg(), expand_indexqual_rowcompare(), find_compatible_agg(), find_simplified_clause(), get_eclass_for_sort_expr(), get_memoize_path(), group_similar_or_args(), initialize_peragg(), inline_function(), inline_set_returning_function(), is_pseudo_constant_clause(), is_pseudo_constant_clause_relids(), is_pseudo_constant_for_index(), is_safe_restriction_clause_for(), is_simple_subquery(), is_simple_values(), IsBinaryTidClause(), IsTidEqualAnyClause(), make_sort_input_target(), mark_nullable_by_grouping(), match_clause_to_ordering_op(), match_clause_to_partition_key(), match_opclause_to_indexcol(), match_orclause_to_indexcol(), match_rowcompare_to_indexcol(), match_saopclause_to_indexcol(), paraminfo_get_equal_hashops(), qual_is_pushdown_safe(), remove_unused_subquery_outputs(), SS_process_ctes(), and subquery_planner().

◆ contain_volatile_functions_after_planning()

bool contain_volatile_functions_after_planning ( Expr expr)

Definition at line 663 of file clauses.c.

664{
665 /* We assume here that expression_planner() won't scribble on its input */
666 expr = expression_planner(expr);
667
668 /* Now we can search for volatile functions */
669 return contain_volatile_functions((Node *) expr);
670}
bool contain_volatile_functions(Node *clause)
Definition: clauses.c:542

References contain_volatile_functions(), and expression_planner().

◆ contain_volatile_functions_not_nextval()

bool contain_volatile_functions_not_nextval ( Node clause)

Definition at line 677 of file clauses.c.

678{
680}
static bool contain_volatile_functions_not_nextval_walker(Node *node, void *context)
Definition: clauses.c:690

References contain_volatile_functions_not_nextval_walker().

Referenced by BeginCopyFrom().

◆ convert_saop_to_hashed_saop()

void convert_saop_to_hashed_saop ( Node node)

Definition at line 2295 of file clauses.c.

2296{
2297 (void) convert_saop_to_hashed_saop_walker(node, NULL);
2298}
static bool convert_saop_to_hashed_saop_walker(Node *node, void *context)
Definition: clauses.c:2301

References convert_saop_to_hashed_saop_walker().

Referenced by preprocess_expression().

◆ count_nonjunk_tlist_entries()

int count_nonjunk_tlist_entries ( List tlist)

Definition at line 186 of file tlist.c.

187{
188 int len = 0;
189 ListCell *l;
190
191 foreach(l, tlist)
192 {
193 TargetEntry *tle = (TargetEntry *) lfirst(l);
194
195 if (!tle->resjunk)
196 len++;
197 }
198 return len;
199}
const void size_t len

References len, and lfirst.

Referenced by get_update_query_targetlist_def(), transformJsonArrayQueryConstructor(), transformMultiAssignRef(), and transformSubLink().

◆ estimate_expression_value()

Node * estimate_expression_value ( PlannerInfo root,
Node node 
)

Definition at line 2403 of file clauses.c.

2404{
2406
2407 context.boundParams = root->glob->boundParams; /* bound Params */
2408 /* we do not need to mark the plan as depending on inlined functions */
2409 context.root = NULL;
2410 context.active_fns = NIL; /* nothing being recursively simplified */
2411 context.case_val = NULL; /* no CASE being examined */
2412 context.estimate = true; /* unsafe transformations OK */
2413 return eval_const_expressions_mutator(node, &context);
2414}
static Node * eval_const_expressions_mutator(Node *node, eval_const_expressions_context *context)
Definition: clauses.c:2448
ParamListInfo boundParams
Definition: clauses.c:66

References eval_const_expressions_context::active_fns, eval_const_expressions_context::boundParams, eval_const_expressions_context::case_val, eval_const_expressions_context::estimate, eval_const_expressions_mutator(), NIL, eval_const_expressions_context::root, and root.

Referenced by array_unnest_support(), bernoulli_samplescangetsamplesize(), clause_selectivity_ext(), generate_series_int4_support(), generate_series_int8_support(), generate_series_numeric_support(), generate_series_timestamp_support(), get_restriction_variable(), gincost_opexpr(), gincost_scalararrayopexpr(), preprocess_limit(), scalararraysel(), system_rows_samplescangetsamplesize(), system_samplescangetsamplesize(), and system_time_samplescangetsamplesize().

◆ eval_const_expressions()

◆ evaluate_expr()

Expr * evaluate_expr ( Expr expr,
Oid  result_type,
int32  result_typmod,
Oid  result_collation 
)

Definition at line 5075 of file clauses.c.

5077{
5078 EState *estate;
5079 ExprState *exprstate;
5080 MemoryContext oldcontext;
5081 Datum const_val;
5082 bool const_is_null;
5083 int16 resultTypLen;
5084 bool resultTypByVal;
5085
5086 /*
5087 * To use the executor, we need an EState.
5088 */
5089 estate = CreateExecutorState();
5090
5091 /* We can use the estate's working context to avoid memory leaks. */
5092 oldcontext = MemoryContextSwitchTo(estate->es_query_cxt);
5093
5094 /* Make sure any opfuncids are filled in. */
5095 fix_opfuncids((Node *) expr);
5096
5097 /*
5098 * Prepare expr for execution. (Note: we can't use ExecPrepareExpr
5099 * because it'd result in recursively invoking eval_const_expressions.)
5100 */
5101 exprstate = ExecInitExpr(expr, NULL);
5102
5103 /*
5104 * And evaluate it.
5105 *
5106 * It is OK to use a default econtext because none of the ExecEvalExpr()
5107 * code used in this situation will use econtext. That might seem
5108 * fortuitous, but it's not so unreasonable --- a constant expression does
5109 * not depend on context, by definition, n'est ce pas?
5110 */
5111 const_val = ExecEvalExprSwitchContext(exprstate,
5112 GetPerTupleExprContext(estate),
5113 &const_is_null);
5114
5115 /* Get info needed about result datatype */
5116 get_typlenbyval(result_type, &resultTypLen, &resultTypByVal);
5117
5118 /* Get back to outer memory context */
5119 MemoryContextSwitchTo(oldcontext);
5120
5121 /*
5122 * Must copy result out of sub-context used by expression eval.
5123 *
5124 * Also, if it's varlena, forcibly detoast it. This protects us against
5125 * storing TOAST pointers into plans that might outlive the referenced
5126 * data. (makeConst would handle detoasting anyway, but it's worth a few
5127 * extra lines here so that we can do the copy and detoast in one step.)
5128 */
5129 if (!const_is_null)
5130 {
5131 if (resultTypLen == -1)
5132 const_val = PointerGetDatum(PG_DETOAST_DATUM_COPY(const_val));
5133 else
5134 const_val = datumCopy(const_val, resultTypByVal, resultTypLen);
5135 }
5136
5137 /* Release all the junk we just created */
5138 FreeExecutorState(estate);
5139
5140 /*
5141 * Make the constant result node.
5142 */
5143 return (Expr *) makeConst(result_type, result_typmod, result_collation,
5144 resultTypLen,
5145 const_val, const_is_null,
5146 resultTypByVal);
5147}
int16_t int16
Definition: c.h:534
Datum datumCopy(Datum value, bool typByVal, int typLen)
Definition: datum.c:132
ExprState * ExecInitExpr(Expr *node, PlanState *parent)
Definition: execExpr.c:143
void FreeExecutorState(EState *estate)
Definition: execUtils.c:192
EState * CreateExecutorState(void)
Definition: execUtils.c:88
#define GetPerTupleExprContext(estate)
Definition: executor.h:653
static Datum ExecEvalExprSwitchContext(ExprState *state, ExprContext *econtext, bool *isNull)
Definition: executor.h:433
#define PG_DETOAST_DATUM_COPY(datum)
Definition: fmgr.h:242
void get_typlenbyval(Oid typid, int16 *typlen, bool *typbyval)
Definition: lsyscache.c:2418
Const * makeConst(Oid consttype, int32 consttypmod, Oid constcollid, int constlen, Datum constvalue, bool constisnull, bool constbyval)
Definition: makefuncs.c:350
void fix_opfuncids(Node *node)
Definition: nodeFuncs.c:1841
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
Definition: palloc.h:124
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:332
uint64_t Datum
Definition: postgres.h:70
MemoryContext es_query_cxt
Definition: execnodes.h:710

References CreateExecutorState(), datumCopy(), EState::es_query_cxt, ExecEvalExprSwitchContext(), ExecInitExpr(), fix_opfuncids(), FreeExecutorState(), get_typlenbyval(), GetPerTupleExprContext, makeConst(), MemoryContextSwitchTo(), PG_DETOAST_DATUM_COPY, and PointerGetDatum().

Referenced by eval_const_expressions_mutator(), evaluate_function(), and transformPartitionBoundValue().

◆ expand_function_arguments()

List * expand_function_arguments ( List args,
bool  include_out_arguments,
Oid  result_type,
HeapTuple  func_tuple 
)

Definition at line 4278 of file clauses.c.

4280{
4281 Form_pg_proc funcform = (Form_pg_proc) GETSTRUCT(func_tuple);
4282 Oid *proargtypes = funcform->proargtypes.values;
4283 int pronargs = funcform->pronargs;
4284 bool has_named_args = false;
4285 ListCell *lc;
4286
4287 /*
4288 * If we are asked to match to OUT arguments, then use the proallargtypes
4289 * array (which includes those); otherwise use proargtypes (which
4290 * doesn't). Of course, if proallargtypes is null, we always use
4291 * proargtypes. (Fetching proallargtypes is annoyingly expensive
4292 * considering that we may have nothing to do here, but fortunately the
4293 * common case is include_out_arguments == false.)
4294 */
4295 if (include_out_arguments)
4296 {
4297 Datum proallargtypes;
4298 bool isNull;
4299
4300 proallargtypes = SysCacheGetAttr(PROCOID, func_tuple,
4301 Anum_pg_proc_proallargtypes,
4302 &isNull);
4303 if (!isNull)
4304 {
4305 ArrayType *arr = DatumGetArrayTypeP(proallargtypes);
4306
4307 pronargs = ARR_DIMS(arr)[0];
4308 if (ARR_NDIM(arr) != 1 ||
4309 pronargs < 0 ||
4310 ARR_HASNULL(arr) ||
4311 ARR_ELEMTYPE(arr) != OIDOID)
4312 elog(ERROR, "proallargtypes is not a 1-D Oid array or it contains nulls");
4313 Assert(pronargs >= funcform->pronargs);
4314 proargtypes = (Oid *) ARR_DATA_PTR(arr);
4315 }
4316 }
4317
4318 /* Do we have any named arguments? */
4319 foreach(lc, args)
4320 {
4321 Node *arg = (Node *) lfirst(lc);
4322
4323 if (IsA(arg, NamedArgExpr))
4324 {
4325 has_named_args = true;
4326 break;
4327 }
4328 }
4329
4330 /* If so, we must apply reorder_function_arguments */
4331 if (has_named_args)
4332 {
4334 /* Recheck argument types and add casts if needed */
4335 recheck_cast_function_args(args, result_type,
4336 proargtypes, pronargs,
4337 func_tuple);
4338 }
4339 else if (list_length(args) < pronargs)
4340 {
4341 /* No named args, but we seem to be short some defaults */
4342 args = add_function_defaults(args, pronargs, func_tuple);
4343 /* Recheck argument types and add casts if needed */
4344 recheck_cast_function_args(args, result_type,
4345 proargtypes, pronargs,
4346 func_tuple);
4347 }
4348
4349 return args;
4350}
#define ARR_NDIM(a)
Definition: array.h:290
#define ARR_DATA_PTR(a)
Definition: array.h:322
#define DatumGetArrayTypeP(X)
Definition: array.h:261
#define ARR_ELEMTYPE(a)
Definition: array.h:292
#define ARR_DIMS(a)
Definition: array.h:294
#define ARR_HASNULL(a)
Definition: array.h:291
static List * add_function_defaults(List *args, int pronargs, HeapTuple func_tuple)
Definition: clauses.c:4429
static List * reorder_function_arguments(List *args, int pronargs, HeapTuple func_tuple)
Definition: clauses.c:4359
static void recheck_cast_function_args(List *args, Oid result_type, Oid *proargtypes, int pronargs, HeapTuple func_tuple)
Definition: clauses.c:4483
#define ERROR
Definition: elog.h:39
static void * GETSTRUCT(const HeapTupleData *tuple)
Definition: htup_details.h:728
FormData_pg_proc * Form_pg_proc
Definition: pg_proc.h:136
int16 pronargs
Definition: pg_proc.h:81
Datum SysCacheGetAttr(int cacheId, HeapTuple tup, AttrNumber attributeNumber, bool *isNull)
Definition: syscache.c:595

References add_function_defaults(), arg, generate_unaccent_rules::args, ARR_DATA_PTR, ARR_DIMS, ARR_ELEMTYPE, ARR_HASNULL, ARR_NDIM, Assert(), DatumGetArrayTypeP, elog, ERROR, GETSTRUCT(), IsA, lfirst, list_length(), pronargs, recheck_cast_function_args(), reorder_function_arguments(), and SysCacheGetAttr().

Referenced by eval_const_expressions_mutator(), simplify_function(), and transformCallStmt().

◆ expression_planner()

Expr * expression_planner ( Expr expr)

Definition at line 6719 of file planner.c.

6720{
6721 Node *result;
6722
6723 /*
6724 * Convert named-argument function calls, insert default arguments and
6725 * simplify constant subexprs
6726 */
6727 result = eval_const_expressions(NULL, (Node *) expr);
6728
6729 /* Fill in opfuncid values if missing */
6730 fix_opfuncids(result);
6731
6732 return (Expr *) result;
6733}
Node * eval_const_expressions(PlannerInfo *root, Node *node)
Definition: clauses.c:2262

References eval_const_expressions(), and fix_opfuncids().

Referenced by ATExecAddColumn(), ATExecSetExpression(), ATPrepAlterColumnType(), BeginCopyFrom(), ComputePartitionAttrs(), contain_mutable_functions_after_planning(), contain_volatile_functions_after_planning(), ExecPrepareCheck(), ExecPrepareExpr(), ExecPrepareQual(), load_domaintype_info(), set_baserel_partition_constraint(), slot_fill_defaults(), and transformPartitionBoundValue().

◆ expression_planner_with_deps()

Expr * expression_planner_with_deps ( Expr expr,
List **  relationOids,
List **  invalItems 
)

Definition at line 6746 of file planner.c.

6749{
6750 Node *result;
6751 PlannerGlobal glob;
6753
6754 /* Make up dummy planner state so we can use setrefs machinery */
6755 MemSet(&glob, 0, sizeof(glob));
6756 glob.type = T_PlannerGlobal;
6757 glob.relationOids = NIL;
6758 glob.invalItems = NIL;
6759
6760 MemSet(&root, 0, sizeof(root));
6761 root.type = T_PlannerInfo;
6762 root.glob = &glob;
6763
6764 /*
6765 * Convert named-argument function calls, insert default arguments and
6766 * simplify constant subexprs. Collect identities of inlined functions
6767 * and elided domains, too.
6768 */
6769 result = eval_const_expressions(&root, (Node *) expr);
6770
6771 /* Fill in opfuncid values if missing */
6772 fix_opfuncids(result);
6773
6774 /*
6775 * Now walk the finished expression to find anything else we ought to
6776 * record as an expression dependency.
6777 */
6778 (void) extract_query_dependencies_walker(result, &root);
6779
6780 *relationOids = glob.relationOids;
6781 *invalItems = glob.invalItems;
6782
6783 return (Expr *) result;
6784}
#define MemSet(start, val, len)
Definition: c.h:1020
bool extract_query_dependencies_walker(Node *node, PlannerInfo *context)
Definition: setrefs.c:3673
List * invalItems
Definition: pathnodes.h:151
List * relationOids
Definition: pathnodes.h:148

References eval_const_expressions(), extract_query_dependencies_walker(), fix_opfuncids(), PlannerGlobal::invalItems, MemSet, NIL, PlannerGlobal::relationOids, and root.

Referenced by GetCachedExpression().

◆ extract_query_dependencies()

void extract_query_dependencies ( Node query,
List **  relationOids,
List **  invalItems,
bool *  hasRowSecurity 
)

Definition at line 3637 of file setrefs.c.

3641{
3642 PlannerGlobal glob;
3644
3645 /* Make up dummy planner state so we can use this module's machinery */
3646 MemSet(&glob, 0, sizeof(glob));
3647 glob.type = T_PlannerGlobal;
3648 glob.relationOids = NIL;
3649 glob.invalItems = NIL;
3650 /* Hack: we use glob.dependsOnRole to collect hasRowSecurity flags */
3651 glob.dependsOnRole = false;
3652
3653 MemSet(&root, 0, sizeof(root));
3654 root.type = T_PlannerInfo;
3655 root.glob = &glob;
3656
3658
3659 *relationOids = glob.relationOids;
3660 *invalItems = glob.invalItems;
3661 *hasRowSecurity = glob.dependsOnRole;
3662}
bool dependsOnRole
Definition: pathnodes.h:169

References PlannerGlobal::dependsOnRole, extract_query_dependencies_walker(), PlannerGlobal::invalItems, MemSet, NIL, PlannerGlobal::relationOids, and root.

Referenced by CompleteCachedPlan(), and RevalidateCachedQuery().

◆ flatten_group_exprs()

Node * flatten_group_exprs ( PlannerInfo root,
Query query,
Node node 
)

Definition at line 968 of file var.c.

969{
971
972 /*
973 * We do not expect this to be applied to the whole Query, only to
974 * expressions or LATERAL subqueries. Hence, if the top node is a Query,
975 * it's okay to immediately increment sublevels_up.
976 */
977 Assert(node != (Node *) query);
978
979 context.root = root;
980 context.query = query;
981 context.sublevels_up = 0;
982 /* flag whether grouping expressions could possibly contain SubLinks */
983 context.possible_sublink = query->hasSubLinks;
984 /* if hasSubLinks is already true, no need to work hard */
985 context.inserted_sublink = query->hasSubLinks;
986
987 return flatten_group_exprs_mutator(node, &context);
988}
PlannerInfo * root
Definition: var.c:66
static Node * flatten_group_exprs_mutator(Node *node, flatten_join_alias_vars_context *context)
Definition: var.c:991

References Assert(), flatten_group_exprs_mutator(), flatten_join_alias_vars_context::inserted_sublink, flatten_join_alias_vars_context::possible_sublink, flatten_join_alias_vars_context::query, flatten_join_alias_vars_context::root, root, and flatten_join_alias_vars_context::sublevels_up.

Referenced by get_query_def(), and subquery_planner().

◆ flatten_join_alias_vars()

Node * flatten_join_alias_vars ( PlannerInfo root,
Query query,
Node node 
)

Definition at line 789 of file var.c.

790{
792
793 /*
794 * We do not expect this to be applied to the whole Query, only to
795 * expressions or LATERAL subqueries. Hence, if the top node is a Query,
796 * it's okay to immediately increment sublevels_up.
797 */
798 Assert(node != (Node *) query);
799
800 context.root = root;
801 context.query = query;
802 context.sublevels_up = 0;
803 /* flag whether join aliases could possibly contain SubLinks */
804 context.possible_sublink = query->hasSubLinks;
805 /* if hasSubLinks is already true, no need to work hard */
806 context.inserted_sublink = query->hasSubLinks;
807
808 return flatten_join_alias_vars_mutator(node, &context);
809}
static Node * flatten_join_alias_vars_mutator(Node *node, flatten_join_alias_vars_context *context)
Definition: var.c:812

References Assert(), flatten_join_alias_vars_mutator(), flatten_join_alias_vars_context::inserted_sublink, flatten_join_alias_vars_context::possible_sublink, flatten_join_alias_vars_context::query, flatten_join_alias_vars_context::root, root, and flatten_join_alias_vars_context::sublevels_up.

Referenced by finalize_grouping_exprs_walker(), parseCheckAggregates(), preprocess_expression(), pull_up_simple_subquery(), and subquery_planner().

◆ get_sortgroupclause_expr()

Node * get_sortgroupclause_expr ( SortGroupClause sgClause,
List targetList 
)

Definition at line 379 of file tlist.c.

380{
381 TargetEntry *tle = get_sortgroupclause_tle(sgClause, targetList);
382
383 return (Node *) tle->expr;
384}
Expr * expr
Definition: primnodes.h:2225
TargetEntry * get_sortgroupclause_tle(SortGroupClause *sgClause, List *targetList)
Definition: tlist.c:367

References TargetEntry::expr, and get_sortgroupclause_tle().

Referenced by get_sortgrouplist_exprs(), make_pathkeys_for_sortclauses_extended(), transformAggregateCall(), and transformWindowDefinitions().

◆ get_sortgroupclause_tle()

◆ get_sortgrouplist_exprs()

List * get_sortgrouplist_exprs ( List sgClauses,
List targetList 
)

Definition at line 392 of file tlist.c.

393{
394 List *result = NIL;
395 ListCell *l;
396
397 foreach(l, sgClauses)
398 {
399 SortGroupClause *sortcl = (SortGroupClause *) lfirst(l);
400 Node *sortexpr;
401
402 sortexpr = get_sortgroupclause_expr(sortcl, targetList);
403 result = lappend(result, sortexpr);
404 }
405 return result;
406}
List * lappend(List *list, void *datum)
Definition: list.c:339
Node * get_sortgroupclause_expr(SortGroupClause *sgClause, List *targetList)
Definition: tlist.c:379

References get_sortgroupclause_expr(), lappend(), lfirst, and NIL.

Referenced by create_final_distinct_paths(), create_partial_distinct_paths(), estimate_path_cost_size(), get_number_of_groups(), get_windowclause_startup_tuples(), and group_by_has_partkey().

◆ get_sortgroupref_clause()

SortGroupClause * get_sortgroupref_clause ( Index  sortref,
List clauses 
)

Definition at line 422 of file tlist.c.

423{
424 ListCell *l;
425
426 foreach(l, clauses)
427 {
429
430 if (cl->tleSortGroupRef == sortref)
431 return cl;
432 }
433
434 elog(ERROR, "ORDER/GROUP BY expression not found in list");
435 return NULL; /* keep compiler quiet */
436}

References elog, ERROR, lfirst, and SortGroupClause::tleSortGroupRef.

Referenced by preprocess_groupclause().

◆ get_sortgroupref_clause_noerr()

SortGroupClause * get_sortgroupref_clause_noerr ( Index  sortref,
List clauses 
)

Definition at line 443 of file tlist.c.

444{
445 ListCell *l;
446
447 foreach(l, clauses)
448 {
450
451 if (cl->tleSortGroupRef == sortref)
452 return cl;
453 }
454
455 return NULL;
456}

References lfirst, and SortGroupClause::tleSortGroupRef.

Referenced by find_em_for_rel_target(), foreign_grouping_ok(), group_keys_reorder_by_pathkeys(), make_group_input_target(), and make_partial_grouping_target().

◆ get_sortgroupref_tle()

TargetEntry * get_sortgroupref_tle ( Index  sortref,
List targetList 
)

Definition at line 345 of file tlist.c.

346{
347 ListCell *l;
348
349 foreach(l, targetList)
350 {
351 TargetEntry *tle = (TargetEntry *) lfirst(l);
352
353 if (tle->ressortgroupref == sortref)
354 return tle;
355 }
356
357 elog(ERROR, "ORDER/GROUP BY expression not found in targetlist");
358 return NULL; /* keep compiler quiet */
359}
Index ressortgroupref
Definition: primnodes.h:2231

References elog, ERROR, lfirst, and TargetEntry::ressortgroupref.

Referenced by convert_subquery_pathkeys(), deparseSortGroupClause(), foreign_expr_walker(), get_rule_sortgroupclause(), get_sortgroupclause_tle(), make_unique_from_pathkeys(), prepare_sort_from_pathkeys(), and transformDistinctOnClause().

◆ is_pseudo_constant_for_index()

bool is_pseudo_constant_for_index ( PlannerInfo root,
Node expr,
IndexOptInfo index 
)

Definition at line 4455 of file indxpath.c.

4456{
4457 /* pull_varnos is cheaper than volatility check, so do that first */
4458 if (bms_is_member(index->rel->relid, pull_varnos(root, expr)))
4459 return false; /* no good, contains Var of table */
4461 return false; /* no good, volatile comparison value */
4462 return true;
4463}
Definition: type.h:96
Relids pull_varnos(PlannerInfo *root, Node *node)
Definition: var.c:114

References bms_is_member(), contain_volatile_functions(), pull_varnos(), and root.

◆ locate_var_of_level()

int locate_var_of_level ( Node node,
int  levelsup 
)

Definition at line 555 of file var.c.

556{
558
559 context.var_location = -1; /* in case we find nothing */
560 context.sublevels_up = levelsup;
561
564 &context,
565 0);
566
567 return context.var_location;
568}
static bool locate_var_of_level_walker(Node *node, locate_var_of_level_context *context)
Definition: var.c:571

References locate_var_of_level_walker(), query_or_expression_tree_walker, locate_var_of_level_context::sublevels_up, and locate_var_of_level_context::var_location.

Referenced by check_agg_arguments(), checkExprIsVarFree(), and transformSetOperationTree().

◆ make_SAOP_expr()

ScalarArrayOpExpr * make_SAOP_expr ( Oid  oper,
Node leftexpr,
Oid  coltype,
Oid  arraycollid,
Oid  inputcollid,
List exprs,
bool  haveNonConst 
)

Definition at line 5552 of file clauses.c.

5554{
5555 Node *arrayNode = NULL;
5556 ScalarArrayOpExpr *saopexpr = NULL;
5557 Oid arraytype = get_array_type(coltype);
5558
5559 if (!OidIsValid(arraytype))
5560 return NULL;
5561
5562 /*
5563 * Assemble an array from the list of constants. It seems more profitable
5564 * to build a const array. But in the presence of other nodes, we don't
5565 * have a specific value here and must employ an ArrayExpr instead.
5566 */
5567 if (haveNonConst)
5568 {
5569 ArrayExpr *arrayExpr = makeNode(ArrayExpr);
5570
5571 /* array_collid will be set by parse_collate.c */
5572 arrayExpr->element_typeid = coltype;
5573 arrayExpr->array_typeid = arraytype;
5574 arrayExpr->multidims = false;
5575 arrayExpr->elements = exprs;
5576 arrayExpr->location = -1;
5577
5578 arrayNode = (Node *) arrayExpr;
5579 }
5580 else
5581 {
5582 int16 typlen;
5583 bool typbyval;
5584 char typalign;
5585 Datum *elems;
5586 bool *nulls;
5587 int i = 0;
5588 ArrayType *arrayConst;
5589 int dims[1] = {list_length(exprs)};
5590 int lbs[1] = {1};
5591
5592 get_typlenbyvalalign(coltype, &typlen, &typbyval, &typalign);
5593
5594 elems = (Datum *) palloc(sizeof(Datum) * list_length(exprs));
5595 nulls = (bool *) palloc(sizeof(bool) * list_length(exprs));
5596 foreach_node(Const, value, exprs)
5597 {
5598 elems[i] = value->constvalue;
5599 nulls[i++] = value->constisnull;
5600 }
5601
5602 arrayConst = construct_md_array(elems, nulls, 1, dims, lbs,
5603 coltype, typlen, typbyval, typalign);
5604 arrayNode = (Node *) makeConst(arraytype, -1, arraycollid,
5605 -1, PointerGetDatum(arrayConst),
5606 false, false);
5607
5608 pfree(elems);
5609 pfree(nulls);
5610 list_free(exprs);
5611 }
5612
5613 /* Build the SAOP expression node */
5614 saopexpr = makeNode(ScalarArrayOpExpr);
5615 saopexpr->opno = oper;
5616 saopexpr->opfuncid = get_opcode(oper);
5617 saopexpr->hashfuncid = InvalidOid;
5618 saopexpr->negfuncid = InvalidOid;
5619 saopexpr->useOr = true;
5620 saopexpr->inputcollid = inputcollid;
5621 saopexpr->args = list_make2(leftexpr, arrayNode);
5622 saopexpr->location = -1;
5623
5624 return saopexpr;
5625}
ArrayType * construct_md_array(Datum *elems, bool *nulls, int ndims, int *dims, int *lbs, Oid elmtype, int elmlen, bool elmbyval, char elmalign)
Definition: arrayfuncs.c:3494
#define OidIsValid(objectId)
Definition: c.h:775
static struct @166 value
int i
Definition: isn.c:77
void list_free(List *list)
Definition: list.c:1546
void get_typlenbyvalalign(Oid typid, int16 *typlen, bool *typbyval, char *typalign)
Definition: lsyscache.c:2438
RegProcedure get_opcode(Oid opno)
Definition: lsyscache.c:1452
Oid get_array_type(Oid typid)
Definition: lsyscache.c:2954
void * palloc(Size size)
Definition: mcxt.c:1365
#define makeNode(_type_)
Definition: nodes.h:161
Operator oper(ParseState *pstate, List *opname, Oid ltypeId, Oid rtypeId, bool noError, int location)
Definition: parse_oper.c:371
#define foreach_node(type, var, lst)
Definition: pg_list.h:496
#define list_make2(x1, x2)
Definition: pg_list.h:214
char typalign
Definition: pg_type.h:176
#define InvalidOid
Definition: postgres_ext.h:37
ParseLoc location
Definition: primnodes.h:1407
ParseLoc location
Definition: primnodes.h:938

References ScalarArrayOpExpr::args, construct_md_array(), foreach_node, get_array_type(), get_opcode(), get_typlenbyvalalign(), i, InvalidOid, list_free(), list_length(), list_make2, ScalarArrayOpExpr::location, ArrayExpr::location, makeConst(), makeNode, OidIsValid, oper(), ScalarArrayOpExpr::opno, palloc(), pfree(), PointerGetDatum(), typalign, ScalarArrayOpExpr::useOr, and value.

Referenced by convert_VALUES_to_ANY(), and match_orclause_to_indexcol().

◆ negate_clause()

Node * negate_clause ( Node node)

Definition at line 73 of file prepqual.c.

74{
75 if (node == NULL) /* should not happen */
76 elog(ERROR, "can't negate an empty subexpression");
77 switch (nodeTag(node))
78 {
79 case T_Const:
80 {
81 Const *c = (Const *) node;
82
83 /* NOT NULL is still NULL */
84 if (c->constisnull)
85 return makeBoolConst(false, true);
86 /* otherwise pretty easy */
87 return makeBoolConst(!DatumGetBool(c->constvalue), false);
88 }
89 break;
90 case T_OpExpr:
91 {
92 /*
93 * Negate operator if possible: (NOT (< A B)) => (>= A B)
94 */
95 OpExpr *opexpr = (OpExpr *) node;
96 Oid negator = get_negator(opexpr->opno);
97
98 if (negator)
99 {
100 OpExpr *newopexpr = makeNode(OpExpr);
101
102 newopexpr->opno = negator;
103 newopexpr->opfuncid = InvalidOid;
104 newopexpr->opresulttype = opexpr->opresulttype;
105 newopexpr->opretset = opexpr->opretset;
106 newopexpr->opcollid = opexpr->opcollid;
107 newopexpr->inputcollid = opexpr->inputcollid;
108 newopexpr->args = opexpr->args;
109 newopexpr->location = opexpr->location;
110 return (Node *) newopexpr;
111 }
112 }
113 break;
114 case T_ScalarArrayOpExpr:
115 {
116 /*
117 * Negate a ScalarArrayOpExpr if its operator has a negator;
118 * for example x = ANY (list) becomes x <> ALL (list)
119 */
120 ScalarArrayOpExpr *saopexpr = (ScalarArrayOpExpr *) node;
121 Oid negator = get_negator(saopexpr->opno);
122
123 if (negator)
124 {
126
127 newopexpr->opno = negator;
128 newopexpr->opfuncid = InvalidOid;
129 newopexpr->hashfuncid = InvalidOid;
130 newopexpr->negfuncid = InvalidOid;
131 newopexpr->useOr = !saopexpr->useOr;
132 newopexpr->inputcollid = saopexpr->inputcollid;
133 newopexpr->args = saopexpr->args;
134 newopexpr->location = saopexpr->location;
135 return (Node *) newopexpr;
136 }
137 }
138 break;
139 case T_BoolExpr:
140 {
141 BoolExpr *expr = (BoolExpr *) node;
142
143 switch (expr->boolop)
144 {
145 /*--------------------
146 * Apply DeMorgan's Laws:
147 * (NOT (AND A B)) => (OR (NOT A) (NOT B))
148 * (NOT (OR A B)) => (AND (NOT A) (NOT B))
149 * i.e., swap AND for OR and negate each subclause.
150 *
151 * If the input is already AND/OR flat and has no NOT
152 * directly above AND or OR, this transformation preserves
153 * those properties. For example, if no direct child of
154 * the given AND clause is an AND or a NOT-above-OR, then
155 * the recursive calls of negate_clause() can't return any
156 * OR clauses. So we needn't call pull_ors() before
157 * building a new OR clause. Similarly for the OR case.
158 *--------------------
159 */
160 case AND_EXPR:
161 {
162 List *nargs = NIL;
163 ListCell *lc;
164
165 foreach(lc, expr->args)
166 {
167 nargs = lappend(nargs,
168 negate_clause(lfirst(lc)));
169 }
170 return (Node *) make_orclause(nargs);
171 }
172 break;
173 case OR_EXPR:
174 {
175 List *nargs = NIL;
176 ListCell *lc;
177
178 foreach(lc, expr->args)
179 {
180 nargs = lappend(nargs,
181 negate_clause(lfirst(lc)));
182 }
183 return (Node *) make_andclause(nargs);
184 }
185 break;
186 case NOT_EXPR:
187
188 /*
189 * NOT underneath NOT: they cancel. We assume the
190 * input is already simplified, so no need to recurse.
191 */
192 return (Node *) linitial(expr->args);
193 default:
194 elog(ERROR, "unrecognized boolop: %d",
195 (int) expr->boolop);
196 break;
197 }
198 }
199 break;
200 case T_NullTest:
201 {
202 NullTest *expr = (NullTest *) node;
203
204 /*
205 * In the rowtype case, the two flavors of NullTest are *not*
206 * logical inverses, so we can't simplify. But it does work
207 * for scalar datatypes.
208 */
209 if (!expr->argisrow)
210 {
211 NullTest *newexpr = makeNode(NullTest);
212
213 newexpr->arg = expr->arg;
214 newexpr->nulltesttype = (expr->nulltesttype == IS_NULL ?
216 newexpr->argisrow = expr->argisrow;
217 newexpr->location = expr->location;
218 return (Node *) newexpr;
219 }
220 }
221 break;
222 case T_BooleanTest:
223 {
224 BooleanTest *expr = (BooleanTest *) node;
225 BooleanTest *newexpr = makeNode(BooleanTest);
226
227 newexpr->arg = expr->arg;
228 switch (expr->booltesttype)
229 {
230 case IS_TRUE:
231 newexpr->booltesttype = IS_NOT_TRUE;
232 break;
233 case IS_NOT_TRUE:
234 newexpr->booltesttype = IS_TRUE;
235 break;
236 case IS_FALSE:
237 newexpr->booltesttype = IS_NOT_FALSE;
238 break;
239 case IS_NOT_FALSE:
240 newexpr->booltesttype = IS_FALSE;
241 break;
242 case IS_UNKNOWN:
243 newexpr->booltesttype = IS_NOT_UNKNOWN;
244 break;
245 case IS_NOT_UNKNOWN:
246 newexpr->booltesttype = IS_UNKNOWN;
247 break;
248 default:
249 elog(ERROR, "unrecognized booltesttype: %d",
250 (int) expr->booltesttype);
251 break;
252 }
253 newexpr->location = expr->location;
254 return (Node *) newexpr;
255 }
256 break;
257 default:
258 /* else fall through */
259 break;
260 }
261
262 /*
263 * Otherwise we don't know how to simplify this, so just tack on an
264 * explicit NOT node.
265 */
266 return (Node *) make_notclause((Expr *) node);
267}
Oid get_negator(Oid opno)
Definition: lsyscache.c:1700
Expr * make_orclause(List *orclauses)
Definition: makefuncs.c:743
Node * makeBoolConst(bool value, bool isnull)
Definition: makefuncs.c:408
Expr * make_andclause(List *andclauses)
Definition: makefuncs.c:727
Expr * make_notclause(Expr *notclause)
Definition: makefuncs.c:759
#define nodeTag(nodeptr)
Definition: nodes.h:139
Node * negate_clause(Node *node)
Definition: prepqual.c:73
char * c
@ IS_NOT_TRUE
Definition: primnodes.h:1987
@ IS_NOT_FALSE
Definition: primnodes.h:1987
@ IS_NOT_UNKNOWN
Definition: primnodes.h:1987
@ IS_TRUE
Definition: primnodes.h:1987
@ IS_UNKNOWN
Definition: primnodes.h:1987
@ IS_FALSE
Definition: primnodes.h:1987
@ AND_EXPR
Definition: primnodes.h:950
@ OR_EXPR
Definition: primnodes.h:950
@ NOT_EXPR
Definition: primnodes.h:950
@ IS_NOT_NULL
Definition: primnodes.h:1963
BoolExprType boolop
Definition: primnodes.h:958
List * args
Definition: primnodes.h:959
ParseLoc location
Definition: primnodes.h:1995
BoolTestType booltesttype
Definition: primnodes.h:1994
Expr * arg
Definition: primnodes.h:1993
NullTestType nulltesttype
Definition: primnodes.h:1970
ParseLoc location
Definition: primnodes.h:1973
Expr * arg
Definition: primnodes.h:1969
ParseLoc location
Definition: primnodes.h:858

References AND_EXPR, NullTest::arg, BooleanTest::arg, OpExpr::args, ScalarArrayOpExpr::args, BoolExpr::args, BoolExpr::boolop, BooleanTest::booltesttype, DatumGetBool(), elog, ERROR, get_negator(), InvalidOid, IS_FALSE, IS_NOT_FALSE, IS_NOT_NULL, IS_NOT_TRUE, IS_NOT_UNKNOWN, IS_NULL, IS_TRUE, IS_UNKNOWN, lappend(), lfirst, linitial, OpExpr::location, ScalarArrayOpExpr::location, NullTest::location, BooleanTest::location, make_andclause(), make_notclause(), make_orclause(), makeBoolConst(), makeNode, negate_clause(), NIL, nodeTag, NOT_EXPR, NullTest::nulltesttype, OpExpr::opno, ScalarArrayOpExpr::opno, OR_EXPR, and ScalarArrayOpExpr::useOr.

Referenced by eval_const_expressions_mutator(), match_boolean_partition_clause(), negate_clause(), and simplify_boolean_equality().

◆ plan_cluster_use_sort()

bool plan_cluster_use_sort ( Oid  tableOid,
Oid  indexOid 
)

Definition at line 6799 of file planner.c.

6800{
6802 Query *query;
6803 PlannerGlobal *glob;
6804 RangeTblEntry *rte;
6805 RelOptInfo *rel;
6806 IndexOptInfo *indexInfo;
6807 QualCost indexExprCost;
6808 Cost comparisonCost;
6809 Path *seqScanPath;
6810 Path seqScanAndSortPath;
6811 IndexPath *indexScanPath;
6812 ListCell *lc;
6813
6814 /* We can short-circuit the cost comparison if indexscans are disabled */
6815 if (!enable_indexscan)
6816 return true; /* use sort */
6817
6818 /* Set up mostly-dummy planner state */
6819 query = makeNode(Query);
6820 query->commandType = CMD_SELECT;
6821
6822 glob = makeNode(PlannerGlobal);
6823
6825 root->parse = query;
6826 root->glob = glob;
6827 root->query_level = 1;
6828 root->planner_cxt = CurrentMemoryContext;
6829 root->wt_param_id = -1;
6830 root->join_domains = list_make1(makeNode(JoinDomain));
6831
6832 /* Build a minimal RTE for the rel */
6833 rte = makeNode(RangeTblEntry);
6834 rte->rtekind = RTE_RELATION;
6835 rte->relid = tableOid;
6836 rte->relkind = RELKIND_RELATION; /* Don't be too picky. */
6837 rte->rellockmode = AccessShareLock;
6838 rte->lateral = false;
6839 rte->inh = false;
6840 rte->inFromCl = true;
6841 query->rtable = list_make1(rte);
6842 addRTEPermissionInfo(&query->rteperminfos, rte);
6843
6844 /* Set up RTE/RelOptInfo arrays */
6846
6847 /* Build RelOptInfo */
6848 rel = build_simple_rel(root, 1, NULL);
6849
6850 /* Locate IndexOptInfo for the target index */
6851 indexInfo = NULL;
6852 foreach(lc, rel->indexlist)
6853 {
6854 indexInfo = lfirst_node(IndexOptInfo, lc);
6855 if (indexInfo->indexoid == indexOid)
6856 break;
6857 }
6858
6859 /*
6860 * It's possible that get_relation_info did not generate an IndexOptInfo
6861 * for the desired index; this could happen if it's not yet reached its
6862 * indcheckxmin usability horizon, or if it's a system index and we're
6863 * ignoring system indexes. In such cases we should tell CLUSTER to not
6864 * trust the index contents but use seqscan-and-sort.
6865 */
6866 if (lc == NULL) /* not in the list? */
6867 return true; /* use sort */
6868
6869 /*
6870 * Rather than doing all the pushups that would be needed to use
6871 * set_baserel_size_estimates, just do a quick hack for rows and width.
6872 */
6873 rel->rows = rel->tuples;
6874 rel->reltarget->width = get_relation_data_width(tableOid, NULL);
6875
6876 root->total_table_pages = rel->pages;
6877
6878 /*
6879 * Determine eval cost of the index expressions, if any. We need to
6880 * charge twice that amount for each tuple comparison that happens during
6881 * the sort, since tuplesort.c will have to re-evaluate the index
6882 * expressions each time. (XXX that's pretty inefficient...)
6883 */
6884 cost_qual_eval(&indexExprCost, indexInfo->indexprs, root);
6885 comparisonCost = 2.0 * (indexExprCost.startup + indexExprCost.per_tuple);
6886
6887 /* Estimate the cost of seq scan + sort */
6888 seqScanPath = create_seqscan_path(root, rel, NULL, 0);
6889 cost_sort(&seqScanAndSortPath, root, NIL,
6890 seqScanPath->disabled_nodes,
6891 seqScanPath->total_cost, rel->tuples, rel->reltarget->width,
6892 comparisonCost, maintenance_work_mem, -1.0);
6893
6894 /* Estimate the cost of index scan */
6895 indexScanPath = create_index_path(root, indexInfo,
6896 NIL, NIL, NIL, NIL,
6897 ForwardScanDirection, false,
6898 NULL, 1.0, false);
6899
6900 return (seqScanAndSortPath.total_cost < indexScanPath->path.total_cost);
6901}
void cost_sort(Path *path, PlannerInfo *root, List *pathkeys, int input_disabled_nodes, Cost input_cost, double tuples, int width, Cost comparison_cost, int sort_mem, double limit_tuples)
Definition: costsize.c:2144
void cost_qual_eval(QualCost *cost, List *quals, PlannerInfo *root)
Definition: costsize.c:4791
bool enable_indexscan
Definition: costsize.c:146
int maintenance_work_mem
Definition: globals.c:133
#define AccessShareLock
Definition: lockdefs.h:36
MemoryContext CurrentMemoryContext
Definition: mcxt.c:160
double Cost
Definition: nodes.h:261
@ CMD_SELECT
Definition: nodes.h:275
RTEPermissionInfo * addRTEPermissionInfo(List **rteperminfos, RangeTblEntry *rte)
IndexPath * create_index_path(PlannerInfo *root, IndexOptInfo *index, List *indexclauses, List *indexorderbys, List *indexorderbycols, List *pathkeys, ScanDirection indexscandir, bool indexonly, Relids required_outer, double loop_count, bool partial_path)
Definition: pathnode.c:1047
Path * create_seqscan_path(PlannerInfo *root, RelOptInfo *rel, Relids required_outer, int parallel_workers)
Definition: pathnode.c:981
#define lfirst_node(type, lc)
Definition: pg_list.h:176
#define list_make1(x1)
Definition: pg_list.h:212
int32 get_relation_data_width(Oid relid, int32 *attr_widths)
Definition: plancat.c:1323
void setup_simple_rel_arrays(PlannerInfo *root)
Definition: relnode.c:94
RelOptInfo * build_simple_rel(PlannerInfo *root, int relid, RelOptInfo *parent)
Definition: relnode.c:192
@ ForwardScanDirection
Definition: sdir.h:28
Path path
Definition: pathnodes.h:1868
int disabled_nodes
Definition: pathnodes.h:1819
Cost total_cost
Definition: pathnodes.h:1821
Cost per_tuple
Definition: pathnodes.h:48
Cost startup
Definition: pathnodes.h:47
List * rtable
Definition: parsenodes.h:175
CmdType commandType
Definition: parsenodes.h:121
RTEKind rtekind
Definition: parsenodes.h:1077
struct PathTarget * reltarget
Definition: pathnodes.h:930
BlockNumber pages
Definition: pathnodes.h:980
List * indexlist
Definition: pathnodes.h:976
Cardinality rows
Definition: pathnodes.h:914

References AccessShareLock, addRTEPermissionInfo(), build_simple_rel(), CMD_SELECT, Query::commandType, cost_qual_eval(), cost_sort(), create_index_path(), create_seqscan_path(), CurrentMemoryContext, Path::disabled_nodes, enable_indexscan, ForwardScanDirection, get_relation_data_width(), RelOptInfo::indexlist, IndexOptInfo::indexoid, RangeTblEntry::inh, lfirst_node, list_make1, maintenance_work_mem, makeNode, NIL, RelOptInfo::pages, IndexPath::path, QualCost::per_tuple, RelOptInfo::reltarget, root, RelOptInfo::rows, Query::rtable, RTE_RELATION, RangeTblEntry::rtekind, setup_simple_rel_arrays(), QualCost::startup, Path::total_cost, RelOptInfo::tuples, and PathTarget::width.

Referenced by copy_table_data().

◆ plan_create_index_workers()

int plan_create_index_workers ( Oid  tableOid,
Oid  indexOid 
)

Definition at line 6921 of file planner.c.

6922{
6924 Query *query;
6925 PlannerGlobal *glob;
6926 RangeTblEntry *rte;
6927 Relation heap;
6929 RelOptInfo *rel;
6930 int parallel_workers;
6931 BlockNumber heap_blocks;
6932 double reltuples;
6933 double allvisfrac;
6934
6935 /*
6936 * We don't allow performing parallel operation in standalone backend or
6937 * when parallelism is disabled.
6938 */
6940 return 0;
6941
6942 /* Set up largely-dummy planner state */
6943 query = makeNode(Query);
6944 query->commandType = CMD_SELECT;
6945
6946 glob = makeNode(PlannerGlobal);
6947
6949 root->parse = query;
6950 root->glob = glob;
6951 root->query_level = 1;
6952 root->planner_cxt = CurrentMemoryContext;
6953 root->wt_param_id = -1;
6954 root->join_domains = list_make1(makeNode(JoinDomain));
6955
6956 /*
6957 * Build a minimal RTE.
6958 *
6959 * Mark the RTE with inh = true. This is a kludge to prevent
6960 * get_relation_info() from fetching index info, which is necessary
6961 * because it does not expect that any IndexOptInfo is currently
6962 * undergoing REINDEX.
6963 */
6964 rte = makeNode(RangeTblEntry);
6965 rte->rtekind = RTE_RELATION;
6966 rte->relid = tableOid;
6967 rte->relkind = RELKIND_RELATION; /* Don't be too picky. */
6968 rte->rellockmode = AccessShareLock;
6969 rte->lateral = false;
6970 rte->inh = true;
6971 rte->inFromCl = true;
6972 query->rtable = list_make1(rte);
6973 addRTEPermissionInfo(&query->rteperminfos, rte);
6974
6975 /* Set up RTE/RelOptInfo arrays */
6977
6978 /* Build RelOptInfo */
6979 rel = build_simple_rel(root, 1, NULL);
6980
6981 /* Rels are assumed already locked by the caller */
6982 heap = table_open(tableOid, NoLock);
6983 index = index_open(indexOid, NoLock);
6984
6985 /*
6986 * Determine if it's safe to proceed.
6987 *
6988 * Currently, parallel workers can't access the leader's temporary tables.
6989 * Furthermore, any index predicate or index expressions must be parallel
6990 * safe.
6991 */
6992 if (heap->rd_rel->relpersistence == RELPERSISTENCE_TEMP ||
6995 {
6996 parallel_workers = 0;
6997 goto done;
6998 }
6999
7000 /*
7001 * If parallel_workers storage parameter is set for the table, accept that
7002 * as the number of parallel worker processes to launch (though still cap
7003 * at max_parallel_maintenance_workers). Note that we deliberately do not
7004 * consider any other factor when parallel_workers is set. (e.g., memory
7005 * use by workers.)
7006 */
7007 if (rel->rel_parallel_workers != -1)
7008 {
7009 parallel_workers = Min(rel->rel_parallel_workers,
7011 goto done;
7012 }
7013
7014 /*
7015 * Estimate heap relation size ourselves, since rel->pages cannot be
7016 * trusted (heap RTE was marked as inheritance parent)
7017 */
7018 estimate_rel_size(heap, NULL, &heap_blocks, &reltuples, &allvisfrac);
7019
7020 /*
7021 * Determine number of workers to scan the heap relation using generic
7022 * model
7023 */
7024 parallel_workers = compute_parallel_worker(rel, heap_blocks, -1,
7026
7027 /*
7028 * Cap workers based on available maintenance_work_mem as needed.
7029 *
7030 * Note that each tuplesort participant receives an even share of the
7031 * total maintenance_work_mem budget. Aim to leave participants
7032 * (including the leader as a participant) with no less than 32MB of
7033 * memory. This leaves cases where maintenance_work_mem is set to 64MB
7034 * immediately past the threshold of being capable of launching a single
7035 * parallel worker to sort.
7036 */
7037 while (parallel_workers > 0 &&
7038 maintenance_work_mem / (parallel_workers + 1) < 32 * 1024)
7039 parallel_workers--;
7040
7041done:
7043 table_close(heap, NoLock);
7044
7045 return parallel_workers;
7046}
int compute_parallel_worker(RelOptInfo *rel, double heap_pages, double index_pages, int max_workers)
Definition: allpaths.c:4241
uint32 BlockNumber
Definition: block.h:31
#define Min(x, y)
Definition: c.h:1004
bool is_parallel_safe(PlannerInfo *root, Node *node)
Definition: clauses.c:757
int max_parallel_maintenance_workers
Definition: globals.c:134
bool IsUnderPostmaster
Definition: globals.c:120
void index_close(Relation relation, LOCKMODE lockmode)
Definition: indexam.c:177
Relation index_open(Oid relationId, LOCKMODE lockmode)
Definition: indexam.c:133
#define NoLock
Definition: lockdefs.h:34
void estimate_rel_size(Relation rel, int32 *attr_widths, BlockNumber *pages, double *tuples, double *allvisfrac)
Definition: plancat.c:1156
List * RelationGetIndexPredicate(Relation relation)
Definition: relcache.c:5210
List * RelationGetIndexExpressions(Relation relation)
Definition: relcache.c:5097
int rel_parallel_workers
Definition: pathnodes.h:988
Form_pg_class rd_rel
Definition: rel.h:111
void table_close(Relation relation, LOCKMODE lockmode)
Definition: table.c:126
Relation table_open(Oid relationId, LOCKMODE lockmode)
Definition: table.c:40

References AccessShareLock, addRTEPermissionInfo(), build_simple_rel(), CMD_SELECT, Query::commandType, compute_parallel_worker(), CurrentMemoryContext, estimate_rel_size(), index_close(), index_open(), RangeTblEntry::inh, is_parallel_safe(), IsUnderPostmaster, list_make1, maintenance_work_mem, makeNode, max_parallel_maintenance_workers, Min, NoLock, RelationData::rd_rel, RelOptInfo::rel_parallel_workers, RelationGetIndexExpressions(), RelationGetIndexPredicate(), root, Query::rtable, RTE_RELATION, RangeTblEntry::rtekind, setup_simple_rel_arrays(), table_close(), and table_open().

Referenced by index_build().

◆ planner()

PlannedStmt * planner ( Query parse,
const char *  query_string,
int  cursorOptions,
ParamListInfo  boundParams 
)

Definition at line 293 of file planner.c.

295{
296 PlannedStmt *result;
297
298 if (planner_hook)
299 result = (*planner_hook) (parse, query_string, cursorOptions, boundParams);
300 else
301 result = standard_planner(parse, query_string, cursorOptions, boundParams);
302
303 pgstat_report_plan_id(result->planId, false);
304
305 return result;
306}
void pgstat_report_plan_id(int64 plan_id, bool force)
planner_hook_type planner_hook
Definition: planner.c:74
PlannedStmt * standard_planner(Query *parse, const char *query_string, int cursorOptions, ParamListInfo boundParams)
Definition: planner.c:309
static struct subre * parse(struct vars *v, int stopper, int type, struct state *init, struct state *final)
Definition: regcomp.c:717

References parse(), pgstat_report_plan_id(), PlannedStmt::planId, planner_hook, and standard_planner().

Referenced by pg_plan_query().

◆ predicate_implied_by()

bool predicate_implied_by ( List predicate_list,
List clause_list,
bool  weak 
)

Definition at line 152 of file predtest.c.

154{
155 Node *p,
156 *c;
157
158 if (predicate_list == NIL)
159 return true; /* no predicate: implication is vacuous */
160 if (clause_list == NIL)
161 return false; /* no restriction: implication must fail */
162
163 /*
164 * If either input is a single-element list, replace it with its lone
165 * member; this avoids one useless level of AND-recursion. We only need
166 * to worry about this at top level, since eval_const_expressions should
167 * have gotten rid of any trivial ANDs or ORs below that.
168 */
169 if (list_length(predicate_list) == 1)
170 p = (Node *) linitial(predicate_list);
171 else
172 p = (Node *) predicate_list;
173 if (list_length(clause_list) == 1)
174 c = (Node *) linitial(clause_list);
175 else
176 c = (Node *) clause_list;
177
178 /* And away we go ... */
179 return predicate_implied_by_recurse(c, p, weak);
180}
static bool predicate_implied_by_recurse(Node *clause, Node *predicate, bool weak)
Definition: predtest.c:290

References linitial, list_length(), NIL, and predicate_implied_by_recurse().

Referenced by add_predicate_to_index_quals(), build_paths_for_OR(), check_index_predicates(), choose_bitmap_and(), ConstraintImpliedByRelConstraint(), create_bitmap_scan_plan(), create_bitmap_subplan(), create_indexscan_plan(), infer_arbiter_indexes(), and test_predtest().

◆ predicate_refuted_by()

bool predicate_refuted_by ( List predicate_list,
List clause_list,
bool  weak 
)

Definition at line 222 of file predtest.c.

224{
225 Node *p,
226 *c;
227
228 if (predicate_list == NIL)
229 return false; /* no predicate: no refutation is possible */
230 if (clause_list == NIL)
231 return false; /* no restriction: refutation must fail */
232
233 /*
234 * If either input is a single-element list, replace it with its lone
235 * member; this avoids one useless level of AND-recursion. We only need
236 * to worry about this at top level, since eval_const_expressions should
237 * have gotten rid of any trivial ANDs or ORs below that.
238 */
239 if (list_length(predicate_list) == 1)
240 p = (Node *) linitial(predicate_list);
241 else
242 p = (Node *) predicate_list;
243 if (list_length(clause_list) == 1)
244 c = (Node *) linitial(clause_list);
245 else
246 c = (Node *) clause_list;
247
248 /* And away we go ... */
249 return predicate_refuted_by_recurse(c, p, weak);
250}
static bool predicate_refuted_by_recurse(Node *clause, Node *predicate, bool weak)
Definition: predtest.c:531

References linitial, list_length(), NIL, and predicate_refuted_by_recurse().

Referenced by gen_partprune_steps_internal(), relation_excluded_by_constraints(), and test_predtest().

◆ pull_var_clause()

List * pull_var_clause ( Node node,
int  flags 
)

Definition at line 653 of file var.c.

654{
656
657 /* Assert that caller has not specified inconsistent flags */
664
665 context.varlist = NIL;
666 context.flags = flags;
667
668 pull_var_clause_walker(node, &context);
669 return context.varlist;
670}
#define PVC_RECURSE_AGGREGATES
Definition: optimizer.h:184
#define PVC_RECURSE_PLACEHOLDERS
Definition: optimizer.h:188
#define PVC_RECURSE_WINDOWFUNCS
Definition: optimizer.h:186
#define PVC_INCLUDE_WINDOWFUNCS
Definition: optimizer.h:185
#define PVC_INCLUDE_PLACEHOLDERS
Definition: optimizer.h:187
#define PVC_INCLUDE_AGGREGATES
Definition: optimizer.h:183
static bool pull_var_clause_walker(Node *node, pull_var_clause_context *context)
Definition: var.c:673

References Assert(), pull_var_clause_context::flags, NIL, pull_var_clause_walker(), PVC_INCLUDE_AGGREGATES, PVC_INCLUDE_PLACEHOLDERS, PVC_INCLUDE_WINDOWFUNCS, PVC_RECURSE_AGGREGATES, PVC_RECURSE_PLACEHOLDERS, PVC_RECURSE_WINDOWFUNCS, and pull_var_clause_context::varlist.

Referenced by add_paths_with_pathkeys_for_rel(), AddRelationNewConstraints(), build_base_rel_tlists(), build_remote_returning(), build_tlist_to_deparse(), CreateTriggerFiringOn(), distribute_qual_to_rels(), estimate_num_groups(), find_computable_ec_member(), find_placeholders_in_expr(), fix_placeholder_input_needed_levels(), foreign_grouping_ok(), generate_base_implied_equalities_no_const(), make_group_input_target(), make_partial_grouping_target(), make_sort_input_target(), make_window_input_target(), preprocess_targetlist(), process_implied_equality(), qual_is_pushdown_safe(), rebuild_eclass_attr_needed(), rebuild_joinclause_attr_needed(), rebuild_placeholder_attr_needed(), semijoin_target_ok(), and StoreRelCheck().

◆ pull_varattnos()

void pull_varattnos ( Node node,
Index  varno,
Bitmapset **  varattnos 
)

◆ pull_varnos()

Bitmapset * pull_varnos ( PlannerInfo root,
Node node 
)

Definition at line 114 of file var.c.

115{
116 pull_varnos_context context;
117
118 context.varnos = NULL;
119 context.root = root;
120 context.sublevels_up = 0;
121
122 /*
123 * Must be prepared to start with a Query or a bare expression tree; if
124 * it's a Query, we don't want to increment sublevels_up.
125 */
128 &context,
129 0);
130
131 return context.varnos;
132}
PlannerInfo * root
Definition: var.c:36
int sublevels_up
Definition: var.c:37
Relids varnos
Definition: var.c:35
static bool pull_varnos_walker(Node *node, pull_varnos_context *context)
Definition: var.c:161

References pull_varnos_walker(), query_or_expression_tree_walker, pull_varnos_context::root, root, pull_varnos_context::sublevels_up, and pull_varnos_context::varnos.

Referenced by compute_semijoin_info(), convert_ANY_sublink_to_join(), convert_EXISTS_sublink_to_join(), cost_incremental_sort(), distribute_qual_to_rels(), examine_variable(), expand_indexqual_rowcompare(), extract_lateral_vars_from_PHVs(), find_placeholder_info(), get_eclass_for_sort_expr(), is_pseudo_constant_for_index(), IsTidEqualAnyClause(), join_is_removable(), make_outerjoininfo(), make_plain_restrictinfo(), match_rowcompare_to_indexcol(), match_saopclause_to_indexcol(), NumRelids(), pg_get_expr_worker(), process_implied_equality(), pullup_replace_vars_callback(), remove_leftjoinrel_from_query(), and subquery_planner().

◆ pull_varnos_of_level()

Bitmapset * pull_varnos_of_level ( PlannerInfo root,
Node node,
int  levelsup 
)

Definition at line 140 of file var.c.

141{
142 pull_varnos_context context;
143
144 context.varnos = NULL;
145 context.root = root;
146 context.sublevels_up = levelsup;
147
148 /*
149 * Must be prepared to start with a Query or a bare expression tree; if
150 * it's a Query, we don't want to increment sublevels_up.
151 */
154 &context,
155 0);
156
157 return context.varnos;
158}

References pull_varnos_walker(), query_or_expression_tree_walker, pull_varnos_context::root, root, pull_varnos_context::sublevels_up, and pull_varnos_context::varnos.

Referenced by add_nullingrels_if_needed(), convert_ANY_sublink_to_join(), is_simple_subquery(), jointree_contains_lateral_outer_refs(), and mark_nullable_by_grouping().

◆ pull_vars_of_level()

List * pull_vars_of_level ( Node node,
int  levelsup 
)

Definition at line 339 of file var.c.

340{
341 pull_vars_context context;
342
343 context.vars = NIL;
344 context.sublevels_up = levelsup;
345
346 /*
347 * Must be prepared to start with a Query or a bare expression tree; if
348 * it's a Query, we don't want to increment sublevels_up.
349 */
352 &context,
353 0);
354
355 return context.vars;
356}
int sublevels_up
Definition: var.c:49
List * vars
Definition: var.c:48
static bool pull_vars_walker(Node *node, pull_vars_context *context)
Definition: var.c:359

References NIL, pull_vars_walker(), query_or_expression_tree_walker, pull_vars_context::sublevels_up, and pull_vars_context::vars.

Referenced by extract_lateral_references(), and extract_lateral_vars_from_PHVs().

◆ var_is_nonnullable()

bool var_is_nonnullable ( PlannerInfo root,
Var var,
bool  use_rel_info 
)

Definition at line 4206 of file clauses.c.

4207{
4208 Bitmapset *notnullattnums = NULL;
4209
4210 Assert(IsA(var, Var));
4211
4212 /* skip upper-level Vars */
4213 if (var->varlevelsup != 0)
4214 return false;
4215
4216 /* could the Var be nulled by any outer joins or grouping sets? */
4217 if (!bms_is_empty(var->varnullingrels))
4218 return false;
4219
4220 /* system columns cannot be NULL */
4221 if (var->varattno < 0)
4222 return true;
4223
4224 /*
4225 * Check if the Var is defined as NOT NULL. We retrieve the column NOT
4226 * NULL constraint information from the corresponding RelOptInfo if it is
4227 * available; otherwise, we search the hash table for this information.
4228 */
4229 if (use_rel_info)
4230 {
4231 RelOptInfo *rel = find_base_rel(root, var->varno);
4232
4233 notnullattnums = rel->notnullattnums;
4234 }
4235 else
4236 {
4238
4239 /*
4240 * We must skip inheritance parent tables, as some child tables may
4241 * have a NOT NULL constraint for a column while others may not. This
4242 * cannot happen with partitioned tables, though.
4243 */
4244 if (rte->inh && rte->relkind != RELKIND_PARTITIONED_TABLE)
4245 return false;
4246
4247 notnullattnums = find_relation_notnullatts(root, rte->relid);
4248 }
4249
4250 if (var->varattno > 0 &&
4251 bms_is_member(var->varattno, notnullattnums))
4252 return true;
4253
4254 return false;
4255}
#define bms_is_empty(a)
Definition: bitmapset.h:118
#define planner_rt_fetch(rti, root)
Definition: pathnodes.h:591
Bitmapset * find_relation_notnullatts(PlannerInfo *root, Oid relid)
Definition: plancat.c:755
Bitmapset * notnullattnums
Definition: pathnodes.h:968
AttrNumber varattno
Definition: primnodes.h:274

References Assert(), bms_is_empty, bms_is_member(), find_base_rel(), find_relation_notnullatts(), RangeTblEntry::inh, IsA, RelOptInfo::notnullattnums, planner_rt_fetch, root, Var::varattno, Var::varlevelsup, and Var::varno.

Referenced by eval_const_expressions_mutator(), and expr_is_nonnullable().

Variable Documentation

◆ cpu_index_tuple_cost

PGDLLIMPORT double cpu_index_tuple_cost
extern

Definition at line 133 of file costsize.c.

Referenced by genericcostestimate(), and gincostestimate().

◆ cpu_operator_cost

◆ cpu_tuple_cost

◆ debug_parallel_query

PGDLLIMPORT int debug_parallel_query
extern

Definition at line 69 of file planner.c.

Referenced by ProcessParallelMessage(), query_planner(), and standard_planner().

◆ effective_cache_size

PGDLLIMPORT int effective_cache_size
extern

Definition at line 139 of file costsize.c.

Referenced by gistBuildCallback(), gistInitBuffering(), and index_pages_fetched().

◆ enable_distinct_reordering

PGDLLIMPORT bool enable_distinct_reordering
extern

Definition at line 71 of file planner.c.

Referenced by get_useful_pathkeys_for_distinct().

◆ parallel_leader_participation

PGDLLIMPORT bool parallel_leader_participation
extern

Definition at line 70 of file planner.c.

Referenced by ExecGather(), ExecGatherMerge(), ExecInitGather(), and get_parallel_divisor().

◆ parallel_setup_cost

PGDLLIMPORT double parallel_setup_cost
extern

Definition at line 136 of file costsize.c.

Referenced by cost_gather(), cost_gather_merge(), and standard_planner().

◆ parallel_tuple_cost

PGDLLIMPORT double parallel_tuple_cost
extern

Definition at line 135 of file costsize.c.

Referenced by cost_gather(), cost_gather_merge(), and standard_planner().

◆ random_page_cost

PGDLLIMPORT double random_page_cost
extern

◆ recursive_worktable_factor

PGDLLIMPORT double recursive_worktable_factor
extern

Definition at line 137 of file costsize.c.

Referenced by set_cte_size_estimates().

◆ seq_page_cost