Commit e006a24a authored by Tom Lane's avatar Tom Lane

Implement SEMI and ANTI joins in the planner and executor. (Semijoins replace

the old JOIN_IN code, but antijoins are new functionality.)  Teach the planner
to convert appropriate EXISTS and NOT EXISTS subqueries into semi and anti
joins respectively.  Also, LEFT JOINs with suitable upper-level IS NULL
filters are recognized as being anti joins.  Unify the InClauseInfo and
OuterJoinInfo infrastructure into "SpecialJoinInfo".  With that change,
it becomes possible to associate a SpecialJoinInfo with every join attempt,
which permits some cleanup of join selectivity estimation.  That needs to be
taken much further than this patch does, but the next step is to change the
API for oprjoin selectivity functions, which seems like material for a
separate patch.  So for the moment the output size estimates for semi and
especially anti joins are quite bogus.
parent ef1c807c
<!-- $PostgreSQL: pgsql/doc/src/sgml/indexam.sgml,v 2.26 2008/04/14 17:05:32 tgl Exp $ --> <!-- $PostgreSQL: pgsql/doc/src/sgml/indexam.sgml,v 2.27 2008/08/14 18:47:58 tgl Exp $ -->
<chapter id="indexam"> <chapter id="indexam">
<title>Index Access Method Interface Definition</title> <title>Index Access Method Interface Definition</title>
...@@ -879,7 +879,8 @@ amcostestimate (PlannerInfo *root, ...@@ -879,7 +879,8 @@ amcostestimate (PlannerInfo *root,
<programlisting> <programlisting>
*indexSelectivity = clauselist_selectivity(root, indexQuals, *indexSelectivity = clauselist_selectivity(root, indexQuals,
index-&gt;rel-&gt;relid, JOIN_INNER); index-&gt;rel-&gt;relid,
JOIN_INNER, NULL);
</programlisting> </programlisting>
</para> </para>
</step> </step>
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1994-5, Regents of the University of California * Portions Copyright (c) 1994-5, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/explain.c,v 1.176 2008/08/07 03:04:03 tgl Exp $ * $PostgreSQL: pgsql/src/backend/commands/explain.c,v 1.177 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -450,8 +450,11 @@ explain_outNode(StringInfo str, ...@@ -450,8 +450,11 @@ explain_outNode(StringInfo str,
case JOIN_RIGHT: case JOIN_RIGHT:
pname = "Nested Loop Right Join"; pname = "Nested Loop Right Join";
break; break;
case JOIN_IN: case JOIN_SEMI:
pname = "Nested Loop IN Join"; pname = "Nested Loop Semi Join";
break;
case JOIN_ANTI:
pname = "Nested Loop Anti Join";
break; break;
default: default:
pname = "Nested Loop ??? Join"; pname = "Nested Loop ??? Join";
...@@ -473,8 +476,11 @@ explain_outNode(StringInfo str, ...@@ -473,8 +476,11 @@ explain_outNode(StringInfo str,
case JOIN_RIGHT: case JOIN_RIGHT:
pname = "Merge Right Join"; pname = "Merge Right Join";
break; break;
case JOIN_IN: case JOIN_SEMI:
pname = "Merge IN Join"; pname = "Merge Semi Join";
break;
case JOIN_ANTI:
pname = "Merge Anti Join";
break; break;
default: default:
pname = "Merge ??? Join"; pname = "Merge ??? Join";
...@@ -496,8 +502,11 @@ explain_outNode(StringInfo str, ...@@ -496,8 +502,11 @@ explain_outNode(StringInfo str,
case JOIN_RIGHT: case JOIN_RIGHT:
pname = "Hash Right Join"; pname = "Hash Right Join";
break; break;
case JOIN_IN: case JOIN_SEMI:
pname = "Hash IN Join"; pname = "Hash Semi Join";
break;
case JOIN_ANTI:
pname = "Hash Anti Join";
break; break;
default: default:
pname = "Hash ??? Join"; pname = "Hash ??? Join";
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeHashjoin.c,v 1.93 2008/01/01 19:45:49 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeHashjoin.c,v 1.94 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -22,6 +22,9 @@ ...@@ -22,6 +22,9 @@
#include "utils/memutils.h" #include "utils/memutils.h"
/* Returns true for JOIN_LEFT and JOIN_ANTI jointypes */
#define HASHJOIN_IS_OUTER(hjstate) ((hjstate)->hj_NullInnerTupleSlot != NULL)
static TupleTableSlot *ExecHashJoinOuterGetTuple(PlanState *outerNode, static TupleTableSlot *ExecHashJoinOuterGetTuple(PlanState *outerNode,
HashJoinState *hjstate, HashJoinState *hjstate,
uint32 *hashvalue); uint32 *hashvalue);
...@@ -89,14 +92,6 @@ ExecHashJoin(HashJoinState *node) ...@@ -89,14 +92,6 @@ ExecHashJoin(HashJoinState *node)
node->js.ps.ps_TupFromTlist = false; node->js.ps.ps_TupFromTlist = false;
} }
/*
* If we're doing an IN join, we want to return at most one row per outer
* tuple; so we can stop scanning the inner scan if we matched on the
* previous try.
*/
if (node->js.jointype == JOIN_IN && node->hj_MatchedOuter)
node->hj_NeedNewOuter = true;
/* /*
* Reset per-tuple memory context to free any expression evaluation * Reset per-tuple memory context to free any expression evaluation
* storage allocated in the previous tuple cycle. Note this can't happen * storage allocated in the previous tuple cycle. Note this can't happen
...@@ -129,7 +124,7 @@ ExecHashJoin(HashJoinState *node) ...@@ -129,7 +124,7 @@ ExecHashJoin(HashJoinState *node)
* outer plan node. If we succeed, we have to stash it away for later * outer plan node. If we succeed, we have to stash it away for later
* consumption by ExecHashJoinOuterGetTuple. * consumption by ExecHashJoinOuterGetTuple.
*/ */
if (node->js.jointype == JOIN_LEFT || if (HASHJOIN_IS_OUTER(node) ||
(outerNode->plan->startup_cost < hashNode->ps.plan->total_cost && (outerNode->plan->startup_cost < hashNode->ps.plan->total_cost &&
!node->hj_OuterNotEmpty)) !node->hj_OuterNotEmpty))
{ {
...@@ -162,7 +157,7 @@ ExecHashJoin(HashJoinState *node) ...@@ -162,7 +157,7 @@ ExecHashJoin(HashJoinState *node)
* If the inner relation is completely empty, and we're not doing an * If the inner relation is completely empty, and we're not doing an
* outer join, we can quit without scanning the outer relation. * outer join, we can quit without scanning the outer relation.
*/ */
if (hashtable->totalTuples == 0 && node->js.jointype != JOIN_LEFT) if (hashtable->totalTuples == 0 && !HASHJOIN_IS_OUTER(node))
return NULL; return NULL;
/* /*
...@@ -263,6 +258,21 @@ ExecHashJoin(HashJoinState *node) ...@@ -263,6 +258,21 @@ ExecHashJoin(HashJoinState *node)
{ {
node->hj_MatchedOuter = true; node->hj_MatchedOuter = true;
/* In an antijoin, we never return a matched tuple */
if (node->js.jointype == JOIN_ANTI)
{
node->hj_NeedNewOuter = true;
break; /* out of loop over hash bucket */
}
else
{
/*
* In a semijoin, we'll consider returning the first match,
* but after that we're done with this outer tuple.
*/
if (node->js.jointype == JOIN_SEMI)
node->hj_NeedNewOuter = true;
if (otherqual == NIL || ExecQual(otherqual, econtext, false)) if (otherqual == NIL || ExecQual(otherqual, econtext, false))
{ {
TupleTableSlot *result; TupleTableSlot *result;
...@@ -278,11 +288,10 @@ ExecHashJoin(HashJoinState *node) ...@@ -278,11 +288,10 @@ ExecHashJoin(HashJoinState *node)
} }
/* /*
* If we didn't return a tuple, may need to set NeedNewOuter * If semijoin and we didn't return the tuple, we're still
* done with this outer tuple.
*/ */
if (node->js.jointype == JOIN_IN) if (node->js.jointype == JOIN_SEMI)
{
node->hj_NeedNewOuter = true;
break; /* out of loop over hash bucket */ break; /* out of loop over hash bucket */
} }
} }
...@@ -296,7 +305,7 @@ ExecHashJoin(HashJoinState *node) ...@@ -296,7 +305,7 @@ ExecHashJoin(HashJoinState *node)
node->hj_NeedNewOuter = true; node->hj_NeedNewOuter = true;
if (!node->hj_MatchedOuter && if (!node->hj_MatchedOuter &&
node->js.jointype == JOIN_LEFT) HASHJOIN_IS_OUTER(node))
{ {
/* /*
* We are doing an outer join and there were no join matches for * We are doing an outer join and there were no join matches for
...@@ -305,7 +314,7 @@ ExecHashJoin(HashJoinState *node) ...@@ -305,7 +314,7 @@ ExecHashJoin(HashJoinState *node)
*/ */
econtext->ecxt_innertuple = node->hj_NullInnerTupleSlot; econtext->ecxt_innertuple = node->hj_NullInnerTupleSlot;
if (ExecQual(otherqual, econtext, false)) if (otherqual == NIL || ExecQual(otherqual, econtext, false))
{ {
/* /*
* qualification was satisfied so we project and return the * qualification was satisfied so we project and return the
...@@ -398,12 +407,14 @@ ExecInitHashJoin(HashJoin *node, EState *estate, int eflags) ...@@ -398,12 +407,14 @@ ExecInitHashJoin(HashJoin *node, EState *estate, int eflags)
ExecInitResultTupleSlot(estate, &hjstate->js.ps); ExecInitResultTupleSlot(estate, &hjstate->js.ps);
hjstate->hj_OuterTupleSlot = ExecInitExtraTupleSlot(estate); hjstate->hj_OuterTupleSlot = ExecInitExtraTupleSlot(estate);
/* note: HASHJOIN_IS_OUTER macro depends on this initialization */
switch (node->join.jointype) switch (node->join.jointype)
{ {
case JOIN_INNER: case JOIN_INNER:
case JOIN_IN: case JOIN_SEMI:
break; break;
case JOIN_LEFT: case JOIN_LEFT:
case JOIN_ANTI:
hjstate->hj_NullInnerTupleSlot = hjstate->hj_NullInnerTupleSlot =
ExecInitNullTupleSlot(estate, ExecInitNullTupleSlot(estate,
ExecGetResultType(innerPlanState(hjstate))); ExecGetResultType(innerPlanState(hjstate)));
...@@ -570,7 +581,7 @@ ExecHashJoinOuterGetTuple(PlanState *outerNode, ...@@ -570,7 +581,7 @@ ExecHashJoinOuterGetTuple(PlanState *outerNode,
if (ExecHashGetHashValue(hashtable, econtext, if (ExecHashGetHashValue(hashtable, econtext,
hjstate->hj_OuterHashKeys, hjstate->hj_OuterHashKeys,
true, /* outer tuple */ true, /* outer tuple */
(hjstate->js.jointype == JOIN_LEFT), HASHJOIN_IS_OUTER(hjstate),
hashvalue)) hashvalue))
{ {
/* remember outer relation is not empty for possible rescan */ /* remember outer relation is not empty for possible rescan */
...@@ -650,7 +661,7 @@ start_over: ...@@ -650,7 +661,7 @@ start_over:
* sides. We can sometimes skip over batches that are empty on only one * sides. We can sometimes skip over batches that are empty on only one
* side, but there are exceptions: * side, but there are exceptions:
* *
* 1. In a LEFT JOIN, we have to process outer batches even if the inner * 1. In an outer join, we have to process outer batches even if the inner
* batch is empty. * batch is empty.
* *
* 2. If we have increased nbatch since the initial estimate, we have to * 2. If we have increased nbatch since the initial estimate, we have to
...@@ -667,7 +678,7 @@ start_over: ...@@ -667,7 +678,7 @@ start_over:
hashtable->innerBatchFile[curbatch] == NULL)) hashtable->innerBatchFile[curbatch] == NULL))
{ {
if (hashtable->outerBatchFile[curbatch] && if (hashtable->outerBatchFile[curbatch] &&
hjstate->js.jointype == JOIN_LEFT) HASHJOIN_IS_OUTER(hjstate))
break; /* must process due to rule 1 */ break; /* must process due to rule 1 */
if (hashtable->innerBatchFile[curbatch] && if (hashtable->innerBatchFile[curbatch] &&
nbatch != hashtable->nbatch_original) nbatch != hashtable->nbatch_original)
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeMergejoin.c,v 1.91 2008/04/13 20:51:20 tgl Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeMergejoin.c,v 1.92 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -757,7 +757,7 @@ ExecMergeJoin(MergeJoinState *node) ...@@ -757,7 +757,7 @@ ExecMergeJoin(MergeJoinState *node)
innerTupleSlot = node->mj_InnerTupleSlot; innerTupleSlot = node->mj_InnerTupleSlot;
econtext->ecxt_innertuple = innerTupleSlot; econtext->ecxt_innertuple = innerTupleSlot;
if (node->js.jointype == JOIN_IN && if (node->js.jointype == JOIN_SEMI &&
node->mj_MatchedOuter) node->mj_MatchedOuter)
qualResult = false; qualResult = false;
else else
...@@ -772,6 +772,10 @@ ExecMergeJoin(MergeJoinState *node) ...@@ -772,6 +772,10 @@ ExecMergeJoin(MergeJoinState *node)
node->mj_MatchedOuter = true; node->mj_MatchedOuter = true;
node->mj_MatchedInner = true; node->mj_MatchedInner = true;
/* In an antijoin, we never return a matched tuple */
if (node->js.jointype == JOIN_ANTI)
break;
qualResult = (otherqual == NIL || qualResult = (otherqual == NIL ||
ExecQual(otherqual, econtext, false)); ExecQual(otherqual, econtext, false));
MJ_DEBUG_QUAL(otherqual, qualResult); MJ_DEBUG_QUAL(otherqual, qualResult);
...@@ -1472,11 +1476,12 @@ ExecInitMergeJoin(MergeJoin *node, EState *estate, int eflags) ...@@ -1472,11 +1476,12 @@ ExecInitMergeJoin(MergeJoin *node, EState *estate, int eflags)
switch (node->join.jointype) switch (node->join.jointype)
{ {
case JOIN_INNER: case JOIN_INNER:
case JOIN_IN: case JOIN_SEMI:
mergestate->mj_FillOuter = false; mergestate->mj_FillOuter = false;
mergestate->mj_FillInner = false; mergestate->mj_FillInner = false;
break; break;
case JOIN_LEFT: case JOIN_LEFT:
case JOIN_ANTI:
mergestate->mj_FillOuter = true; mergestate->mj_FillOuter = true;
mergestate->mj_FillInner = false; mergestate->mj_FillInner = false;
mergestate->mj_NullInnerTupleSlot = mergestate->mj_NullInnerTupleSlot =
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeNestloop.c,v 1.46 2008/01/01 19:45:49 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeNestloop.c,v 1.47 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -101,15 +101,6 @@ ExecNestLoop(NestLoopState *node) ...@@ -101,15 +101,6 @@ ExecNestLoop(NestLoopState *node)
node->js.ps.ps_TupFromTlist = false; node->js.ps.ps_TupFromTlist = false;
} }
/*
* If we're doing an IN join, we want to return at most one row per outer
* tuple; so we can stop scanning the inner scan if we matched on the
* previous try.
*/
if (node->js.jointype == JOIN_IN &&
node->nl_MatchedOuter)
node->nl_NeedNewOuter = true;
/* /*
* Reset per-tuple memory context to free any expression evaluation * Reset per-tuple memory context to free any expression evaluation
* storage allocated in the previous tuple cycle. Note this can't happen * storage allocated in the previous tuple cycle. Note this can't happen
...@@ -177,7 +168,8 @@ ExecNestLoop(NestLoopState *node) ...@@ -177,7 +168,8 @@ ExecNestLoop(NestLoopState *node)
node->nl_NeedNewOuter = true; node->nl_NeedNewOuter = true;
if (!node->nl_MatchedOuter && if (!node->nl_MatchedOuter &&
node->js.jointype == JOIN_LEFT) (node->js.jointype == JOIN_LEFT ||
node->js.jointype == JOIN_ANTI))
{ {
/* /*
* We are doing an outer join and there were no join matches * We are doing an outer join and there were no join matches
...@@ -189,7 +181,7 @@ ExecNestLoop(NestLoopState *node) ...@@ -189,7 +181,7 @@ ExecNestLoop(NestLoopState *node)
ENL1_printf("testing qualification for outer-join tuple"); ENL1_printf("testing qualification for outer-join tuple");
if (ExecQual(otherqual, econtext, false)) if (otherqual == NIL || ExecQual(otherqual, econtext, false))
{ {
/* /*
* qualification was satisfied so we project and return * qualification was satisfied so we project and return
...@@ -232,11 +224,23 @@ ExecNestLoop(NestLoopState *node) ...@@ -232,11 +224,23 @@ ExecNestLoop(NestLoopState *node)
{ {
node->nl_MatchedOuter = true; node->nl_MatchedOuter = true;
/* In an antijoin, we never return a matched tuple */
if (node->js.jointype == JOIN_ANTI)
node->nl_NeedNewOuter = true;
else
{
/*
* In a semijoin, we'll consider returning the first match,
* but after that we're done with this outer tuple.
*/
if (node->js.jointype == JOIN_SEMI)
node->nl_NeedNewOuter = true;
if (otherqual == NIL || ExecQual(otherqual, econtext, false)) if (otherqual == NIL || ExecQual(otherqual, econtext, false))
{ {
/* /*
* qualification was satisfied so we project and return the * qualification was satisfied so we project and return
* slot containing the result tuple using ExecProject(). * the slot containing the result tuple using
* ExecProject().
*/ */
TupleTableSlot *result; TupleTableSlot *result;
ExprDoneCond isDone; ExprDoneCond isDone;
...@@ -252,10 +256,7 @@ ExecNestLoop(NestLoopState *node) ...@@ -252,10 +256,7 @@ ExecNestLoop(NestLoopState *node)
return result; return result;
} }
} }
}
/* If we didn't return a tuple, may need to set NeedNewOuter */
if (node->js.jointype == JOIN_IN)
node->nl_NeedNewOuter = true;
} }
/* /*
...@@ -333,9 +334,10 @@ ExecInitNestLoop(NestLoop *node, EState *estate, int eflags) ...@@ -333,9 +334,10 @@ ExecInitNestLoop(NestLoop *node, EState *estate, int eflags)
switch (node->join.jointype) switch (node->join.jointype)
{ {
case JOIN_INNER: case JOIN_INNER:
case JOIN_IN: case JOIN_SEMI:
break; break;
case JOIN_LEFT: case JOIN_LEFT:
case JOIN_ANTI:
nlstate->nl_NullInnerTupleSlot = nlstate->nl_NullInnerTupleSlot =
ExecInitNullTupleSlot(estate, ExecInitNullTupleSlot(estate,
ExecGetResultType(innerPlanState(nlstate))); ExecGetResultType(innerPlanState(nlstate)));
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/copyfuncs.c,v 1.399 2008/08/07 19:35:02 tgl Exp $ * $PostgreSQL: pgsql/src/backend/nodes/copyfuncs.c,v 1.400 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1444,36 +1444,37 @@ _copyRestrictInfo(RestrictInfo *from) ...@@ -1444,36 +1444,37 @@ _copyRestrictInfo(RestrictInfo *from)
} }
/* /*
* _copyOuterJoinInfo * _copyFlattenedSubLink
*/ */
static OuterJoinInfo * static FlattenedSubLink *
_copyOuterJoinInfo(OuterJoinInfo *from) _copyFlattenedSubLink(FlattenedSubLink *from)
{ {
OuterJoinInfo *newnode = makeNode(OuterJoinInfo); FlattenedSubLink *newnode = makeNode(FlattenedSubLink);
COPY_BITMAPSET_FIELD(min_lefthand); COPY_SCALAR_FIELD(jointype);
COPY_BITMAPSET_FIELD(min_righthand); COPY_BITMAPSET_FIELD(lefthand);
COPY_BITMAPSET_FIELD(syn_lefthand); COPY_BITMAPSET_FIELD(righthand);
COPY_BITMAPSET_FIELD(syn_righthand); COPY_NODE_FIELD(quals);
COPY_SCALAR_FIELD(is_full_join);
COPY_SCALAR_FIELD(lhs_strict);
COPY_SCALAR_FIELD(delay_upper_joins);
return newnode; return newnode;
} }
/* /*
* _copyInClauseInfo * _copySpecialJoinInfo
*/ */
static InClauseInfo * static SpecialJoinInfo *
_copyInClauseInfo(InClauseInfo *from) _copySpecialJoinInfo(SpecialJoinInfo *from)
{ {
InClauseInfo *newnode = makeNode(InClauseInfo); SpecialJoinInfo *newnode = makeNode(SpecialJoinInfo);
COPY_BITMAPSET_FIELD(lefthand); COPY_BITMAPSET_FIELD(min_lefthand);
COPY_BITMAPSET_FIELD(righthand); COPY_BITMAPSET_FIELD(min_righthand);
COPY_NODE_FIELD(sub_targetlist); COPY_BITMAPSET_FIELD(syn_lefthand);
COPY_NODE_FIELD(in_operators); COPY_BITMAPSET_FIELD(syn_righthand);
COPY_SCALAR_FIELD(jointype);
COPY_SCALAR_FIELD(lhs_strict);
COPY_SCALAR_FIELD(delay_upper_joins);
COPY_NODE_FIELD(join_quals);
return newnode; return newnode;
} }
...@@ -3233,11 +3234,11 @@ copyObject(void *from) ...@@ -3233,11 +3234,11 @@ copyObject(void *from)
case T_RestrictInfo: case T_RestrictInfo:
retval = _copyRestrictInfo(from); retval = _copyRestrictInfo(from);
break; break;
case T_OuterJoinInfo: case T_FlattenedSubLink:
retval = _copyOuterJoinInfo(from); retval = _copyFlattenedSubLink(from);
break; break;
case T_InClauseInfo: case T_SpecialJoinInfo:
retval = _copyInClauseInfo(from); retval = _copySpecialJoinInfo(from);
break; break;
case T_AppendRelInfo: case T_AppendRelInfo:
retval = _copyAppendRelInfo(from); retval = _copyAppendRelInfo(from);
......
...@@ -18,7 +18,7 @@ ...@@ -18,7 +18,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/equalfuncs.c,v 1.326 2008/08/07 01:11:47 tgl Exp $ * $PostgreSQL: pgsql/src/backend/nodes/equalfuncs.c,v 1.327 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -702,26 +702,27 @@ _equalRestrictInfo(RestrictInfo *a, RestrictInfo *b) ...@@ -702,26 +702,27 @@ _equalRestrictInfo(RestrictInfo *a, RestrictInfo *b)
} }
static bool static bool
_equalOuterJoinInfo(OuterJoinInfo *a, OuterJoinInfo *b) _equalFlattenedSubLink(FlattenedSubLink *a, FlattenedSubLink *b)
{ {
COMPARE_BITMAPSET_FIELD(min_lefthand); COMPARE_SCALAR_FIELD(jointype);
COMPARE_BITMAPSET_FIELD(min_righthand); COMPARE_BITMAPSET_FIELD(lefthand);
COMPARE_BITMAPSET_FIELD(syn_lefthand); COMPARE_BITMAPSET_FIELD(righthand);
COMPARE_BITMAPSET_FIELD(syn_righthand); COMPARE_NODE_FIELD(quals);
COMPARE_SCALAR_FIELD(is_full_join);
COMPARE_SCALAR_FIELD(lhs_strict);
COMPARE_SCALAR_FIELD(delay_upper_joins);
return true; return true;
} }
static bool static bool
_equalInClauseInfo(InClauseInfo *a, InClauseInfo *b) _equalSpecialJoinInfo(SpecialJoinInfo *a, SpecialJoinInfo *b)
{ {
COMPARE_BITMAPSET_FIELD(lefthand); COMPARE_BITMAPSET_FIELD(min_lefthand);
COMPARE_BITMAPSET_FIELD(righthand); COMPARE_BITMAPSET_FIELD(min_righthand);
COMPARE_NODE_FIELD(sub_targetlist); COMPARE_BITMAPSET_FIELD(syn_lefthand);
COMPARE_NODE_FIELD(in_operators); COMPARE_BITMAPSET_FIELD(syn_righthand);
COMPARE_SCALAR_FIELD(jointype);
COMPARE_SCALAR_FIELD(lhs_strict);
COMPARE_SCALAR_FIELD(delay_upper_joins);
COMPARE_NODE_FIELD(join_quals);
return true; return true;
} }
...@@ -2185,11 +2186,11 @@ equal(void *a, void *b) ...@@ -2185,11 +2186,11 @@ equal(void *a, void *b)
case T_RestrictInfo: case T_RestrictInfo:
retval = _equalRestrictInfo(a, b); retval = _equalRestrictInfo(a, b);
break; break;
case T_OuterJoinInfo: case T_FlattenedSubLink:
retval = _equalOuterJoinInfo(a, b); retval = _equalFlattenedSubLink(a, b);
break; break;
case T_InClauseInfo: case T_SpecialJoinInfo:
retval = _equalInClauseInfo(a, b); retval = _equalSpecialJoinInfo(a, b);
break; break;
case T_AppendRelInfo: case T_AppendRelInfo:
retval = _equalAppendRelInfo(a, b); retval = _equalAppendRelInfo(a, b);
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/list.c,v 1.69 2008/01/01 19:45:50 momjian Exp $ * $PostgreSQL: pgsql/src/backend/nodes/list.c,v 1.70 2008/08/14 18:47:58 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -783,6 +783,42 @@ list_union_oid(List *list1, List *list2) ...@@ -783,6 +783,42 @@ list_union_oid(List *list1, List *list2)
return result; return result;
} }
/*
* Return a list that contains all the cells that are in both list1 and
* list2. The returned list is freshly allocated via palloc(), but the
* cells themselves point to the same objects as the cells of the
* input lists.
*
* Duplicate entries in list1 will not be suppressed, so it's only a true
* "intersection" if list1 is known unique beforehand.
*
* This variant works on lists of pointers, and determines list
* membership via equal(). Note that the list1 member will be pointed
* to in the result.
*/
List *
list_intersection(List *list1, List *list2)
{
List *result;
ListCell *cell;
if (list1 == NIL || list2 == NIL)
return NIL;
Assert(IsPointerList(list1));
Assert(IsPointerList(list2));
result = NIL;
foreach(cell, list1)
{
if (list_member(list2, lfirst(cell)))
result = lappend(result, lfirst(cell));
}
check_list_invariants(result);
return result;
}
/* /*
* Return a list that contains all the cells in list1 that are not in * Return a list that contains all the cells in list1 that are not in
* list2. The returned list is freshly allocated via palloc(), but the * list2. The returned list is freshly allocated via palloc(), but the
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.333 2008/08/07 19:35:02 tgl Exp $ * $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.334 2008/08/14 18:47:58 tgl Exp $
* *
* NOTES * NOTES
* Every node type that can appear in stored rules' parsetrees *must* * Every node type that can appear in stored rules' parsetrees *must*
...@@ -1267,6 +1267,8 @@ _outUniquePath(StringInfo str, UniquePath *node) ...@@ -1267,6 +1267,8 @@ _outUniquePath(StringInfo str, UniquePath *node)
WRITE_NODE_FIELD(subpath); WRITE_NODE_FIELD(subpath);
WRITE_ENUM_FIELD(umethod, UniquePathMethod); WRITE_ENUM_FIELD(umethod, UniquePathMethod);
WRITE_NODE_FIELD(in_operators);
WRITE_NODE_FIELD(uniq_exprs);
WRITE_FLOAT_FIELD(rows, "%.0f"); WRITE_FLOAT_FIELD(rows, "%.0f");
} }
...@@ -1332,8 +1334,7 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node) ...@@ -1332,8 +1334,7 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node)
WRITE_NODE_FIELD(left_join_clauses); WRITE_NODE_FIELD(left_join_clauses);
WRITE_NODE_FIELD(right_join_clauses); WRITE_NODE_FIELD(right_join_clauses);
WRITE_NODE_FIELD(full_join_clauses); WRITE_NODE_FIELD(full_join_clauses);
WRITE_NODE_FIELD(oj_info_list); WRITE_NODE_FIELD(join_info_list);
WRITE_NODE_FIELD(in_info_list);
WRITE_NODE_FIELD(append_rel_list); WRITE_NODE_FIELD(append_rel_list);
WRITE_NODE_FIELD(query_pathkeys); WRITE_NODE_FIELD(query_pathkeys);
WRITE_NODE_FIELD(group_pathkeys); WRITE_NODE_FIELD(group_pathkeys);
...@@ -1342,7 +1343,6 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node) ...@@ -1342,7 +1343,6 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node)
WRITE_FLOAT_FIELD(total_table_pages, "%.0f"); WRITE_FLOAT_FIELD(total_table_pages, "%.0f");
WRITE_FLOAT_FIELD(tuple_fraction, "%.4f"); WRITE_FLOAT_FIELD(tuple_fraction, "%.4f");
WRITE_BOOL_FIELD(hasJoinRTEs); WRITE_BOOL_FIELD(hasJoinRTEs);
WRITE_BOOL_FIELD(hasOuterJoins);
WRITE_BOOL_FIELD(hasHavingQual); WRITE_BOOL_FIELD(hasHavingQual);
WRITE_BOOL_FIELD(hasPseudoConstantQuals); WRITE_BOOL_FIELD(hasPseudoConstantQuals);
} }
...@@ -1479,28 +1479,29 @@ _outInnerIndexscanInfo(StringInfo str, InnerIndexscanInfo *node) ...@@ -1479,28 +1479,29 @@ _outInnerIndexscanInfo(StringInfo str, InnerIndexscanInfo *node)
} }
static void static void
_outOuterJoinInfo(StringInfo str, OuterJoinInfo *node) _outFlattenedSubLink(StringInfo str, FlattenedSubLink *node)
{
WRITE_NODE_TYPE("FLATTENEDSUBLINK");
WRITE_ENUM_FIELD(jointype, JoinType);
WRITE_BITMAPSET_FIELD(lefthand);
WRITE_BITMAPSET_FIELD(righthand);
WRITE_NODE_FIELD(quals);
}
static void
_outSpecialJoinInfo(StringInfo str, SpecialJoinInfo *node)
{ {
WRITE_NODE_TYPE("OUTERJOININFO"); WRITE_NODE_TYPE("SPECIALJOININFO");
WRITE_BITMAPSET_FIELD(min_lefthand); WRITE_BITMAPSET_FIELD(min_lefthand);
WRITE_BITMAPSET_FIELD(min_righthand); WRITE_BITMAPSET_FIELD(min_righthand);
WRITE_BITMAPSET_FIELD(syn_lefthand); WRITE_BITMAPSET_FIELD(syn_lefthand);
WRITE_BITMAPSET_FIELD(syn_righthand); WRITE_BITMAPSET_FIELD(syn_righthand);
WRITE_BOOL_FIELD(is_full_join); WRITE_ENUM_FIELD(jointype, JoinType);
WRITE_BOOL_FIELD(lhs_strict); WRITE_BOOL_FIELD(lhs_strict);
WRITE_BOOL_FIELD(delay_upper_joins); WRITE_BOOL_FIELD(delay_upper_joins);
} WRITE_NODE_FIELD(join_quals);
static void
_outInClauseInfo(StringInfo str, InClauseInfo *node)
{
WRITE_NODE_TYPE("INCLAUSEINFO");
WRITE_BITMAPSET_FIELD(lefthand);
WRITE_BITMAPSET_FIELD(righthand);
WRITE_NODE_FIELD(sub_targetlist);
WRITE_NODE_FIELD(in_operators);
} }
static void static void
...@@ -2352,11 +2353,11 @@ _outNode(StringInfo str, void *obj) ...@@ -2352,11 +2353,11 @@ _outNode(StringInfo str, void *obj)
case T_InnerIndexscanInfo: case T_InnerIndexscanInfo:
_outInnerIndexscanInfo(str, obj); _outInnerIndexscanInfo(str, obj);
break; break;
case T_OuterJoinInfo: case T_FlattenedSubLink:
_outOuterJoinInfo(str, obj); _outFlattenedSubLink(str, obj);
break; break;
case T_InClauseInfo: case T_SpecialJoinInfo:
_outInClauseInfo(str, obj); _outSpecialJoinInfo(str, obj);
break; break;
case T_AppendRelInfo: case T_AppendRelInfo:
_outAppendRelInfo(str, obj); _outAppendRelInfo(str, obj);
......
$PostgreSQL: pgsql/src/backend/optimizer/README,v 1.47 2008/08/02 21:31:59 tgl Exp $ $PostgreSQL: pgsql/src/backend/optimizer/README,v 1.48 2008/08/14 18:47:59 tgl Exp $
Optimizer Optimizer
========= =========
...@@ -114,9 +114,8 @@ no choice but to generate a clauseless Cartesian-product join; so we ...@@ -114,9 +114,8 @@ no choice but to generate a clauseless Cartesian-product join; so we
consider joining that rel to each other available rel. But in the presence consider joining that rel to each other available rel. But in the presence
of join clauses we will only consider joins that use available join of join clauses we will only consider joins that use available join
clauses. Note that join-order restrictions induced by outer joins and clauses. Note that join-order restrictions induced by outer joins and
IN clauses are treated as if they were real join clauses, to ensure that IN/EXISTS clauses are also checked, to ensure that we find a workable join
we find a workable join order in cases where those restrictions force a order in cases where those restrictions force a clauseless join to be done.)
clauseless join to be done.)
If we only had two relations in the list, we are done: we just pick If we only had two relations in the list, we are done: we just pick
the cheapest path for the join RelOptInfo. If we had more than two, we now the cheapest path for the join RelOptInfo. If we had more than two, we now
...@@ -174,9 +173,9 @@ for it or the cheapest path with the desired ordering (if that's cheaper ...@@ -174,9 +173,9 @@ for it or the cheapest path with the desired ordering (if that's cheaper
than applying a sort to the cheapest other path). than applying a sort to the cheapest other path).
If the query contains one-sided outer joins (LEFT or RIGHT joins), or If the query contains one-sided outer joins (LEFT or RIGHT joins), or
"IN (sub-select)" WHERE clauses that were converted to joins, then some of IN or EXISTS WHERE clauses that were converted to joins, then some of
the possible join orders may be illegal. These are excluded by having the possible join orders may be illegal. These are excluded by having
join_is_legal consult side lists of outer joins and IN joins to see join_is_legal consult a side list of such "special" joins to see
whether a proposed join is illegal. (The same consultation allows it whether a proposed join is illegal. (The same consultation allows it
to see which join style should be applied for a valid join, ie, to see which join style should be applied for a valid join, ie,
JOIN_INNER, JOIN_LEFT, etc.) JOIN_INNER, JOIN_LEFT, etc.)
...@@ -219,10 +218,10 @@ FULL JOIN ordering is enforced by not collapsing FULL JOIN nodes when ...@@ -219,10 +218,10 @@ FULL JOIN ordering is enforced by not collapsing FULL JOIN nodes when
translating the jointree to "joinlist" representation. LEFT and RIGHT translating the jointree to "joinlist" representation. LEFT and RIGHT
JOIN nodes are normally collapsed so that they participate fully in the JOIN nodes are normally collapsed so that they participate fully in the
join order search. To avoid generating illegal join orders, the planner join order search. To avoid generating illegal join orders, the planner
creates an OuterJoinInfo node for each outer join, and join_is_legal creates a SpecialJoinInfo node for each outer join, and join_is_legal
checks this list to decide if a proposed join is legal. checks this list to decide if a proposed join is legal.
What we store in OuterJoinInfo nodes are the minimum sets of Relids What we store in SpecialJoinInfo nodes are the minimum sets of Relids
required on each side of the join to form the outer join. Note that required on each side of the join to form the outer join. Note that
these are minimums; there's no explicit maximum, since joining other these are minimums; there's no explicit maximum, since joining other
rels to the OJ's syntactic rels may be legal. Per identities 1 and 2, rels to the OJ's syntactic rels may be legal. Per identities 1 and 2,
...@@ -273,7 +272,7 @@ planner() ...@@ -273,7 +272,7 @@ planner()
set up for recursive handling of subqueries set up for recursive handling of subqueries
do final cleanup after planning do final cleanup after planning
-subquery_planner() -subquery_planner()
pull up subqueries from rangetable, if possible pull up sublinks and subqueries from rangetable, if possible
canonicalize qual canonicalize qual
Attempt to simplify WHERE clause to the most useful form; this includes Attempt to simplify WHERE clause to the most useful form; this includes
flattening nested AND/ORs and detecting clauses that are duplicated in flattening nested AND/ORs and detecting clauses that are duplicated in
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/clausesel.c,v 1.90 2008/01/11 17:00:45 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/clausesel.c,v 1.91 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -94,7 +94,8 @@ Selectivity ...@@ -94,7 +94,8 @@ Selectivity
clauselist_selectivity(PlannerInfo *root, clauselist_selectivity(PlannerInfo *root,
List *clauses, List *clauses,
int varRelid, int varRelid,
JoinType jointype) JoinType jointype,
SpecialJoinInfo *sjinfo)
{ {
Selectivity s1 = 1.0; Selectivity s1 = 1.0;
RangeQueryClause *rqlist = NULL; RangeQueryClause *rqlist = NULL;
...@@ -106,7 +107,7 @@ clauselist_selectivity(PlannerInfo *root, ...@@ -106,7 +107,7 @@ clauselist_selectivity(PlannerInfo *root,
*/ */
if (list_length(clauses) == 1) if (list_length(clauses) == 1)
return clause_selectivity(root, (Node *) linitial(clauses), return clause_selectivity(root, (Node *) linitial(clauses),
varRelid, jointype); varRelid, jointype, sjinfo);
/* /*
* Initial scan over clauses. Anything that doesn't look like a potential * Initial scan over clauses. Anything that doesn't look like a potential
...@@ -120,7 +121,7 @@ clauselist_selectivity(PlannerInfo *root, ...@@ -120,7 +121,7 @@ clauselist_selectivity(PlannerInfo *root,
Selectivity s2; Selectivity s2;
/* Always compute the selectivity using clause_selectivity */ /* Always compute the selectivity using clause_selectivity */
s2 = clause_selectivity(root, clause, varRelid, jointype); s2 = clause_selectivity(root, clause, varRelid, jointype, sjinfo);
/* /*
* Check for being passed a RestrictInfo. * Check for being passed a RestrictInfo.
...@@ -227,9 +228,8 @@ clauselist_selectivity(PlannerInfo *root, ...@@ -227,9 +228,8 @@ clauselist_selectivity(PlannerInfo *root,
s2 = rqlist->hibound + rqlist->lobound - 1.0; s2 = rqlist->hibound + rqlist->lobound - 1.0;
/* Adjust for double-exclusion of NULLs */ /* Adjust for double-exclusion of NULLs */
/* HACK: disable nulltestsel's special outer-join logic */
s2 += nulltestsel(root, IS_NULL, rqlist->var, s2 += nulltestsel(root, IS_NULL, rqlist->var,
varRelid, JOIN_INNER); varRelid, jointype, sjinfo);
/* /*
* A zero or slightly negative s2 should be converted into a * A zero or slightly negative s2 should be converted into a
...@@ -420,13 +420,32 @@ bms_is_subset_singleton(const Bitmapset *s, int x) ...@@ -420,13 +420,32 @@ bms_is_subset_singleton(const Bitmapset *s, int x)
* is appropriate for ordinary join clauses and restriction clauses. * is appropriate for ordinary join clauses and restriction clauses.
* *
* jointype is the join type, if the clause is a join clause. Pass JOIN_INNER * jointype is the join type, if the clause is a join clause. Pass JOIN_INNER
* if the clause isn't a join clause or the context is uncertain. * if the clause isn't a join clause.
*
* sjinfo is NULL for a non-join clause, otherwise it provides additional
* context information about the join being performed. There are some
* special cases:
* 1. For a special (not INNER) join, sjinfo is always a member of
* root->join_info_list.
* 2. For an INNER join, sjinfo is just a transient struct, and only the
* relids and jointype fields in it can be trusted.
* 3. XXX sjinfo might be NULL even though it really is a join. This case
* will go away soon, but fixing it requires API changes for oprjoin and
* amcostestimate functions.
* It is possible for jointype to be different from sjinfo->jointype.
* This indicates we are considering a variant join: either with
* the LHS and RHS switched, or with one input unique-ified.
*
* Note: when passing nonzero varRelid, it's normally appropriate to set
* jointype == JOIN_INNER, sjinfo == NULL, even if the clause is really a
* join clause; because we aren't treating it as a join clause.
*/ */
Selectivity Selectivity
clause_selectivity(PlannerInfo *root, clause_selectivity(PlannerInfo *root,
Node *clause, Node *clause,
int varRelid, int varRelid,
JoinType jointype) JoinType jointype,
SpecialJoinInfo *sjinfo)
{ {
Selectivity s1 = 0.5; /* default for any unhandled clause type */ Selectivity s1 = 0.5; /* default for any unhandled clause type */
RestrictInfo *rinfo = NULL; RestrictInfo *rinfo = NULL;
...@@ -457,36 +476,15 @@ clause_selectivity(PlannerInfo *root, ...@@ -457,36 +476,15 @@ clause_selectivity(PlannerInfo *root,
* If possible, cache the result of the selectivity calculation for * If possible, cache the result of the selectivity calculation for
* the clause. We can cache if varRelid is zero or the clause * the clause. We can cache if varRelid is zero or the clause
* contains only vars of that relid --- otherwise varRelid will affect * contains only vars of that relid --- otherwise varRelid will affect
* the result, so mustn't cache. We also have to be careful about the * the result, so mustn't cache.
* jointype. It's OK to cache when jointype is JOIN_INNER or one of
* the outer join types (any given outer-join clause should always be
* examined with the same jointype, so result won't change). It's not
* OK to cache when jointype is one of the special types associated
* with IN processing, because the same clause may be examined with
* different jointypes and the result should vary.
*/ */
if (varRelid == 0 || if (varRelid == 0 ||
bms_is_subset_singleton(rinfo->clause_relids, varRelid)) bms_is_subset_singleton(rinfo->clause_relids, varRelid))
{ {
switch (jointype)
{
case JOIN_INNER:
case JOIN_LEFT:
case JOIN_FULL:
case JOIN_RIGHT:
/* Cacheable --- do we already have the result? */ /* Cacheable --- do we already have the result? */
if (rinfo->this_selec >= 0) if (rinfo->this_selec >= 0)
return rinfo->this_selec; return rinfo->this_selec;
cacheable = true; cacheable = true;
break;
case JOIN_IN:
case JOIN_REVERSE_IN:
case JOIN_UNIQUE_OUTER:
case JOIN_UNIQUE_INNER:
/* unsafe to cache */
break;
}
} }
/* /*
...@@ -568,7 +566,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -568,7 +566,8 @@ clause_selectivity(PlannerInfo *root,
s1 = 1.0 - clause_selectivity(root, s1 = 1.0 - clause_selectivity(root,
(Node *) get_notclausearg((Expr *) clause), (Node *) get_notclausearg((Expr *) clause),
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (and_clause(clause)) else if (and_clause(clause))
{ {
...@@ -576,7 +575,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -576,7 +575,8 @@ clause_selectivity(PlannerInfo *root,
s1 = clauselist_selectivity(root, s1 = clauselist_selectivity(root,
((BoolExpr *) clause)->args, ((BoolExpr *) clause)->args,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (or_clause(clause)) else if (or_clause(clause))
{ {
...@@ -594,7 +594,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -594,7 +594,8 @@ clause_selectivity(PlannerInfo *root,
Selectivity s2 = clause_selectivity(root, Selectivity s2 = clause_selectivity(root,
(Node *) lfirst(arg), (Node *) lfirst(arg),
varRelid, varRelid,
jointype); jointype,
sjinfo);
s1 = s1 + s2 - s1 * s2; s1 = s1 + s2 - s1 * s2;
} }
...@@ -700,7 +701,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -700,7 +701,8 @@ clause_selectivity(PlannerInfo *root,
(ScalarArrayOpExpr *) clause, (ScalarArrayOpExpr *) clause,
is_join_clause, is_join_clause,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (IsA(clause, RowCompareExpr)) else if (IsA(clause, RowCompareExpr))
{ {
...@@ -708,7 +710,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -708,7 +710,8 @@ clause_selectivity(PlannerInfo *root,
s1 = rowcomparesel(root, s1 = rowcomparesel(root,
(RowCompareExpr *) clause, (RowCompareExpr *) clause,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (IsA(clause, NullTest)) else if (IsA(clause, NullTest))
{ {
...@@ -717,7 +720,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -717,7 +720,8 @@ clause_selectivity(PlannerInfo *root,
((NullTest *) clause)->nulltesttype, ((NullTest *) clause)->nulltesttype,
(Node *) ((NullTest *) clause)->arg, (Node *) ((NullTest *) clause)->arg,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (IsA(clause, BooleanTest)) else if (IsA(clause, BooleanTest))
{ {
...@@ -726,7 +730,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -726,7 +730,8 @@ clause_selectivity(PlannerInfo *root,
((BooleanTest *) clause)->booltesttype, ((BooleanTest *) clause)->booltesttype,
(Node *) ((BooleanTest *) clause)->arg, (Node *) ((BooleanTest *) clause)->arg,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (IsA(clause, CurrentOfExpr)) else if (IsA(clause, CurrentOfExpr))
{ {
...@@ -743,7 +748,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -743,7 +748,8 @@ clause_selectivity(PlannerInfo *root,
s1 = clause_selectivity(root, s1 = clause_selectivity(root,
(Node *) ((RelabelType *) clause)->arg, (Node *) ((RelabelType *) clause)->arg,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
else if (IsA(clause, CoerceToDomain)) else if (IsA(clause, CoerceToDomain))
{ {
...@@ -751,7 +757,8 @@ clause_selectivity(PlannerInfo *root, ...@@ -751,7 +757,8 @@ clause_selectivity(PlannerInfo *root,
s1 = clause_selectivity(root, s1 = clause_selectivity(root,
(Node *) ((CoerceToDomain *) clause)->arg, (Node *) ((CoerceToDomain *) clause)->arg,
varRelid, varRelid,
jointype); jointype,
sjinfo);
} }
/* Cache the result if possible */ /* Cache the result if possible */
......
This diff is collapsed.
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/indxpath.c,v 1.231 2008/05/27 00:13:09 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/indxpath.c,v 1.232 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1631,16 +1631,16 @@ best_inner_indexscan(PlannerInfo *root, RelOptInfo *rel, ...@@ -1631,16 +1631,16 @@ best_inner_indexscan(PlannerInfo *root, RelOptInfo *rel,
*cheapest_startup = *cheapest_total = NULL; *cheapest_startup = *cheapest_total = NULL;
/* /*
* Nestloop only supports inner, left, and IN joins. * Nestloop only supports inner, left, semi, and anti joins.
*/ */
switch (jointype) switch (jointype)
{ {
case JOIN_INNER: case JOIN_INNER:
case JOIN_IN:
case JOIN_UNIQUE_OUTER:
isouterjoin = false; isouterjoin = false;
break; break;
case JOIN_LEFT: case JOIN_LEFT:
case JOIN_SEMI:
case JOIN_ANTI:
isouterjoin = true; isouterjoin = true;
break; break;
default: default:
......
This diff is collapsed.
This diff is collapsed.
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/path/orindxpath.c,v 1.84 2008/01/09 20:42:27 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/path/orindxpath.c,v 1.85 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -169,11 +169,11 @@ create_or_index_quals(PlannerInfo *root, RelOptInfo *rel) ...@@ -169,11 +169,11 @@ create_or_index_quals(PlannerInfo *root, RelOptInfo *rel)
* selectivity will stay cached ...) * selectivity will stay cached ...)
*/ */
or_selec = clause_selectivity(root, (Node *) or_rinfo, or_selec = clause_selectivity(root, (Node *) or_rinfo,
0, JOIN_INNER); 0, JOIN_INNER, NULL);
if (or_selec > 0 && or_selec < 1) if (or_selec > 0 && or_selec < 1)
{ {
orig_selec = clause_selectivity(root, (Node *) bestrinfo, orig_selec = clause_selectivity(root, (Node *) bestrinfo,
0, JOIN_INNER); 0, JOIN_INNER, NULL);
bestrinfo->this_selec = orig_selec / or_selec; bestrinfo->this_selec = orig_selec / or_selec;
/* clamp result to sane range */ /* clamp result to sane range */
if (bestrinfo->this_selec > 1) if (bestrinfo->this_selec > 1)
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/plan/createplan.c,v 1.244 2008/08/07 19:35:02 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/plan/createplan.c,v 1.245 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -595,8 +595,8 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path) ...@@ -595,8 +595,8 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path)
{ {
Plan *plan; Plan *plan;
Plan *subplan; Plan *subplan;
List *uniq_exprs;
List *in_operators; List *in_operators;
List *uniq_exprs;
List *newtlist; List *newtlist;
int nextresno; int nextresno;
bool newitems; bool newitems;
...@@ -611,7 +611,7 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path) ...@@ -611,7 +611,7 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path)
if (best_path->umethod == UNIQUE_PATH_NOOP) if (best_path->umethod == UNIQUE_PATH_NOOP)
return subplan; return subplan;
/*---------- /*
* As constructed, the subplan has a "flat" tlist containing just the * As constructed, the subplan has a "flat" tlist containing just the
* Vars needed here and at upper levels. The values we are supposed * Vars needed here and at upper levels. The values we are supposed
* to unique-ify may be expressions in these variables. We have to * to unique-ify may be expressions in these variables. We have to
...@@ -626,29 +626,9 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path) ...@@ -626,29 +626,9 @@ create_unique_plan(PlannerInfo *root, UniquePath *best_path)
* Therefore newtlist starts from build_relation_tlist() not just a * Therefore newtlist starts from build_relation_tlist() not just a
* copy of the subplan's tlist; and we don't install it into the subplan * copy of the subplan's tlist; and we don't install it into the subplan
* unless we are sorting or stuff has to be added. * unless we are sorting or stuff has to be added.
*
* To find the correct list of values to unique-ify, we look in the
* information saved for IN expressions. If this code is ever used in
* other scenarios, some other way of finding what to unique-ify will
* be needed. The IN clause's operators are needed too, since they
* determine what the meaning of "unique" is in this context.
*----------
*/ */
uniq_exprs = NIL; /* just to keep compiler quiet */ in_operators = best_path->in_operators;
in_operators = NIL; uniq_exprs = best_path->uniq_exprs;
foreach(l, root->in_info_list)
{
InClauseInfo *ininfo = (InClauseInfo *) lfirst(l);
if (bms_equal(ininfo->righthand, best_path->path.parent->relids))
{
uniq_exprs = ininfo->sub_targetlist;
in_operators = ininfo->in_operators;
break;
}
}
if (l == NULL) /* fell out of loop? */
elog(ERROR, "could not find UniquePath in in_info_list");
/* initialize modified subplan tlist as just the "required" vars */ /* initialize modified subplan tlist as just the "required" vars */
newtlist = build_relation_tlist(best_path->path.parent); newtlist = build_relation_tlist(best_path->path.parent);
......
This diff is collapsed.
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/plan/planmain.c,v 1.109 2008/08/05 02:43:17 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/plan/planmain.c,v 1.110 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -131,8 +131,8 @@ query_planner(PlannerInfo *root, List *tlist, ...@@ -131,8 +131,8 @@ query_planner(PlannerInfo *root, List *tlist,
* Init planner lists to empty, and set up the array to hold RelOptInfos * Init planner lists to empty, and set up the array to hold RelOptInfos
* for "simple" rels. * for "simple" rels.
* *
* NOTE: in_info_list and append_rel_list were set up by subquery_planner, * NOTE: append_rel_list was set up by subquery_planner, so do not touch
* do not touch here; eq_classes may contain data already, too. * here; eq_classes may contain data already, too.
*/ */
root->simple_rel_array_size = list_length(parse->rtable) + 1; root->simple_rel_array_size = list_length(parse->rtable) + 1;
root->simple_rel_array = (RelOptInfo **) root->simple_rel_array = (RelOptInfo **)
...@@ -143,7 +143,7 @@ query_planner(PlannerInfo *root, List *tlist, ...@@ -143,7 +143,7 @@ query_planner(PlannerInfo *root, List *tlist,
root->left_join_clauses = NIL; root->left_join_clauses = NIL;
root->right_join_clauses = NIL; root->right_join_clauses = NIL;
root->full_join_clauses = NIL; root->full_join_clauses = NIL;
root->oj_info_list = NIL; root->join_info_list = NIL;
root->initial_rels = NIL; root->initial_rels = NIL;
/* /*
...@@ -215,13 +215,6 @@ query_planner(PlannerInfo *root, List *tlist, ...@@ -215,13 +215,6 @@ query_planner(PlannerInfo *root, List *tlist,
joinlist = deconstruct_jointree(root); joinlist = deconstruct_jointree(root);
/*
* Vars mentioned in InClauseInfo items also have to be added to baserel
* targetlists. Nearly always, they'd have got there from the original
* WHERE qual, but in corner cases maybe not.
*/
add_IN_vars_to_tlists(root);
/* /*
* Reconsider any postponed outer-join quals now that we have built up * Reconsider any postponed outer-join quals now that we have built up
* equivalence classes. (This could result in further additions or * equivalence classes. (This could result in further additions or
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/plan/planner.c,v 1.240 2008/08/07 01:11:50 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/plan/planner.c,v 1.241 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -55,8 +55,7 @@ planner_hook_type planner_hook = NULL; ...@@ -55,8 +55,7 @@ planner_hook_type planner_hook = NULL;
#define EXPRKIND_RTFUNC 2 #define EXPRKIND_RTFUNC 2
#define EXPRKIND_VALUES 3 #define EXPRKIND_VALUES 3
#define EXPRKIND_LIMIT 4 #define EXPRKIND_LIMIT 4
#define EXPRKIND_ININFO 5 #define EXPRKIND_APPINFO 5
#define EXPRKIND_APPINFO 6
static Node *preprocess_expression(PlannerInfo *root, Node *expr, int kind); static Node *preprocess_expression(PlannerInfo *root, Node *expr, int kind);
...@@ -255,6 +254,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -255,6 +254,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
PlannerInfo *root; PlannerInfo *root;
Plan *plan; Plan *plan;
List *newHaving; List *newHaving;
bool hasOuterJoins;
ListCell *l; ListCell *l;
/* Create a PlannerInfo data structure for this subquery */ /* Create a PlannerInfo data structure for this subquery */
...@@ -265,23 +265,22 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -265,23 +265,22 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
root->planner_cxt = CurrentMemoryContext; root->planner_cxt = CurrentMemoryContext;
root->init_plans = NIL; root->init_plans = NIL;
root->eq_classes = NIL; root->eq_classes = NIL;
root->in_info_list = NIL;
root->append_rel_list = NIL; root->append_rel_list = NIL;
/* /*
* Look for IN clauses at the top level of WHERE, and transform them into * Look for ANY and EXISTS SubLinks at the top level of WHERE, and try to
* joins. Note that this step only handles IN clauses originally at top * transform them into joins. Note that this step only handles SubLinks
* level of WHERE; if we pull up any subqueries below, their INs are * originally at top level of WHERE; if we pull up any subqueries below,
* processed just before pulling them up. * their SubLinks are processed just before pulling them up.
*/ */
if (parse->hasSubLinks) if (parse->hasSubLinks)
parse->jointree->quals = pull_up_IN_clauses(root, parse->jointree->quals = pull_up_sublinks(root,
parse->jointree->quals); parse->jointree->quals);
/* /*
* Scan the rangetable for set-returning functions, and inline them * Scan the rangetable for set-returning functions, and inline them
* if possible (producing subqueries that might get pulled up next). * if possible (producing subqueries that might get pulled up next).
* Recursion issues here are handled in the same way as for IN clauses. * Recursion issues here are handled in the same way as for SubLinks.
*/ */
inline_set_returning_functions(root); inline_set_returning_functions(root);
...@@ -295,16 +294,11 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -295,16 +294,11 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
/* /*
* Detect whether any rangetable entries are RTE_JOIN kind; if not, we can * Detect whether any rangetable entries are RTE_JOIN kind; if not, we can
* avoid the expense of doing flatten_join_alias_vars(). Also check for * avoid the expense of doing flatten_join_alias_vars(). Also check for
* outer joins --- if none, we can skip reduce_outer_joins() and some * outer joins --- if none, we can skip reduce_outer_joins().
* other processing. This must be done after we have done * This must be done after we have done pull_up_subqueries, of course.
* pull_up_subqueries, of course.
*
* Note: if reduce_outer_joins manages to eliminate all outer joins,
* root->hasOuterJoins is not reset currently. This is OK since its
* purpose is merely to suppress unnecessary processing in simple cases.
*/ */
root->hasJoinRTEs = false; root->hasJoinRTEs = false;
root->hasOuterJoins = false; hasOuterJoins = false;
foreach(l, parse->rtable) foreach(l, parse->rtable)
{ {
RangeTblEntry *rte = (RangeTblEntry *) lfirst(l); RangeTblEntry *rte = (RangeTblEntry *) lfirst(l);
...@@ -314,7 +308,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -314,7 +308,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
root->hasJoinRTEs = true; root->hasJoinRTEs = true;
if (IS_OUTER_JOIN(rte->jointype)) if (IS_OUTER_JOIN(rte->jointype))
{ {
root->hasOuterJoins = true; hasOuterJoins = true;
/* Can quit scanning once we find an outer join */ /* Can quit scanning once we find an outer join */
break; break;
} }
...@@ -362,9 +356,6 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -362,9 +356,6 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
parse->limitCount = preprocess_expression(root, parse->limitCount, parse->limitCount = preprocess_expression(root, parse->limitCount,
EXPRKIND_LIMIT); EXPRKIND_LIMIT);
root->in_info_list = (List *)
preprocess_expression(root, (Node *) root->in_info_list,
EXPRKIND_ININFO);
root->append_rel_list = (List *) root->append_rel_list = (List *)
preprocess_expression(root, (Node *) root->append_rel_list, preprocess_expression(root, (Node *) root->append_rel_list,
EXPRKIND_APPINFO); EXPRKIND_APPINFO);
...@@ -442,7 +433,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse, ...@@ -442,7 +433,7 @@ subquery_planner(PlannerGlobal *glob, Query *parse,
* This step is most easily done after we've done expression * This step is most easily done after we've done expression
* preprocessing. * preprocessing.
*/ */
if (root->hasOuterJoins) if (hasOuterJoins)
reduce_outer_joins(root); reduce_outer_joins(root);
/* /*
...@@ -639,20 +630,15 @@ inheritance_planner(PlannerInfo *root) ...@@ -639,20 +630,15 @@ inheritance_planner(PlannerInfo *root)
continue; continue;
/* /*
* Generate modified query with this rel as target. We have to be * Generate modified query with this rel as target.
* prepared to translate varnos in in_info_list as well as in the
* Query proper.
*/ */
memcpy(&subroot, root, sizeof(PlannerInfo)); memcpy(&subroot, root, sizeof(PlannerInfo));
subroot.parse = (Query *) subroot.parse = (Query *)
adjust_appendrel_attrs((Node *) parse, adjust_appendrel_attrs((Node *) parse,
appinfo); appinfo);
subroot.in_info_list = (List *)
adjust_appendrel_attrs((Node *) root->in_info_list,
appinfo);
subroot.init_plans = NIL; subroot.init_plans = NIL;
/* There shouldn't be any OJ info to translate, as yet */ /* There shouldn't be any OJ info to translate, as yet */
Assert(subroot.oj_info_list == NIL); Assert(subroot.join_info_list == NIL);
/* Generate plan */ /* Generate plan */
subplan = grouping_planner(&subroot, 0.0 /* retrieve all tuples */ ); subplan = grouping_planner(&subroot, 0.0 /* retrieve all tuples */ );
......
This diff is collapsed.
This diff is collapsed.
...@@ -22,7 +22,7 @@ ...@@ -22,7 +22,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/prep/prepunion.c,v 1.152 2008/08/07 19:35:02 tgl Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/prep/prepunion.c,v 1.153 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1465,25 +1465,25 @@ adjust_appendrel_attrs_mutator(Node *node, AppendRelInfo *context) ...@@ -1465,25 +1465,25 @@ adjust_appendrel_attrs_mutator(Node *node, AppendRelInfo *context)
j->rtindex = context->child_relid; j->rtindex = context->child_relid;
return (Node *) j; return (Node *) j;
} }
if (IsA(node, InClauseInfo)) if (IsA(node, FlattenedSubLink))
{ {
/* Copy the InClauseInfo node with correct mutation of subnodes */ /* Copy the FlattenedSubLink node with correct mutation of subnodes */
InClauseInfo *ininfo; FlattenedSubLink *fslink;
ininfo = (InClauseInfo *) expression_tree_mutator(node, fslink = (FlattenedSubLink *) expression_tree_mutator(node,
adjust_appendrel_attrs_mutator, adjust_appendrel_attrs_mutator,
(void *) context); (void *) context);
/* now fix InClauseInfo's relid sets */ /* now fix FlattenedSubLink's relid sets */
ininfo->lefthand = adjust_relid_set(ininfo->lefthand, fslink->lefthand = adjust_relid_set(fslink->lefthand,
context->parent_relid, context->parent_relid,
context->child_relid); context->child_relid);
ininfo->righthand = adjust_relid_set(ininfo->righthand, fslink->righthand = adjust_relid_set(fslink->righthand,
context->parent_relid, context->parent_relid,
context->child_relid); context->child_relid);
return (Node *) ininfo; return (Node *) fslink;
} }
/* Shouldn't need to handle OuterJoinInfo or AppendRelInfo here */ /* Shouldn't need to handle SpecialJoinInfo or AppendRelInfo here */
Assert(!IsA(node, OuterJoinInfo)); Assert(!IsA(node, SpecialJoinInfo));
Assert(!IsA(node, AppendRelInfo)); Assert(!IsA(node, AppendRelInfo));
/* /*
......
This diff is collapsed.
This diff is collapsed.
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/util/relnode.c,v 1.89 2008/01/01 19:45:50 momjian Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/util/relnode.c,v 1.90 2008/08/14 18:47:59 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -273,7 +273,7 @@ find_join_rel(PlannerInfo *root, Relids relids) ...@@ -273,7 +273,7 @@ find_join_rel(PlannerInfo *root, Relids relids)
* 'joinrelids' is the Relids set that uniquely identifies the join * 'joinrelids' is the Relids set that uniquely identifies the join
* 'outer_rel' and 'inner_rel' are relation nodes for the relations to be * 'outer_rel' and 'inner_rel' are relation nodes for the relations to be
* joined * joined
* 'jointype': type of join (inner/outer) * 'sjinfo': join context info
* 'restrictlist_ptr': result variable. If not NULL, *restrictlist_ptr * 'restrictlist_ptr': result variable. If not NULL, *restrictlist_ptr
* receives the list of RestrictInfo nodes that apply to this * receives the list of RestrictInfo nodes that apply to this
* particular pair of joinable relations. * particular pair of joinable relations.
...@@ -286,7 +286,7 @@ build_join_rel(PlannerInfo *root, ...@@ -286,7 +286,7 @@ build_join_rel(PlannerInfo *root,
Relids joinrelids, Relids joinrelids,
RelOptInfo *outer_rel, RelOptInfo *outer_rel,
RelOptInfo *inner_rel, RelOptInfo *inner_rel,
JoinType jointype, SpecialJoinInfo *sjinfo,
List **restrictlist_ptr) List **restrictlist_ptr)
{ {
RelOptInfo *joinrel; RelOptInfo *joinrel;
...@@ -375,7 +375,7 @@ build_join_rel(PlannerInfo *root, ...@@ -375,7 +375,7 @@ build_join_rel(PlannerInfo *root,
* Set estimates of the joinrel's size. * Set estimates of the joinrel's size.
*/ */
set_joinrel_size_estimates(root, joinrel, outer_rel, inner_rel, set_joinrel_size_estimates(root, joinrel, outer_rel, inner_rel,
jointype, restrictlist); sjinfo, restrictlist);
/* /*
* Add the joinrel to the query's joinrel list, and store it into the * Add the joinrel to the query's joinrel list, and store it into the
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
* Portions Copyright (c) 1996-2008, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2008, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/pg_list.h,v 1.58 2008/03/17 02:18:55 tgl Exp $ * $PostgreSQL: pgsql/src/include/nodes/pg_list.h,v 1.59 2008/08/14 18:48:00 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -218,6 +218,9 @@ extern List *list_union_ptr(List *list1, List *list2); ...@@ -218,6 +218,9 @@ extern List *list_union_ptr(List *list1, List *list2);
extern List *list_union_int(List *list1, List *list2); extern List *list_union_int(List *list1, List *list2);
extern List *list_union_oid(List *list1, List *list2); extern List *list_union_oid(List *list1, List *list2);
extern List *list_intersection(List *list1, List *list2);
/* currently, there's no need for list_intersection_int etc */
extern List *list_difference(List *list1, List *list2); extern List *list_difference(List *list1, List *list2);
extern List *list_difference_ptr(List *list1, List *list2); extern List *list_difference_ptr(List *list1, List *list2);
extern List *list_difference_int(List *list1, List *list2); extern List *list_difference_int(List *list1, List *list2);
......
This diff is collapsed.
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2008, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2008, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/optimizer/clauses.h,v 1.91 2008/08/02 21:32:01 tgl Exp $ * $PostgreSQL: pgsql/src/include/optimizer/clauses.h,v 1.92 2008/08/14 18:48:00 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -59,6 +59,9 @@ extern bool contain_mutable_functions(Node *clause); ...@@ -59,6 +59,9 @@ extern bool contain_mutable_functions(Node *clause);
extern bool contain_volatile_functions(Node *clause); extern bool contain_volatile_functions(Node *clause);
extern bool contain_nonstrict_functions(Node *clause); extern bool contain_nonstrict_functions(Node *clause);
extern Relids find_nonnullable_rels(Node *clause); extern Relids find_nonnullable_rels(Node *clause);
extern List *find_nonnullable_vars(Node *clause);
extern List *find_forced_null_vars(Node *clause);
extern Var *find_forced_null_var(Node *clause);
extern bool is_pseudo_constant_clause(Node *clause); extern bool is_pseudo_constant_clause(Node *clause);
extern bool is_pseudo_constant_clause_relids(Node *clause, Relids relids); extern bool is_pseudo_constant_clause_relids(Node *clause, Relids relids);
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment