40#define PATH_PARAM_BY_PARENT(path, rel) \
41 ((path)->param_info && bms_overlap(PATH_REQ_OUTER(path), \
42 (rel)->top_parent_relids))
43#define PATH_PARAM_BY_REL_SELF(path, rel) \
44 ((path)->param_info && bms_overlap(PATH_REQ_OUTER(path), (rel)->relids))
46#define PATH_PARAM_BY_REL(path, rel) \
47 (PATH_PARAM_BY_REL_SELF(path, rel) || PATH_PARAM_BY_PARENT(path, rel))
77 Path *inner_cheapest_total);
87 bool *mergejoin_allowed);
95 Path *inner_cheapest_total,
133 bool mergejoin_allowed =
true;
147 joinrelids = joinrel->
relids;
225 jointype, sjinfo, restrictlist,
242 foreach(lc,
root->join_info_list)
282 if (mergejoin_allowed)
294 if (mergejoin_allowed)
311 if (mergejoin_allowed)
312 match_unsorted_inner(
root, joinrel, outerrel, innerrel,
330 if (joinrel->fdwroutine &&
331 joinrel->fdwroutine->GetForeignJoinPaths)
332 joinrel->fdwroutine->GetForeignJoinPaths(
root, joinrel,
371 return (
bms_overlap(inner_paramrels, outerrelids) &&
388#ifdef USE_ASSERT_CHECKING
402 foreach(lc,
root->join_info_list)
441 List *ph_lateral_vars,
List **param_exprs,
442 List **operators,
bool *binary_mode)
450 *binary_mode =
false;
453 if (param_info != NULL)
479 if (rinfo->outer_is_left)
482 hasheqoperator = rinfo->left_hasheqoperator;
487 hasheqoperator = rinfo->right_hasheqoperator;
505 *operators =
lappend_oid(*operators, hasheqoperator);
506 *param_exprs =
lappend(*param_exprs, expr);
527 foreach(lc, lateral_vars)
559 *param_exprs =
lappend(*param_exprs, expr);
590 if (!
root->hasLateralRTEs)
600 foreach(lc,
root->placeholder_list)
624 ph_lateral_vars =
lappend(ph_lateral_vars, phinfo->
ph_var->phexpr);
641 ph_lateral_vars =
lappend(ph_lateral_vars, node);
651 ph_lateral_vars =
lappend(ph_lateral_vars, node);
660 return ph_lateral_vars;
681 List *hash_operators;
684 List *ph_lateral_vars;
696 if (outer_path->parent->
rows < 2)
711 if ((inner_path->param_info == NULL ||
712 inner_path->param_info->ppi_clauses ==
NIL) &&
714 ph_lateral_vars ==
NIL)
756 if (inner_path->param_info == NULL)
759 ppi_serials = inner_path->param_info->ppi_serials;
790 if (inner_path->param_info != NULL)
792 foreach(lc, inner_path->param_info->ppi_clauses)
803 inner_path->param_info,
804 outerrel->top_parent ?
805 outerrel->top_parent : outerrel,
869 innerrelids = innerrel->
relids;
874 outerrelids = outerrel->
relids;
884 innerrelids, inner_paramrels);
885 if (required_outer &&
896 Assert(!have_unsafe_outer_join_ref(
root, outerrelids, inner_paramrels));
922 outer_path, inner_path, extra);
926 pathkeys, required_outer))
971 if (inner_path->param_info != NULL)
973 Relids inner_paramrels = inner_path->param_info->ppi_req_outer;
985 outerrelids = outerrel->
relids;
1007 outer_path, inner_path, extra);
1038 List *outersortkeys,
1039 List *innersortkeys,
1045 int outer_presorted_keys = 0;
1080 if (required_outer &&
1098 if (outersortkeys &&
1100 &outer_presorted_keys))
1101 outersortkeys =
NIL;
1102 if (innersortkeys &&
1104 innersortkeys =
NIL;
1110 outer_path, inner_path,
1111 outersortkeys, innersortkeys,
1112 outer_presorted_keys,
1117 pathkeys, required_outer))
1133 outer_presorted_keys));
1154 List *outersortkeys,
1155 List *innersortkeys,
1159 int outer_presorted_keys = 0;
1180 if (outersortkeys &&
1182 &outer_presorted_keys))
1183 outersortkeys =
NIL;
1184 if (innersortkeys &&
1186 innersortkeys =
NIL;
1192 outer_path, inner_path,
1193 outersortkeys, innersortkeys,
1194 outer_presorted_keys,
1216 outer_presorted_keys));
1253 if (required_outer &&
1266 outer_path, inner_path, extra,
false);
1270 NIL, required_outer))
1328 outer_path, inner_path, extra, parallel_hash);
1370 Path *cheapest_partial_outer = NULL;
1371 Path *cheapest_safe_inner = NULL;
1412 outer_path, extra->
sjinfo);
1419 inner_path, extra->
sjinfo);
1443 cheapest_safe_inner = inner_path;
1445 cheapest_safe_inner =
1481 foreach(l, all_pathkeys)
1484 List *cur_mergeclauses;
1487 List *merge_pathkeys;
1491 outerkeys =
lcons(front_pathkey,
1495 outerkeys = all_pathkeys;
1538 if (cheapest_partial_outer && cheapest_safe_inner)
1541 cheapest_partial_outer,
1542 cheapest_safe_inner,
1574 Path *inner_cheapest_total,
1575 List *merge_pathkeys,
1579 List *innersortkeys;
1580 List *trialsortkeys;
1581 Path *cheapest_startup_inner;
1582 Path *cheapest_total_inner;
1605 if (mergeclauses ==
NIL)
1612 if (useallclauses &&
1630 inner_cheapest_total,
1676 cheapest_startup_inner = inner_cheapest_total;
1677 cheapest_total_inner = inner_cheapest_total;
1682 cheapest_startup_inner = NULL;
1683 cheapest_total_inner = NULL;
1686 if (num_sortkeys > 1 && !useallclauses)
1687 trialsortkeys =
list_copy(innersortkeys);
1689 trialsortkeys = innersortkeys;
1691 for (sortkeycnt = num_sortkeys; sortkeycnt > 0; sortkeycnt--)
1707 if (innerpath != NULL &&
1708 (cheapest_total_inner == NULL ||
1714 if (sortkeycnt < num_sortkeys)
1723 newclauses = mergeclauses;
1735 cheapest_total_inner = innerpath;
1743 if (innerpath != NULL &&
1744 (cheapest_startup_inner == NULL ||
1749 if (innerpath != cheapest_total_inner)
1755 if (newclauses ==
NIL)
1757 if (sortkeycnt < num_sortkeys)
1766 newclauses = mergeclauses;
1780 cheapest_startup_inner = innerpath;
1826 Path *matpath = NULL;
1850 useallclauses =
false;
1856 useallclauses =
true;
1862 useallclauses =
false;
1865 elog(
ERROR,
"unrecognized join type: %d",
1868 useallclauses =
false;
1878 inner_cheapest_total = NULL;
1887 if (inner_cheapest_total == NULL)
1889 inner_cheapest_total = (
Path *)
1891 Assert(inner_cheapest_total);
1893 else if (nestjoinOK)
1909 List *merge_pathkeys;
1927 outerpath, extra->
sjinfo);
1948 inner_cheapest_total,
1953 else if (nestjoinOK)
1981 innerpath, outerpath, jointype,
1994 if (matpath != NULL)
2009 if (inner_cheapest_total == NULL)
2014 save_jointype, extra, useallclauses,
2015 inner_cheapest_total, merge_pathkeys,
2038 save_jointype, extra);
2045 if (inner_cheapest_total == NULL ||
2054 if (inner_cheapest_total)
2056 save_jointype, extra,
2057 inner_cheapest_total);
2080 Path *inner_cheapest_total)
2088 List *merge_pathkeys;
2097 extra,
false, inner_cheapest_total,
2098 merge_pathkeys,
true);
2123 Path *matpath = NULL;
2190 pathkeys, jointype, extra);
2197 innerpath, outerpath, jointype,
2201 pathkeys, jointype, extra);
2205 if (matpath != NULL)
2207 pathkeys, jointype, extra);
2254 if (!restrictinfo->can_join ||
2255 restrictinfo->hashjoinoperator ==
InvalidOid)
2275 if (!restrictinfo->outer_is_left &&
2279 hashclauses =
lappend(hashclauses, restrictinfo);
2307 cheapest_total_outer = (
Path *)
2309 cheapest_total_outer, extra->
sjinfo);
2310 Assert(cheapest_total_outer);
2314 cheapest_total_outer,
2315 cheapest_total_inner,
2323 cheapest_total_inner = (
Path *)
2325 cheapest_total_inner, extra->
sjinfo);
2326 Assert(cheapest_total_inner);
2330 cheapest_total_outer,
2331 cheapest_total_inner,
2335 if (cheapest_startup_outer != NULL &&
2336 cheapest_startup_outer != cheapest_total_outer)
2339 cheapest_startup_outer,
2340 cheapest_total_inner,
2357 if (cheapest_startup_outer != NULL)
2360 cheapest_startup_outer,
2361 cheapest_total_inner,
2388 if (outerpath == cheapest_startup_outer &&
2389 innerpath == cheapest_total_inner)
2415 Path *cheapest_partial_outer;
2416 Path *cheapest_partial_inner = NULL;
2417 Path *cheapest_safe_inner = NULL;
2419 cheapest_partial_outer =
2431 cheapest_partial_inner =
2434 cheapest_partial_outer,
2435 cheapest_partial_inner,
2436 hashclauses, jointype, extra,
2453 cheapest_safe_inner = NULL;
2455 cheapest_safe_inner = cheapest_total_inner;
2457 cheapest_safe_inner =
2460 if (cheapest_safe_inner != NULL)
2462 cheapest_partial_outer,
2463 cheapest_safe_inner,
2464 hashclauses, jointype, extra,
2499 bool *mergejoin_allowed)
2503 bool have_nonmergeable_joinclause =
false;
2512 *mergejoin_allowed =
false;
2516 foreach(l, restrictlist)
2530 if (!restrictinfo->can_join ||
2531 restrictinfo->mergeopfamilies ==
NIL)
2540 have_nonmergeable_joinclause =
true;
2550 have_nonmergeable_joinclause =
true;
2564 if (!restrictinfo->outer_is_left &&
2567 have_nonmergeable_joinclause =
true;
2584 have_nonmergeable_joinclause =
true;
2588 result_list =
lappend(result_list, restrictinfo);
2599 *mergejoin_allowed = !have_nonmergeable_joinclause;
2602 *mergejoin_allowed =
true;
bool innerrel_is_unique(PlannerInfo *root, Relids joinrelids, Relids outerrelids, RelOptInfo *innerrel, JoinType jointype, List *restrictlist, bool force_cache)
Bitmapset * bms_difference(const Bitmapset *a, const Bitmapset *b)
Bitmapset * bms_intersect(const Bitmapset *a, const Bitmapset *b)
bool bms_equal(const Bitmapset *a, const Bitmapset *b)
bool bms_is_subset(const Bitmapset *a, const Bitmapset *b)
void bms_free(Bitmapset *a)
bool bms_is_member(int x, const Bitmapset *a)
Bitmapset * bms_add_members(Bitmapset *a, const Bitmapset *b)
BMS_Membership bms_membership(const Bitmapset *a)
bool bms_overlap(const Bitmapset *a, const Bitmapset *b)
Bitmapset * bms_join(Bitmapset *a, Bitmapset *b)
bool bms_nonempty_difference(const Bitmapset *a, const Bitmapset *b)
#define OidIsValid(objectId)
bool contain_volatile_functions(Node *clause)
void compute_semi_anti_join_factors(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, SpecialJoinInfo *sjinfo, List *restrictlist, SemiAntiJoinFactors *semifactors)
void initial_cost_hashjoin(PlannerInfo *root, JoinCostWorkspace *workspace, JoinType jointype, List *hashclauses, Path *outer_path, Path *inner_path, JoinPathExtraData *extra, bool parallel_hash)
void initial_cost_mergejoin(PlannerInfo *root, JoinCostWorkspace *workspace, JoinType jointype, List *mergeclauses, Path *outer_path, Path *inner_path, List *outersortkeys, List *innersortkeys, int outer_presorted_keys, JoinPathExtraData *extra)
void initial_cost_nestloop(PlannerInfo *root, JoinCostWorkspace *workspace, JoinType jointype, Path *outer_path, Path *inner_path, JoinPathExtraData *extra)
bool enable_parallel_hash
bool ExecMaterializesOutput(NodeTag plantype)
Assert(PointerIsAligned(start, uint64))
if(TABLE==NULL||TABLE_index==NULL)
static void try_hashjoin_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *hashclauses, JoinType jointype, JoinPathExtraData *extra)
static List * select_mergejoin_clauses(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, List *restrictlist, JoinType jointype, bool *mergejoin_allowed)
static void sort_inner_and_outer(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra)
static List * extract_lateral_vars_from_PHVs(PlannerInfo *root, Relids innerrelids)
void add_paths_to_joinrel(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, SpecialJoinInfo *sjinfo, List *restrictlist)
static bool paraminfo_get_equal_hashops(PlannerInfo *root, ParamPathInfo *param_info, RelOptInfo *outerrel, RelOptInfo *innerrel, List *ph_lateral_vars, List **param_exprs, List **operators, bool *binary_mode)
static void try_nestloop_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *pathkeys, JoinType jointype, JoinPathExtraData *extra)
static Path * get_memoize_path(PlannerInfo *root, RelOptInfo *innerrel, RelOptInfo *outerrel, Path *inner_path, Path *outer_path, JoinType jointype, JoinPathExtraData *extra)
static void consider_parallel_nestloop(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra)
#define PATH_PARAM_BY_PARENT(path, rel)
set_join_pathlist_hook_type set_join_pathlist_hook
static void try_mergejoin_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *pathkeys, List *mergeclauses, List *outersortkeys, List *innersortkeys, JoinType jointype, JoinPathExtraData *extra, bool is_partial)
static void consider_parallel_mergejoin(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra, Path *inner_cheapest_total)
static void generate_mergejoin_paths(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *innerrel, Path *outerpath, JoinType jointype, JoinPathExtraData *extra, bool useallclauses, Path *inner_cheapest_total, List *merge_pathkeys, bool is_partial)
static void hash_inner_and_outer(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra)
static void try_partial_mergejoin_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *pathkeys, List *mergeclauses, List *outersortkeys, List *innersortkeys, JoinType jointype, JoinPathExtraData *extra)
#define PATH_PARAM_BY_REL(path, rel)
static bool allow_star_schema_join(PlannerInfo *root, Relids outerrelids, Relids inner_paramrels)
static void try_partial_hashjoin_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *hashclauses, JoinType jointype, JoinPathExtraData *extra, bool parallel_hash)
static void match_unsorted_outer(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra)
static void try_partial_nestloop_path(PlannerInfo *root, RelOptInfo *joinrel, Path *outer_path, Path *inner_path, List *pathkeys, JoinType jointype, JoinPathExtraData *extra)
bool have_dangerous_phv(PlannerInfo *root, Relids outer_relids, Relids inner_params)
List * lappend(List *list, void *datum)
List * list_delete_nth_cell(List *list, int n)
List * list_concat(List *list1, const List *list2)
List * list_copy(const List *oldlist)
List * lappend_oid(List *list, Oid datum)
List * lcons(void *datum, List *list)
void list_free(List *list)
List * list_truncate(List *list, int new_size)
bool list_member(const List *list, const void *datum)
Oid get_commutator(Oid opno)
Oid exprType(const Node *expr)
#define IsA(nodeptr, _type_)
#define IS_OUTER_JOIN(jointype)
#define castNode(_type_, nodeptr)
Path * get_cheapest_path_for_pathkeys(List *paths, List *pathkeys, Relids required_outer, CostSelector cost_criterion, bool require_parallel_safe)
bool pathkeys_count_contained_in(List *keys1, List *keys2, int *n_common)
List * make_inner_pathkeys_for_merge(PlannerInfo *root, List *mergeclauses, List *outer_pathkeys)
List * find_mergeclauses_for_outer_pathkeys(PlannerInfo *root, List *pathkeys, List *restrictinfos)
void update_mergeclause_eclasses(PlannerInfo *root, RestrictInfo *restrictinfo)
List * trim_mergeclauses_for_inner_pathkeys(PlannerInfo *root, List *mergeclauses, List *pathkeys)
List * select_outer_pathkeys_for_merge(PlannerInfo *root, List *mergeclauses, RelOptInfo *joinrel)
bool pathkeys_contained_in(List *keys1, List *keys2)
List * build_join_pathkeys(PlannerInfo *root, RelOptInfo *joinrel, JoinType jointype, List *outer_pathkeys)
Path * get_cheapest_parallel_safe_total_inner(List *paths)
Relids calc_non_nestloop_required_outer(Path *outer_path, Path *inner_path)
bool path_is_reparameterizable_by_child(Path *path, RelOptInfo *child_rel)
Relids calc_nestloop_required_outer(Relids outerrelids, Relids outer_paramrels, Relids innerrelids, Relids inner_paramrels)
MemoizePath * create_memoize_path(PlannerInfo *root, RelOptInfo *rel, Path *subpath, List *param_exprs, List *hash_operators, bool singlerow, bool binary_mode, double calls)
HashPath * create_hashjoin_path(PlannerInfo *root, RelOptInfo *joinrel, JoinType jointype, JoinCostWorkspace *workspace, JoinPathExtraData *extra, Path *outer_path, Path *inner_path, bool parallel_hash, List *restrict_clauses, Relids required_outer, List *hashclauses)
UniquePath * create_unique_path(PlannerInfo *root, RelOptInfo *rel, Path *subpath, SpecialJoinInfo *sjinfo)
void add_partial_path(RelOptInfo *parent_rel, Path *new_path)
MaterialPath * create_material_path(RelOptInfo *rel, Path *subpath)
void add_path(RelOptInfo *parent_rel, Path *new_path)
int compare_path_costs(Path *path1, Path *path2, CostSelector criterion)
bool add_path_precheck(RelOptInfo *parent_rel, int disabled_nodes, Cost startup_cost, Cost total_cost, List *pathkeys, Relids required_outer)
bool add_partial_path_precheck(RelOptInfo *parent_rel, int disabled_nodes, Cost total_cost, List *pathkeys)
MergePath * create_mergejoin_path(PlannerInfo *root, RelOptInfo *joinrel, JoinType jointype, JoinCostWorkspace *workspace, JoinPathExtraData *extra, Path *outer_path, Path *inner_path, List *restrict_clauses, List *pathkeys, Relids required_outer, List *mergeclauses, List *outersortkeys, List *innersortkeys, int outer_presorted_keys)
NestPath * create_nestloop_path(PlannerInfo *root, RelOptInfo *joinrel, JoinType jointype, JoinCostWorkspace *workspace, JoinPathExtraData *extra, Path *outer_path, Path *inner_path, List *restrict_clauses, List *pathkeys, Relids required_outer)
#define EC_MUST_BE_REDUNDANT(eclass)
#define RINFO_IS_PUSHED_DOWN(rinfo, joinrelids)
#define PATH_REQ_OUTER(path)
void(* set_join_pathlist_hook_type)(PlannerInfo *root, RelOptInfo *joinrel, RelOptInfo *outerrel, RelOptInfo *innerrel, JoinType jointype, JoinPathExtraData *extra)
static int list_length(const List *l)
#define foreach_current_index(var_or_cell)
#define foreach_node(type, var, lst)
static ListCell * list_head(const List *l)
PlaceHolderInfo * find_placeholder_info(PlannerInfo *root, PlaceHolderVar *phv)
static bool clause_sides_match_join(RestrictInfo *rinfo, Relids outerrelids, Relids innerrelids)
struct PathTarget * reltarget
List * cheapest_parameterized_paths
struct Path * cheapest_startup_path
struct Path * cheapest_total_path
TypeCacheEntry * lookup_type_cache(Oid type_id, int flags)
#define TYPECACHE_HASH_PROC
Relids pull_varnos(PlannerInfo *root, Node *node)
List * pull_vars_of_level(Node *node, int levelsup)