1996-07-09 06:22:35 +00:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
1999-02-13 23:22:53 +00:00
|
|
|
* clauses.c
|
1997-09-07 05:04:48 +00:00
|
|
|
* routines to manipulate qualification clauses
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
2000-01-26 05:58:53 +00:00
|
|
|
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
|
|
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* IDENTIFICATION
|
2000-05-30 00:49:57 +00:00
|
|
|
* $Header: /cvsroot/pgsql/src/backend/optimizer/util/clauses.c,v 1.68 2000/05/30 00:49:49 momjian Exp $
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
* HISTORY
|
1997-09-07 05:04:48 +00:00
|
|
|
* AUTHOR DATE MAJOR EVENT
|
|
|
|
* Andrew Yu Nov 3, 1994 clause.c and clauses.c combined
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "postgres.h"
|
|
|
|
|
1999-07-15 23:04:24 +00:00
|
|
|
#include "catalog/pg_operator.h"
|
1999-09-26 02:28:44 +00:00
|
|
|
#include "catalog/pg_proc.h"
|
|
|
|
#include "catalog/pg_type.h"
|
|
|
|
#include "executor/executor.h"
|
1996-07-09 06:22:35 +00:00
|
|
|
#include "nodes/makefuncs.h"
|
|
|
|
#include "nodes/nodeFuncs.h"
|
|
|
|
#include "optimizer/clauses.h"
|
1999-08-22 20:15:04 +00:00
|
|
|
#include "optimizer/tlist.h"
|
1996-07-09 06:22:35 +00:00
|
|
|
#include "optimizer/var.h"
|
1999-09-26 02:28:44 +00:00
|
|
|
#include "parser/parse_type.h"
|
1999-12-09 05:58:56 +00:00
|
|
|
#include "parser/parsetree.h"
|
1999-07-16 05:00:38 +00:00
|
|
|
#include "utils/lsyscache.h"
|
1999-09-26 02:28:44 +00:00
|
|
|
#include "utils/syscache.h"
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
|
1999-10-07 04:23:24 +00:00
|
|
|
/* note that pg_type.h hardwires size of bool as 1 ... duplicate it */
|
|
|
|
#define MAKEBOOLCONST(val,isnull) \
|
|
|
|
((Node *) makeConst(BOOLOID, 1, (Datum) (val), \
|
|
|
|
(isnull), true, false, false))
|
|
|
|
|
1999-12-13 01:27:21 +00:00
|
|
|
static bool contain_agg_clause_walker(Node *node, void *context);
|
1999-08-22 20:15:04 +00:00
|
|
|
static bool pull_agg_clause_walker(Node *node, List **listptr);
|
2000-04-04 01:21:48 +00:00
|
|
|
static bool contain_subplans_walker(Node *node, void *context);
|
|
|
|
static bool pull_subplans_walker(Node *node, List **listptr);
|
1999-08-22 20:15:04 +00:00
|
|
|
static bool check_subplans_for_ungrouped_vars_walker(Node *node,
|
2000-04-12 17:17:23 +00:00
|
|
|
Query *context);
|
|
|
|
static int is_single_func(Node *node);
|
|
|
|
static Node *eval_const_expressions_mutator(Node *node, void *context);
|
2000-03-19 18:20:38 +00:00
|
|
|
static Expr *simplify_op_or_func(Expr *expr, List *args);
|
1997-08-19 21:40:56 +00:00
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_clause(int type, Node *oper, List *args)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-10-02 04:37:52 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
|
|
|
|
|
|
|
switch (type)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-10-02 04:37:52 +00:00
|
|
|
case AND_EXPR:
|
|
|
|
case OR_EXPR:
|
|
|
|
case NOT_EXPR:
|
|
|
|
expr->typeOid = BOOLOID;
|
|
|
|
break;
|
|
|
|
case OP_EXPR:
|
|
|
|
expr->typeOid = ((Oper *) oper)->opresulttype;
|
|
|
|
break;
|
|
|
|
case FUNC_EXPR:
|
|
|
|
expr->typeOid = ((Func *) oper)->functype;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
elog(ERROR, "make_clause: unsupported type %d", type);
|
|
|
|
break;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-10-02 04:37:52 +00:00
|
|
|
expr->opType = type;
|
|
|
|
expr->oper = oper; /* ignored for AND, OR, NOT */
|
|
|
|
expr->args = args;
|
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
1997-09-07 05:04:48 +00:00
|
|
|
* OPERATOR clause functions
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* is_opclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns t iff the clause is an operator clause:
|
1997-09-07 05:04:48 +00:00
|
|
|
* (op expr expr) or (op expr).
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
* [historical note: is_clause has the exact functionality and is used
|
1997-09-07 05:04:48 +00:00
|
|
|
* throughout the code. They're renamed to is_opclause for clarity.
|
|
|
|
* - ay 10/94.]
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
bool
|
1997-09-08 21:56:23 +00:00
|
|
|
is_opclause(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-02-03 21:18:02 +00:00
|
|
|
return (clause != NULL &&
|
1999-08-10 03:00:15 +00:00
|
|
|
IsA(clause, Expr) &&
|
|
|
|
((Expr *) clause)->opType == OP_EXPR);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* make_opclause
|
1997-09-07 05:04:48 +00:00
|
|
|
* Creates a clause given its operator left operand and right
|
|
|
|
* operand (if it is non-null).
|
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_opclause(Oper *op, Var *leftop, Var *rightop)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-10-02 04:37:52 +00:00
|
|
|
expr->typeOid = op->opresulttype;
|
1997-09-07 05:04:48 +00:00
|
|
|
expr->opType = OP_EXPR;
|
|
|
|
expr->oper = (Node *) op;
|
1999-02-14 22:24:25 +00:00
|
|
|
if (rightop)
|
|
|
|
expr->args = lcons(leftop, lcons(rightop, NIL));
|
|
|
|
else
|
|
|
|
expr->args = lcons(leftop, NIL);
|
1997-09-07 05:04:48 +00:00
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* get_leftop
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns the left operand of a clause of the form (op expr expr)
|
1997-09-07 05:04:48 +00:00
|
|
|
* or (op expr)
|
1999-09-09 02:36:04 +00:00
|
|
|
*
|
|
|
|
* NB: for historical reasons, the result is declared Var *, even
|
|
|
|
* though many callers can cope with results that are not Vars.
|
|
|
|
* The result really ought to be declared Expr * or Node *.
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Var *
|
1997-09-08 21:56:23 +00:00
|
|
|
get_leftop(Expr *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-07 05:04:48 +00:00
|
|
|
if (clause->args != NULL)
|
1998-09-01 03:29:17 +00:00
|
|
|
return lfirst(clause->args);
|
1997-09-07 05:04:48 +00:00
|
|
|
else
|
|
|
|
return NULL;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1996-07-09 06:22:35 +00:00
|
|
|
* get_rightop
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns the right operand in a clause of the form (op expr expr).
|
1999-02-14 22:24:25 +00:00
|
|
|
* NB: result will be NULL if applied to a unary op clause.
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Var *
|
1997-09-08 21:56:23 +00:00
|
|
|
get_rightop(Expr *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-07 05:04:48 +00:00
|
|
|
if (clause->args != NULL && lnext(clause->args) != NULL)
|
1998-09-01 03:29:17 +00:00
|
|
|
return lfirst(lnext(clause->args));
|
1997-09-07 05:04:48 +00:00
|
|
|
else
|
|
|
|
return NULL;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
1997-09-07 05:04:48 +00:00
|
|
|
* FUNC clause functions
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* is_funcclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns t iff the clause is a function clause: (func { expr }).
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
bool
|
1997-09-08 21:56:23 +00:00
|
|
|
is_funcclause(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-02-03 21:18:02 +00:00
|
|
|
return (clause != NULL &&
|
1999-08-10 03:00:15 +00:00
|
|
|
IsA(clause, Expr) &&
|
1999-06-19 03:41:45 +00:00
|
|
|
((Expr *) clause)->opType == FUNC_EXPR);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* make_funcclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Creates a function clause given the FUNC node and the functional
|
|
|
|
* arguments.
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_funcclause(Func *func, List *funcargs)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-08-16 02:17:58 +00:00
|
|
|
expr->typeOid = func->functype;
|
1997-09-07 05:04:48 +00:00
|
|
|
expr->opType = FUNC_EXPR;
|
|
|
|
expr->oper = (Node *) func;
|
|
|
|
expr->args = funcargs;
|
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
1997-09-07 05:04:48 +00:00
|
|
|
* OR clause functions
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* or_clause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns t iff the clause is an 'or' clause: (OR { expr }).
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
bool
|
1997-09-08 21:56:23 +00:00
|
|
|
or_clause(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-08-10 03:00:15 +00:00
|
|
|
return (clause != NULL &&
|
|
|
|
IsA(clause, Expr) &&
|
|
|
|
((Expr *) clause)->opType == OR_EXPR);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* make_orclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Creates an 'or' clause given a list of its subclauses.
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_orclause(List *orclauses)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
expr->typeOid = BOOLOID;
|
1997-09-07 05:04:48 +00:00
|
|
|
expr->opType = OR_EXPR;
|
|
|
|
expr->oper = NULL;
|
|
|
|
expr->args = orclauses;
|
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
1997-09-07 05:04:48 +00:00
|
|
|
* NOT clause functions
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* not_clause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns t iff this is a 'not' clause: (NOT expr).
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
bool
|
1997-09-08 21:56:23 +00:00
|
|
|
not_clause(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-02-03 21:18:02 +00:00
|
|
|
return (clause != NULL &&
|
1999-08-10 03:00:15 +00:00
|
|
|
IsA(clause, Expr) &&
|
1999-06-19 03:41:45 +00:00
|
|
|
((Expr *) clause)->opType == NOT_EXPR);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* make_notclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Create a 'not' clause given the expression to be negated.
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_notclause(Expr *notclause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
expr->typeOid = BOOLOID;
|
1997-09-07 05:04:48 +00:00
|
|
|
expr->opType = NOT_EXPR;
|
|
|
|
expr->oper = NULL;
|
|
|
|
expr->args = lcons(notclause, NIL);
|
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* get_notclausearg
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Retrieve the clause within a 'not' clause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
get_notclausearg(Expr *notclause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1998-09-01 03:29:17 +00:00
|
|
|
return lfirst(notclause->args);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
1997-09-07 05:04:48 +00:00
|
|
|
* AND clause functions
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* and_clause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns t iff its argument is an 'and' clause: (AND { expr }).
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
bool
|
1997-09-08 21:56:23 +00:00
|
|
|
and_clause(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-02-03 21:18:02 +00:00
|
|
|
return (clause != NULL &&
|
1999-08-10 03:00:15 +00:00
|
|
|
IsA(clause, Expr) &&
|
1999-06-19 03:41:45 +00:00
|
|
|
((Expr *) clause)->opType == AND_EXPR);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* make_andclause
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Create an 'and' clause given its arguments in a list.
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
1998-02-26 04:46:47 +00:00
|
|
|
Expr *
|
1997-09-08 21:56:23 +00:00
|
|
|
make_andclause(List *andclauses)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Expr *expr = makeNode(Expr);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
expr->typeOid = BOOLOID;
|
1997-09-07 05:04:48 +00:00
|
|
|
expr->opType = AND_EXPR;
|
|
|
|
expr->oper = NULL;
|
|
|
|
expr->args = andclauses;
|
|
|
|
return expr;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1999-07-24 23:21:14 +00:00
|
|
|
/*
|
1999-10-07 04:23:24 +00:00
|
|
|
* Sometimes (such as in the result of canonicalize_qual or the input of
|
|
|
|
* ExecQual), we use lists of expression nodes with implicit AND semantics.
|
|
|
|
*
|
|
|
|
* These functions convert between an AND-semantics expression list and the
|
|
|
|
* ordinary representation of a boolean expression.
|
|
|
|
*
|
|
|
|
* Note that an empty list is considered equivalent to TRUE.
|
1999-07-24 23:21:14 +00:00
|
|
|
*/
|
|
|
|
Expr *
|
|
|
|
make_ands_explicit(List *andclauses)
|
|
|
|
{
|
|
|
|
if (andclauses == NIL)
|
1999-10-07 04:23:24 +00:00
|
|
|
return (Expr *) MAKEBOOLCONST(true, false);
|
|
|
|
else if (lnext(andclauses) == NIL)
|
1999-07-24 23:21:14 +00:00
|
|
|
return (Expr *) lfirst(andclauses);
|
|
|
|
else
|
|
|
|
return make_andclause(andclauses);
|
|
|
|
}
|
1998-12-04 15:34:49 +00:00
|
|
|
|
1999-07-27 03:51:11 +00:00
|
|
|
List *
|
|
|
|
make_ands_implicit(Expr *clause)
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-10-07 04:23:24 +00:00
|
|
|
/*
|
|
|
|
* NB: because the parser sets the qual field to NULL in a query that
|
|
|
|
* has no WHERE clause, we must consider a NULL input clause as TRUE,
|
2000-04-12 17:17:23 +00:00
|
|
|
* even though one might more reasonably think it FALSE. Grumble. If
|
|
|
|
* this causes trouble, consider changing the parser's behavior.
|
1999-10-07 04:23:24 +00:00
|
|
|
*/
|
1999-07-27 03:51:11 +00:00
|
|
|
if (clause == NULL)
|
1999-10-07 04:23:24 +00:00
|
|
|
return NIL; /* NULL -> NIL list == TRUE */
|
1999-07-27 03:51:11 +00:00
|
|
|
else if (and_clause((Node *) clause))
|
|
|
|
return clause->args;
|
1999-10-07 04:23:24 +00:00
|
|
|
else if (IsA(clause, Const) &&
|
2000-04-12 17:17:23 +00:00
|
|
|
!((Const *) clause)->constisnull &&
|
1999-10-07 04:23:24 +00:00
|
|
|
DatumGetInt32(((Const *) clause)->constvalue))
|
|
|
|
return NIL; /* constant TRUE input -> NIL list */
|
1999-07-27 03:51:11 +00:00
|
|
|
else
|
|
|
|
return lcons(clause, NIL);
|
|
|
|
}
|
|
|
|
|
1998-12-04 15:34:49 +00:00
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
/*****************************************************************************
|
2000-04-04 01:21:48 +00:00
|
|
|
* Aggregate-function clause manipulation
|
1996-07-09 06:22:35 +00:00
|
|
|
*****************************************************************************/
|
|
|
|
|
1999-12-13 01:27:21 +00:00
|
|
|
/*
|
|
|
|
* contain_agg_clause
|
|
|
|
* Recursively search for Aggref nodes within a clause.
|
|
|
|
*
|
|
|
|
* Returns true if any aggregate found.
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
contain_agg_clause(Node *clause)
|
|
|
|
{
|
|
|
|
return contain_agg_clause_walker(clause, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
contain_agg_clause_walker(Node *node, void *context)
|
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
|
|
|
if (IsA(node, Aggref))
|
2000-04-12 17:17:23 +00:00
|
|
|
return true; /* abort the tree traversal and return
|
|
|
|
* true */
|
1999-12-13 01:27:21 +00:00
|
|
|
return expression_tree_walker(node, contain_agg_clause_walker, context);
|
|
|
|
}
|
|
|
|
|
1999-08-22 20:15:04 +00:00
|
|
|
/*
|
|
|
|
* pull_agg_clause
|
|
|
|
* Recursively pulls all Aggref nodes from an expression tree.
|
|
|
|
*
|
|
|
|
* Returns list of Aggref nodes found. Note the nodes themselves are not
|
|
|
|
* copied, only referenced.
|
1999-12-13 01:27:21 +00:00
|
|
|
*
|
|
|
|
* Note: this also checks for nested aggregates, which are an error.
|
1999-08-22 20:15:04 +00:00
|
|
|
*/
|
|
|
|
List *
|
|
|
|
pull_agg_clause(Node *clause)
|
|
|
|
{
|
|
|
|
List *result = NIL;
|
|
|
|
|
|
|
|
pull_agg_clause_walker(clause, &result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
pull_agg_clause_walker(Node *node, List **listptr)
|
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
|
|
|
if (IsA(node, Aggref))
|
|
|
|
{
|
|
|
|
*listptr = lappend(*listptr, node);
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-12-13 01:27:21 +00:00
|
|
|
/*
|
|
|
|
* Complain if the aggregate's argument contains any aggregates;
|
|
|
|
* nested agg functions are semantically nonsensical.
|
1999-08-22 20:15:04 +00:00
|
|
|
*/
|
1999-12-13 01:27:21 +00:00
|
|
|
if (contain_agg_clause(((Aggref *) node)->target))
|
|
|
|
elog(ERROR, "Aggregate function calls may not be nested");
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-12-13 01:27:21 +00:00
|
|
|
/*
|
|
|
|
* Having checked that, we need not recurse into the argument.
|
|
|
|
*/
|
|
|
|
return false;
|
1999-08-22 20:15:04 +00:00
|
|
|
}
|
|
|
|
return expression_tree_walker(node, pull_agg_clause_walker,
|
|
|
|
(void *) listptr);
|
|
|
|
}
|
|
|
|
|
2000-04-04 01:21:48 +00:00
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
* Subplan clause manipulation
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* contain_subplans
|
|
|
|
* Recursively search for subplan nodes within a clause.
|
|
|
|
*
|
|
|
|
* If we see a SubLink node, we will return TRUE. This is only possible if
|
|
|
|
* the expression tree hasn't yet been transformed by subselect.c. We do not
|
|
|
|
* know whether the node will produce a true subplan or just an initplan,
|
|
|
|
* but we make the conservative assumption that it will be a subplan.
|
|
|
|
*
|
|
|
|
* Returns true if any subplan found.
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
contain_subplans(Node *clause)
|
|
|
|
{
|
|
|
|
return contain_subplans_walker(clause, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
contain_subplans_walker(Node *node, void *context)
|
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
|
|
|
if (is_subplan(node) || IsA(node, SubLink))
|
2000-04-12 17:17:23 +00:00
|
|
|
return true; /* abort the tree traversal and return
|
|
|
|
* true */
|
2000-04-04 01:21:48 +00:00
|
|
|
return expression_tree_walker(node, contain_subplans_walker, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* pull_subplans
|
|
|
|
* Recursively pulls all subplans from an expression tree.
|
|
|
|
*
|
2000-04-12 17:17:23 +00:00
|
|
|
* Returns list of subplan nodes found. Note the nodes themselves are not
|
2000-04-04 01:21:48 +00:00
|
|
|
* copied, only referenced.
|
|
|
|
*/
|
|
|
|
List *
|
|
|
|
pull_subplans(Node *clause)
|
|
|
|
{
|
|
|
|
List *result = NIL;
|
|
|
|
|
|
|
|
pull_subplans_walker(clause, &result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
pull_subplans_walker(Node *node, List **listptr)
|
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
|
|
|
if (is_subplan(node))
|
|
|
|
{
|
|
|
|
*listptr = lappend(*listptr, ((Expr *) node)->oper);
|
|
|
|
/* fall through to check args to subplan */
|
|
|
|
}
|
|
|
|
return expression_tree_walker(node, pull_subplans_walker,
|
|
|
|
(void *) listptr);
|
|
|
|
}
|
|
|
|
|
1999-08-22 20:15:04 +00:00
|
|
|
/*
|
|
|
|
* check_subplans_for_ungrouped_vars
|
|
|
|
* Check for subplans that are being passed ungrouped variables as
|
1999-12-09 05:58:56 +00:00
|
|
|
* parameters; generate an error message if any are found.
|
1999-08-22 20:15:04 +00:00
|
|
|
*
|
|
|
|
* In most contexts, ungrouped variables will be detected by the parser (see
|
1999-12-09 05:58:56 +00:00
|
|
|
* parse_agg.c, check_ungrouped_columns()). But that routine currently does
|
|
|
|
* not check subplans, because the necessary info is not computed until the
|
2000-03-21 05:12:12 +00:00
|
|
|
* planner runs. So we do it here, after we have processed sublinks into
|
|
|
|
* subplans. This ought to be cleaned up someday.
|
1999-08-22 20:15:04 +00:00
|
|
|
*
|
|
|
|
* 'clause' is the expression tree to be searched for subplans.
|
2000-03-21 05:12:12 +00:00
|
|
|
* 'query' provides the GROUP BY list, the target list that the group clauses
|
|
|
|
* refer to, and the range table.
|
1999-08-22 20:15:04 +00:00
|
|
|
*/
|
1999-12-09 05:58:56 +00:00
|
|
|
void
|
1999-08-22 20:15:04 +00:00
|
|
|
check_subplans_for_ungrouped_vars(Node *clause,
|
2000-03-21 05:12:12 +00:00
|
|
|
Query *query)
|
1999-08-22 20:15:04 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* No special setup needed; context for walker is just the Query
|
|
|
|
* pointer
|
|
|
|
*/
|
2000-03-21 05:12:12 +00:00
|
|
|
check_subplans_for_ungrouped_vars_walker(clause, query);
|
1999-08-22 20:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
check_subplans_for_ungrouped_vars_walker(Node *node,
|
2000-03-21 05:12:12 +00:00
|
|
|
Query *context)
|
1999-08-22 20:15:04 +00:00
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-08-22 20:15:04 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* We can ignore Vars other than in subplan args lists, since the
|
|
|
|
* parser already checked 'em.
|
1999-08-22 20:15:04 +00:00
|
|
|
*/
|
|
|
|
if (is_subplan(node))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-08-22 20:15:04 +00:00
|
|
|
/*
|
|
|
|
* The args list of the subplan node represents attributes from
|
|
|
|
* outside passed into the sublink.
|
|
|
|
*/
|
2000-04-12 17:17:23 +00:00
|
|
|
List *t;
|
1999-08-22 20:15:04 +00:00
|
|
|
|
|
|
|
foreach(t, ((Expr *) node)->args)
|
|
|
|
{
|
|
|
|
Node *thisarg = lfirst(t);
|
1999-12-09 05:58:56 +00:00
|
|
|
Var *var;
|
|
|
|
bool contained_in_group_clause;
|
1999-08-22 20:15:04 +00:00
|
|
|
List *gl;
|
|
|
|
|
1999-12-09 05:58:56 +00:00
|
|
|
/*
|
|
|
|
* We do not care about args that are not local variables;
|
|
|
|
* params or outer-level vars are not our responsibility to
|
2000-04-12 17:17:23 +00:00
|
|
|
* check. (The outer-level query passing them to us needs to
|
|
|
|
* worry, instead.)
|
1999-12-09 05:58:56 +00:00
|
|
|
*/
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!IsA(thisarg, Var))
|
1999-12-09 05:58:56 +00:00
|
|
|
continue;
|
|
|
|
var = (Var *) thisarg;
|
|
|
|
if (var->varlevelsup > 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Else, see if it is a grouping column.
|
|
|
|
*/
|
|
|
|
contained_in_group_clause = false;
|
2000-03-21 05:12:12 +00:00
|
|
|
foreach(gl, context->groupClause)
|
1999-08-22 20:15:04 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
GroupClause *gcl = lfirst(gl);
|
|
|
|
Node *groupexpr;
|
1999-08-22 20:15:04 +00:00
|
|
|
|
|
|
|
groupexpr = get_sortgroupclause_expr(gcl,
|
|
|
|
context->targetList);
|
|
|
|
if (equal(thisarg, groupexpr))
|
|
|
|
{
|
|
|
|
contained_in_group_clause = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!contained_in_group_clause)
|
1999-12-09 05:58:56 +00:00
|
|
|
{
|
|
|
|
/* Found an ungrouped argument. Complain. */
|
2000-04-12 17:17:23 +00:00
|
|
|
RangeTblEntry *rte;
|
|
|
|
char *attname;
|
1999-12-09 05:58:56 +00:00
|
|
|
|
|
|
|
Assert(var->varno > 0 &&
|
2000-03-21 05:12:12 +00:00
|
|
|
var->varno <= length(context->rtable));
|
|
|
|
rte = rt_fetch(var->varno, context->rtable);
|
1999-12-09 05:58:56 +00:00
|
|
|
attname = get_attname(rte->relid, var->varattno);
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!attname)
|
1999-12-09 05:58:56 +00:00
|
|
|
elog(ERROR, "cache lookup of attribute %d in relation %u failed",
|
|
|
|
var->varattno, rte->relid);
|
|
|
|
elog(ERROR, "Sub-SELECT uses un-GROUPed attribute %s.%s from outer query",
|
2000-02-15 03:38:29 +00:00
|
|
|
rte->ref->relname, attname);
|
1999-12-09 05:58:56 +00:00
|
|
|
}
|
1999-08-22 20:15:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return expression_tree_walker(node,
|
2000-04-12 17:17:23 +00:00
|
|
|
check_subplans_for_ungrouped_vars_walker,
|
1999-08-22 20:15:04 +00:00
|
|
|
(void *) context);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-04-04 01:21:48 +00:00
|
|
|
/*****************************************************************************
|
|
|
|
* *
|
|
|
|
* General clause-manipulating routines *
|
|
|
|
* *
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* pull_constant_clauses
|
|
|
|
* Scans through a list of qualifications and find those that
|
|
|
|
* contain no variables (of the current query level).
|
|
|
|
*
|
|
|
|
* Returns a list of the constant clauses in constantQual and the remaining
|
|
|
|
* quals as the return value.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
List *
|
|
|
|
pull_constant_clauses(List *quals, List **constantQual)
|
|
|
|
{
|
|
|
|
List *q;
|
|
|
|
List *constqual = NIL;
|
|
|
|
List *restqual = NIL;
|
|
|
|
|
|
|
|
foreach(q, quals)
|
|
|
|
{
|
|
|
|
if (!contain_var_clause(lfirst(q)))
|
|
|
|
constqual = lcons(lfirst(q), constqual);
|
|
|
|
else
|
|
|
|
restqual = lcons(lfirst(q), restqual);
|
|
|
|
}
|
|
|
|
*constantQual = constqual;
|
|
|
|
return restqual;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* clause_relids_vars
|
1999-08-10 03:00:15 +00:00
|
|
|
* Retrieves distinct relids and vars appearing within a clause.
|
|
|
|
*
|
|
|
|
* '*relids' is set to an integer list of all distinct "varno"s appearing
|
|
|
|
* in Vars within the clause.
|
|
|
|
* '*vars' is set to a list of all distinct Vars appearing within the clause.
|
|
|
|
* Var nodes are considered distinct if they have different varno
|
|
|
|
* or varattno values. If there are several occurrences of the same
|
|
|
|
* varno/varattno, you get a randomly chosen one...
|
1999-08-26 05:09:06 +00:00
|
|
|
*
|
|
|
|
* Note that upper-level vars are ignored, since they normally will
|
|
|
|
* become Params with respect to this query level.
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
void
|
1999-05-25 22:43:53 +00:00
|
|
|
clause_get_relids_vars(Node *clause, Relids *relids, List **vars)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-08-26 05:09:06 +00:00
|
|
|
List *clvars = pull_var_clause(clause, false);
|
1997-09-08 02:41:22 +00:00
|
|
|
List *varno_list = NIL;
|
1999-08-10 03:00:15 +00:00
|
|
|
List *var_list = NIL;
|
1999-06-19 03:41:45 +00:00
|
|
|
List *i;
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
foreach(i, clvars)
|
1996-07-25 20:36:46 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Var *var = (Var *) lfirst(i);
|
|
|
|
List *vi;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
if (!intMember(var->varno, varno_list))
|
1999-08-10 03:00:15 +00:00
|
|
|
varno_list = lconsi(var->varno, varno_list);
|
1997-09-07 05:04:48 +00:00
|
|
|
foreach(vi, var_list)
|
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
Var *in_list = (Var *) lfirst(vi);
|
1998-02-26 04:46:47 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
if (in_list->varno == var->varno &&
|
1998-02-13 03:40:23 +00:00
|
|
|
in_list->varattno == var->varattno)
|
1997-09-07 05:04:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (vi == NIL)
|
1999-08-10 03:00:15 +00:00
|
|
|
var_list = lcons(var, var_list);
|
1996-07-25 20:36:46 +00:00
|
|
|
}
|
1999-08-09 00:51:26 +00:00
|
|
|
freeList(clvars);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
*relids = varno_list;
|
|
|
|
*vars = var_list;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* NumRelids
|
|
|
|
* (formerly clause_relids)
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
* Returns the number of different relations referenced in 'clause'.
|
|
|
|
*/
|
|
|
|
int
|
1997-09-08 21:56:23 +00:00
|
|
|
NumRelids(Node *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-08-10 03:00:15 +00:00
|
|
|
List *varno_list = pull_varnos(clause);
|
|
|
|
int result = length(varno_list);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-08-10 03:00:15 +00:00
|
|
|
freeList(varno_list);
|
1999-08-09 00:51:26 +00:00
|
|
|
return result;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-02-13 23:22:53 +00:00
|
|
|
* get_relattval
|
1999-07-25 23:07:26 +00:00
|
|
|
* Extract information from a restriction or join clause for
|
|
|
|
* selectivity estimation. The inputs are an expression
|
|
|
|
* and a relation number (which can be 0 if we don't care which
|
|
|
|
* relation is used; that'd normally be the case for restriction
|
|
|
|
* clauses, where the caller already knows that only one relation
|
|
|
|
* is referenced in the clause). The routine checks that the
|
|
|
|
* expression is of the form (var op something) or (something op var)
|
|
|
|
* where the var is an attribute of the specified relation, or
|
2000-04-12 17:17:23 +00:00
|
|
|
* a function of a var of the specified relation. If so, it
|
1999-07-25 23:07:26 +00:00
|
|
|
* returns the following info:
|
|
|
|
* the found relation number (same as targetrelid unless that is 0)
|
|
|
|
* the found var number (or InvalidAttrNumber if a function)
|
|
|
|
* if the "something" is a constant, the value of the constant
|
|
|
|
* flags indicating whether a constant was found, and on which side.
|
|
|
|
* Default values are returned if the expression is too complicated,
|
1999-09-09 02:36:04 +00:00
|
|
|
* specifically 0 for the relid and attno, 0 for the constant value.
|
|
|
|
*
|
|
|
|
* Note that negative attno values are *not* invalid, but represent
|
2000-04-12 17:17:23 +00:00
|
|
|
* system attributes such as OID. It's sufficient to check for relid=0
|
1999-09-09 02:36:04 +00:00
|
|
|
* to determine whether the routine succeeded.
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
void
|
1997-09-08 21:56:23 +00:00
|
|
|
get_relattval(Node *clause,
|
1999-07-25 23:07:26 +00:00
|
|
|
int targetrelid,
|
1997-09-07 05:04:48 +00:00
|
|
|
int *relid,
|
1997-09-08 20:59:27 +00:00
|
|
|
AttrNumber *attno,
|
1997-09-08 21:56:23 +00:00
|
|
|
Datum *constval,
|
1997-09-07 05:04:48 +00:00
|
|
|
int *flag)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-02-14 22:24:25 +00:00
|
|
|
Var *left,
|
1999-07-25 23:07:26 +00:00
|
|
|
*right,
|
|
|
|
*other;
|
|
|
|
int funcvarno;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-02-14 22:24:25 +00:00
|
|
|
/* Careful; the passed clause might not be a binary operator at all */
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-02-14 22:24:25 +00:00
|
|
|
if (!is_opclause(clause))
|
|
|
|
goto default_results;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-02-14 22:24:25 +00:00
|
|
|
left = get_leftop((Expr *) clause);
|
|
|
|
right = get_rightop((Expr *) clause);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-02-14 22:24:25 +00:00
|
|
|
if (!right)
|
|
|
|
goto default_results;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-07-25 23:07:26 +00:00
|
|
|
/* First look for the var or func */
|
|
|
|
|
|
|
|
if (IsA(left, Var) &&
|
|
|
|
(targetrelid == 0 || targetrelid == left->varno))
|
1999-02-14 22:24:25 +00:00
|
|
|
{
|
|
|
|
*relid = left->varno;
|
|
|
|
*attno = left->varattno;
|
1999-07-25 23:07:26 +00:00
|
|
|
*flag = SEL_RIGHT;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-07-25 23:07:26 +00:00
|
|
|
else if (IsA(right, Var) &&
|
|
|
|
(targetrelid == 0 || targetrelid == right->varno))
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
*relid = right->varno;
|
|
|
|
*attno = right->varattno;
|
|
|
|
*flag = 0;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-07-25 23:07:26 +00:00
|
|
|
else if ((funcvarno = is_single_func((Node *) left)) != 0 &&
|
|
|
|
(targetrelid == 0 || targetrelid == funcvarno))
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
*relid = funcvarno;
|
1997-09-07 05:04:48 +00:00
|
|
|
*attno = InvalidAttrNumber;
|
1999-07-25 23:07:26 +00:00
|
|
|
*flag = SEL_RIGHT;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-07-25 23:07:26 +00:00
|
|
|
else if ((funcvarno = is_single_func((Node *) right)) != 0 &&
|
|
|
|
(targetrelid == 0 || targetrelid == funcvarno))
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
*relid = funcvarno;
|
|
|
|
*attno = InvalidAttrNumber;
|
|
|
|
*flag = 0;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-07-25 23:07:26 +00:00
|
|
|
else
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
/* Duh, it's too complicated for me... */
|
|
|
|
default_results:
|
1999-09-09 02:36:04 +00:00
|
|
|
*relid = 0;
|
|
|
|
*attno = 0;
|
1997-09-07 05:04:48 +00:00
|
|
|
*constval = 0;
|
1999-07-25 23:07:26 +00:00
|
|
|
*flag = 0;
|
|
|
|
return;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
|
|
|
|
1999-07-25 23:07:26 +00:00
|
|
|
/* OK, we identified the var or func; now look at the other side */
|
|
|
|
|
|
|
|
other = (*flag == 0) ? left : right;
|
|
|
|
|
2000-04-16 01:55:45 +00:00
|
|
|
if (IsA(other, Const) &&
|
|
|
|
!((Const *) other)->constisnull)
|
1999-07-25 23:07:26 +00:00
|
|
|
{
|
|
|
|
*constval = ((Const *) other)->constvalue;
|
|
|
|
*flag |= SEL_CONSTANT;
|
1999-02-14 22:24:25 +00:00
|
|
|
}
|
|
|
|
else
|
1997-09-07 05:04:48 +00:00
|
|
|
*constval = 0;
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-07-25 23:07:26 +00:00
|
|
|
* is_single_func
|
2000-04-12 17:17:23 +00:00
|
|
|
* If the given expression is a function of a single relation,
|
|
|
|
* return the relation number; else return 0
|
1999-07-25 23:07:26 +00:00
|
|
|
*/
|
2000-04-04 01:21:48 +00:00
|
|
|
static int
|
|
|
|
is_single_func(Node *node)
|
1999-07-25 23:07:26 +00:00
|
|
|
{
|
|
|
|
if (is_funcclause(node))
|
|
|
|
{
|
1999-08-10 03:00:15 +00:00
|
|
|
List *varnos = pull_varnos(node);
|
1999-07-25 23:07:26 +00:00
|
|
|
|
1999-08-10 03:00:15 +00:00
|
|
|
if (length(varnos) == 1)
|
1999-07-25 23:07:26 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
int funcvarno = lfirsti(varnos);
|
1999-08-10 03:00:15 +00:00
|
|
|
|
|
|
|
freeList(varnos);
|
1999-07-25 23:07:26 +00:00
|
|
|
return funcvarno;
|
|
|
|
}
|
1999-08-10 03:00:15 +00:00
|
|
|
freeList(varnos);
|
1999-07-25 23:07:26 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* get_rels_atts
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1999-07-25 23:07:26 +00:00
|
|
|
* Returns the info
|
1997-09-07 05:04:48 +00:00
|
|
|
* ( relid1 attno1 relid2 attno2 )
|
|
|
|
* for a joinclause.
|
|
|
|
*
|
1999-07-25 23:07:26 +00:00
|
|
|
* If the clause is not of the form (var op var) or if any of the vars
|
1999-09-09 02:36:04 +00:00
|
|
|
* refer to nested attributes, then zeroes are returned.
|
1997-09-07 05:04:48 +00:00
|
|
|
*
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
void
|
1997-09-08 21:56:23 +00:00
|
|
|
get_rels_atts(Node *clause,
|
1997-09-07 05:04:48 +00:00
|
|
|
int *relid1,
|
1997-09-08 20:59:27 +00:00
|
|
|
AttrNumber *attno1,
|
1997-09-07 05:04:48 +00:00
|
|
|
int *relid2,
|
1997-09-08 20:59:27 +00:00
|
|
|
AttrNumber *attno2)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
/* set default values */
|
1999-09-09 02:36:04 +00:00
|
|
|
*relid1 = 0;
|
|
|
|
*attno1 = 0;
|
|
|
|
*relid2 = 0;
|
|
|
|
*attno2 = 0;
|
1999-07-25 23:07:26 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
if (is_opclause(clause))
|
|
|
|
{
|
1999-05-25 16:15:34 +00:00
|
|
|
Var *left = get_leftop((Expr *) clause);
|
|
|
|
Var *right = get_rightop((Expr *) clause);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-02-14 22:24:25 +00:00
|
|
|
if (left && right)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
int funcvarno;
|
1999-02-14 22:24:25 +00:00
|
|
|
|
1999-07-25 23:07:26 +00:00
|
|
|
if (IsA(left, Var))
|
1999-02-14 22:24:25 +00:00
|
|
|
{
|
|
|
|
*relid1 = left->varno;
|
1999-07-25 23:07:26 +00:00
|
|
|
*attno1 = left->varattno;
|
1999-02-14 22:24:25 +00:00
|
|
|
}
|
1999-07-25 23:07:26 +00:00
|
|
|
else if ((funcvarno = is_single_func((Node *) left)) != 0)
|
1999-02-14 22:24:25 +00:00
|
|
|
{
|
1999-07-25 23:07:26 +00:00
|
|
|
*relid1 = funcvarno;
|
|
|
|
*attno1 = InvalidAttrNumber;
|
|
|
|
}
|
1999-02-14 22:24:25 +00:00
|
|
|
|
1999-07-25 23:07:26 +00:00
|
|
|
if (IsA(right, Var))
|
|
|
|
{
|
1999-02-14 22:24:25 +00:00
|
|
|
*relid2 = right->varno;
|
1999-07-25 23:07:26 +00:00
|
|
|
*attno2 = right->varattno;
|
|
|
|
}
|
|
|
|
else if ((funcvarno = is_single_func((Node *) right)) != 0)
|
|
|
|
{
|
|
|
|
*relid2 = funcvarno;
|
|
|
|
*attno2 = InvalidAttrNumber;
|
1999-02-14 22:24:25 +00:00
|
|
|
}
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
/*--------------------
|
|
|
|
* CommuteClause: commute a binary operator clause
|
1999-08-09 00:51:26 +00:00
|
|
|
*
|
|
|
|
* XXX the clause is destructively modified!
|
1999-06-19 03:41:45 +00:00
|
|
|
*--------------------
|
|
|
|
*/
|
1996-07-09 06:22:35 +00:00
|
|
|
void
|
1999-08-12 04:32:54 +00:00
|
|
|
CommuteClause(Expr *clause)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
HeapTuple heapTup;
|
1999-08-12 04:32:54 +00:00
|
|
|
Form_pg_operator commuTup;
|
|
|
|
Oper *commu;
|
|
|
|
Node *temp;
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-08-12 04:32:54 +00:00
|
|
|
if (!is_opclause((Node *) clause) ||
|
|
|
|
length(clause->args) != 2)
|
|
|
|
elog(ERROR, "CommuteClause: applied to non-binary-operator clause");
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
heapTup = (HeapTuple)
|
1999-08-12 04:32:54 +00:00
|
|
|
get_operator_tuple(get_commutator(((Oper *) clause->oper)->opno));
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
if (heapTup == (HeapTuple) NULL)
|
1999-05-10 00:46:32 +00:00
|
|
|
elog(ERROR, "CommuteClause: no commutator for operator %u",
|
1999-08-12 04:32:54 +00:00
|
|
|
((Oper *) clause->oper)->opno);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1998-09-01 03:29:17 +00:00
|
|
|
commuTup = (Form_pg_operator) GETSTRUCT(heapTup);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1998-11-27 19:52:36 +00:00
|
|
|
commu = makeOper(heapTup->t_data->t_oid,
|
1999-03-01 00:10:44 +00:00
|
|
|
commuTup->oprcode,
|
1997-09-07 05:04:48 +00:00
|
|
|
commuTup->oprresult,
|
1999-08-12 04:32:54 +00:00
|
|
|
((Oper *) clause->oper)->opsize,
|
1997-09-07 05:04:48 +00:00
|
|
|
NULL);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
1999-03-01 00:10:44 +00:00
|
|
|
* re-form the clause in-place!
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
1999-08-12 04:32:54 +00:00
|
|
|
clause->oper = (Node *) commu;
|
|
|
|
temp = lfirst(clause->args);
|
|
|
|
lfirst(clause->args) = lsecond(clause->args);
|
|
|
|
lsecond(clause->args) = temp;
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-06-19 03:41:45 +00:00
|
|
|
|
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*--------------------
|
|
|
|
* eval_const_expressions
|
|
|
|
*
|
|
|
|
* Reduce any recognizably constant subexpressions of the given
|
|
|
|
* expression tree, for example "2 + 2" => "4". More interestingly,
|
|
|
|
* we can reduce certain boolean expressions even when they contain
|
|
|
|
* non-constant subexpressions: "x OR true" => "true" no matter what
|
2000-04-12 17:17:23 +00:00
|
|
|
* the subexpression x is. (XXX We assume that no such subexpression
|
1999-09-26 02:28:44 +00:00
|
|
|
* will have important side-effects, which is not necessarily a good
|
|
|
|
* assumption in the presence of user-defined functions; do we need a
|
|
|
|
* pg_proc flag that prevents discarding the execution of a function?)
|
|
|
|
*
|
|
|
|
* We do understand that certain functions may deliver non-constant
|
|
|
|
* results even with constant inputs, "nextval()" being the classic
|
|
|
|
* example. Functions that are not marked "proiscachable" in pg_proc
|
|
|
|
* will not be pre-evaluated here, although we will reduce their
|
|
|
|
* arguments as far as possible. Functions that are the arguments
|
|
|
|
* of Iter nodes are also not evaluated.
|
|
|
|
*
|
|
|
|
* We assume that the tree has already been type-checked and contains
|
|
|
|
* only operators and functions that are reasonable to try to execute.
|
|
|
|
*
|
|
|
|
* This routine should be invoked before converting sublinks to subplans
|
|
|
|
* (subselect.c's SS_process_sublinks()). The converted form contains
|
|
|
|
* bogus "Const" nodes that are actually placeholders where the executor
|
|
|
|
* will insert values from the inner plan, and obviously we mustn't try
|
|
|
|
* to reduce the expression as though these were really constants.
|
|
|
|
* As a safeguard, if we happen to find an already-converted SubPlan node,
|
|
|
|
* we will return it unchanged rather than recursing into it.
|
|
|
|
*--------------------
|
|
|
|
*/
|
|
|
|
Node *
|
|
|
|
eval_const_expressions(Node *node)
|
|
|
|
{
|
|
|
|
/* no context or special setup needed, so away we go... */
|
|
|
|
return eval_const_expressions_mutator(node, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Node *
|
2000-04-12 17:17:23 +00:00
|
|
|
eval_const_expressions_mutator(Node *node, void *context)
|
1999-09-26 02:28:44 +00:00
|
|
|
{
|
|
|
|
if (node == NULL)
|
|
|
|
return NULL;
|
|
|
|
if (IsA(node, Expr))
|
|
|
|
{
|
|
|
|
Expr *expr = (Expr *) node;
|
|
|
|
List *args;
|
|
|
|
Const *const_input;
|
2000-04-12 17:17:23 +00:00
|
|
|
Expr *newexpr;
|
1999-09-26 02:28:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Reduce constants in the Expr's arguments. We know args is
|
2000-04-12 17:17:23 +00:00
|
|
|
* either NIL or a List node, so we can call
|
|
|
|
* expression_tree_mutator directly rather than recursing to self.
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
args = (List *) expression_tree_mutator((Node *) expr->args,
|
2000-04-12 17:17:23 +00:00
|
|
|
eval_const_expressions_mutator,
|
1999-09-26 02:28:44 +00:00
|
|
|
(void *) context);
|
|
|
|
|
|
|
|
switch (expr->opType)
|
|
|
|
{
|
|
|
|
case OP_EXPR:
|
|
|
|
case FUNC_EXPR:
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
2000-03-19 18:20:38 +00:00
|
|
|
* Code for op/func case is pretty bulky, so split it out
|
|
|
|
* as a separate function.
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
2000-03-19 18:20:38 +00:00
|
|
|
newexpr = simplify_op_or_func(expr, args);
|
|
|
|
if (newexpr) /* successfully simplified it */
|
|
|
|
return (Node *) newexpr;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* else fall out to build new Expr node with simplified
|
|
|
|
* args
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
2000-04-12 17:17:23 +00:00
|
|
|
break;
|
|
|
|
case OR_EXPR:
|
1999-09-26 02:28:44 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* OR arguments are handled as follows: non constant:
|
|
|
|
* keep FALSE: drop (does not affect result) TRUE:
|
|
|
|
* force result to TRUE NULL: keep only one We keep
|
|
|
|
* one NULL input because ExecEvalOr returns NULL when
|
|
|
|
* no input is TRUE and at least one is NULL.
|
|
|
|
*/
|
|
|
|
List *newargs = NIL;
|
|
|
|
List *arg;
|
|
|
|
bool haveNull = false;
|
|
|
|
bool forceTrue = false;
|
|
|
|
|
|
|
|
foreach(arg, args)
|
1999-09-26 02:28:44 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!IsA(lfirst(arg), Const))
|
|
|
|
{
|
|
|
|
newargs = lappend(newargs, lfirst(arg));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const_input = (Const *) lfirst(arg);
|
|
|
|
if (const_input->constisnull)
|
|
|
|
haveNull = true;
|
|
|
|
else if (DatumGetInt32(const_input->constvalue))
|
|
|
|
forceTrue = true;
|
|
|
|
/* otherwise, we can drop the constant-false input */
|
1999-09-26 02:28:44 +00:00
|
|
|
}
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We could return TRUE before falling out of the
|
|
|
|
* loop, but this coding method will be easier to
|
|
|
|
* adapt if we ever add a notion of non-removable
|
|
|
|
* functions. We'd need to check all the inputs for
|
|
|
|
* non-removability.
|
|
|
|
*/
|
|
|
|
if (forceTrue)
|
|
|
|
return MAKEBOOLCONST(true, false);
|
|
|
|
if (haveNull)
|
|
|
|
newargs = lappend(newargs, MAKEBOOLCONST(false, true));
|
|
|
|
/* If all the inputs are FALSE, result is FALSE */
|
|
|
|
if (newargs == NIL)
|
|
|
|
return MAKEBOOLCONST(false, false);
|
|
|
|
/* If only one nonconst-or-NULL input, it's the result */
|
|
|
|
if (lnext(newargs) == NIL)
|
|
|
|
return (Node *) lfirst(newargs);
|
|
|
|
/* Else we still need an OR node */
|
|
|
|
return (Node *) make_orclause(newargs);
|
1999-09-26 02:28:44 +00:00
|
|
|
}
|
|
|
|
case AND_EXPR:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* AND arguments are handled as follows: non constant:
|
|
|
|
* keep TRUE: drop (does not affect result) FALSE:
|
|
|
|
* force result to FALSE NULL: keep only one We keep
|
|
|
|
* one NULL input because ExecEvalAnd returns NULL
|
|
|
|
* when no input is FALSE and at least one is NULL.
|
|
|
|
*/
|
|
|
|
List *newargs = NIL;
|
|
|
|
List *arg;
|
|
|
|
bool haveNull = false;
|
|
|
|
bool forceFalse = false;
|
|
|
|
|
|
|
|
foreach(arg, args)
|
1999-09-26 02:28:44 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!IsA(lfirst(arg), Const))
|
|
|
|
{
|
|
|
|
newargs = lappend(newargs, lfirst(arg));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const_input = (Const *) lfirst(arg);
|
|
|
|
if (const_input->constisnull)
|
|
|
|
haveNull = true;
|
|
|
|
else if (!DatumGetInt32(const_input->constvalue))
|
|
|
|
forceFalse = true;
|
|
|
|
/* otherwise, we can drop the constant-true input */
|
1999-09-26 02:28:44 +00:00
|
|
|
}
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We could return FALSE before falling out of the
|
|
|
|
* loop, but this coding method will be easier to
|
|
|
|
* adapt if we ever add a notion of non-removable
|
|
|
|
* functions. We'd need to check all the inputs for
|
|
|
|
* non-removability.
|
|
|
|
*/
|
|
|
|
if (forceFalse)
|
|
|
|
return MAKEBOOLCONST(false, false);
|
|
|
|
if (haveNull)
|
|
|
|
newargs = lappend(newargs, MAKEBOOLCONST(false, true));
|
|
|
|
/* If all the inputs are TRUE, result is TRUE */
|
|
|
|
if (newargs == NIL)
|
|
|
|
return MAKEBOOLCONST(true, false);
|
|
|
|
/* If only one nonconst-or-NULL input, it's the result */
|
|
|
|
if (lnext(newargs) == NIL)
|
|
|
|
return (Node *) lfirst(newargs);
|
|
|
|
/* Else we still need an AND node */
|
|
|
|
return (Node *) make_andclause(newargs);
|
1999-09-26 02:28:44 +00:00
|
|
|
}
|
|
|
|
case NOT_EXPR:
|
|
|
|
Assert(length(args) == 1);
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!IsA(lfirst(args), Const))
|
1999-09-26 02:28:44 +00:00
|
|
|
break;
|
|
|
|
const_input = (Const *) lfirst(args);
|
|
|
|
/* NOT NULL => NULL */
|
|
|
|
if (const_input->constisnull)
|
|
|
|
return MAKEBOOLCONST(false, true);
|
|
|
|
/* otherwise pretty easy */
|
2000-04-12 17:17:23 +00:00
|
|
|
return MAKEBOOLCONST(!DatumGetInt32(const_input->constvalue),
|
1999-09-26 02:28:44 +00:00
|
|
|
false);
|
|
|
|
case SUBPLAN_EXPR:
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
|
|
|
* Safety measure per notes at head of this routine:
|
2000-04-12 17:17:23 +00:00
|
|
|
* return a SubPlan unchanged. Too late to do anything
|
1999-09-26 02:28:44 +00:00
|
|
|
* with it. The arglist simplification above was wasted
|
2000-04-12 17:17:23 +00:00
|
|
|
* work (the list probably only contains Var nodes
|
|
|
|
* anyway).
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
return (Node *) expr;
|
|
|
|
default:
|
|
|
|
elog(ERROR, "eval_const_expressions: unexpected opType %d",
|
|
|
|
(int) expr->opType);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we break out of the above switch on opType, then the
|
2000-04-12 17:17:23 +00:00
|
|
|
* expression cannot be simplified any further, so build and
|
|
|
|
* return a replacement Expr node using the possibly-simplified
|
|
|
|
* arguments and the original oper node. Can't use make_clause()
|
|
|
|
* here because we want to be sure the typeOid field is
|
|
|
|
* preserved...
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
newexpr = makeNode(Expr);
|
2000-04-12 17:17:23 +00:00
|
|
|
newexpr->typeOid = expr->typeOid;
|
|
|
|
newexpr->opType = expr->opType;
|
|
|
|
newexpr->oper = expr->oper;
|
|
|
|
newexpr->args = args;
|
|
|
|
return (Node *) newexpr;
|
1999-09-26 02:28:44 +00:00
|
|
|
}
|
2000-02-20 21:32:16 +00:00
|
|
|
if (IsA(node, RelabelType))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-02-20 21:32:16 +00:00
|
|
|
/*
|
|
|
|
* If we can simplify the input to a constant, then we don't need
|
2000-04-12 17:17:23 +00:00
|
|
|
* the RelabelType node anymore: just change the type field of the
|
|
|
|
* Const node. Otherwise, copy the RelabelType node.
|
2000-02-20 21:32:16 +00:00
|
|
|
*/
|
|
|
|
RelabelType *relabel = (RelabelType *) node;
|
|
|
|
Node *arg;
|
|
|
|
|
|
|
|
arg = eval_const_expressions_mutator(relabel->arg, context);
|
|
|
|
if (arg && IsA(arg, Const))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Const *con = (Const *) arg;
|
2000-02-20 21:32:16 +00:00
|
|
|
|
|
|
|
con->consttype = relabel->resulttype;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* relabel's resulttypmod is discarded, which is OK for now;
|
|
|
|
* if the type actually needs a runtime length coercion then
|
2000-04-12 17:17:23 +00:00
|
|
|
* there should be a function call to do it just above this
|
|
|
|
* node.
|
2000-03-19 18:20:38 +00:00
|
|
|
*/
|
2000-02-20 21:32:16 +00:00
|
|
|
return (Node *) con;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
RelabelType *newrelabel = makeNode(RelabelType);
|
|
|
|
|
|
|
|
newrelabel->arg = arg;
|
|
|
|
newrelabel->resulttype = relabel->resulttype;
|
|
|
|
newrelabel->resulttypmod = relabel->resulttypmod;
|
|
|
|
return (Node *) newrelabel;
|
|
|
|
}
|
|
|
|
}
|
1999-09-26 02:28:44 +00:00
|
|
|
if (IsA(node, CaseExpr))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* CASE expressions can be simplified if there are constant
|
|
|
|
* condition clauses: FALSE (or NULL): drop the alternative TRUE:
|
|
|
|
* drop all remaining alternatives If the first non-FALSE
|
|
|
|
* alternative is a constant TRUE, we can simplify the entire CASE
|
|
|
|
* to that alternative's expression. If there are no non-FALSE
|
|
|
|
* alternatives, we simplify the entire CASE to the default result
|
|
|
|
* (ELSE result).
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
CaseExpr *caseexpr = (CaseExpr *) node;
|
|
|
|
CaseExpr *newcase;
|
|
|
|
List *newargs = NIL;
|
|
|
|
Node *defresult;
|
|
|
|
Const *const_input;
|
|
|
|
List *arg;
|
|
|
|
|
|
|
|
foreach(arg, caseexpr->args)
|
|
|
|
{
|
|
|
|
/* Simplify this alternative's condition and result */
|
|
|
|
CaseWhen *casewhen = (CaseWhen *)
|
2000-04-12 17:17:23 +00:00
|
|
|
expression_tree_mutator((Node *) lfirst(arg),
|
|
|
|
eval_const_expressions_mutator,
|
|
|
|
(void *) context);
|
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
Assert(IsA(casewhen, CaseWhen));
|
|
|
|
if (casewhen->expr == NULL ||
|
2000-04-12 17:17:23 +00:00
|
|
|
!IsA(casewhen->expr, Const))
|
1999-09-26 02:28:44 +00:00
|
|
|
{
|
|
|
|
newargs = lappend(newargs, casewhen);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const_input = (Const *) casewhen->expr;
|
|
|
|
if (const_input->constisnull ||
|
2000-04-12 17:17:23 +00:00
|
|
|
!DatumGetInt32(const_input->constvalue))
|
1999-09-26 02:28:44 +00:00
|
|
|
continue; /* drop alternative with FALSE condition */
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* Found a TRUE condition. If it's the first (un-dropped)
|
1999-09-26 02:28:44 +00:00
|
|
|
* alternative, the CASE reduces to just this alternative.
|
|
|
|
*/
|
|
|
|
if (newargs == NIL)
|
|
|
|
return casewhen->result;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
|
|
|
* Otherwise, add it to the list, and drop all the rest.
|
|
|
|
*/
|
|
|
|
newargs = lappend(newargs, casewhen);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Simplify the default result */
|
|
|
|
defresult = eval_const_expressions_mutator(caseexpr->defresult,
|
|
|
|
context);
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If no non-FALSE alternatives, CASE reduces to the default
|
|
|
|
* result
|
|
|
|
*/
|
1999-09-26 02:28:44 +00:00
|
|
|
if (newargs == NIL)
|
|
|
|
return defresult;
|
|
|
|
/* Otherwise we need a new CASE node */
|
|
|
|
newcase = makeNode(CaseExpr);
|
|
|
|
newcase->casetype = caseexpr->casetype;
|
|
|
|
newcase->arg = NULL;
|
|
|
|
newcase->args = newargs;
|
|
|
|
newcase->defresult = defresult;
|
|
|
|
return (Node *) newcase;
|
|
|
|
}
|
|
|
|
if (IsA(node, Iter))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* The argument of an Iter is normally a function call. We must
|
|
|
|
* not try to eliminate the function, but we can try to simplify
|
|
|
|
* its arguments. If, by chance, the arg is NOT a function then
|
|
|
|
* we go ahead and try to simplify it (by falling into
|
|
|
|
* expression_tree_mutator). Is that the right thing?
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
Iter *iter = (Iter *) node;
|
|
|
|
|
|
|
|
if (is_funcclause(iter->iterexpr))
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Expr *func = (Expr *) iter->iterexpr;
|
|
|
|
Expr *newfunc;
|
|
|
|
Iter *newiter;
|
1999-09-26 02:28:44 +00:00
|
|
|
|
|
|
|
newfunc = makeNode(Expr);
|
|
|
|
newfunc->typeOid = func->typeOid;
|
|
|
|
newfunc->opType = func->opType;
|
|
|
|
newfunc->oper = func->oper;
|
|
|
|
newfunc->args = (List *)
|
|
|
|
expression_tree_mutator((Node *) func->args,
|
|
|
|
eval_const_expressions_mutator,
|
|
|
|
(void *) context);
|
|
|
|
newiter = makeNode(Iter);
|
|
|
|
newiter->iterexpr = (Node *) newfunc;
|
|
|
|
newiter->itertype = iter->itertype;
|
|
|
|
return (Node *) newiter;
|
|
|
|
}
|
|
|
|
}
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-09-26 02:28:44 +00:00
|
|
|
/*
|
|
|
|
* For any node type not handled above, we recurse using
|
2000-04-12 17:17:23 +00:00
|
|
|
* expression_tree_mutator, which will copy the node unchanged but try
|
|
|
|
* to simplify its arguments (if any) using this routine. For example:
|
|
|
|
* we cannot eliminate an ArrayRef node, but we might be able to
|
|
|
|
* simplify constant expressions in its subscripts.
|
1999-09-26 02:28:44 +00:00
|
|
|
*/
|
|
|
|
return expression_tree_mutator(node, eval_const_expressions_mutator,
|
|
|
|
(void *) context);
|
|
|
|
}
|
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* Subroutine for eval_const_expressions: try to evaluate an op or func
|
|
|
|
*
|
|
|
|
* Inputs are the op or func Expr node, and the pre-simplified argument list.
|
|
|
|
* Returns a simplified expression if successful, or NULL if cannot
|
|
|
|
* simplify the op/func.
|
|
|
|
*
|
|
|
|
* XXX Possible future improvement: if the func is SQL-language, and its
|
|
|
|
* definition is simply "SELECT expression", we could parse and substitute
|
|
|
|
* the expression here. This would avoid much runtime overhead, and perhaps
|
|
|
|
* expose opportunities for constant-folding within the expression even if
|
|
|
|
* not all the func's input args are constants. It'd be appropriate to do
|
2000-05-28 20:33:28 +00:00
|
|
|
* that here, not in the parser, since we wouldn't want it to happen until
|
2000-03-19 18:20:38 +00:00
|
|
|
* after rule substitution/rewriting.
|
|
|
|
*/
|
|
|
|
static Expr *
|
|
|
|
simplify_op_or_func(Expr *expr, List *args)
|
|
|
|
{
|
|
|
|
List *arg;
|
|
|
|
Oid funcid;
|
|
|
|
Oid result_typeid;
|
|
|
|
HeapTuple func_tuple;
|
|
|
|
Form_pg_proc funcform;
|
|
|
|
Type resultType;
|
2000-04-12 17:17:23 +00:00
|
|
|
Expr *newexpr;
|
2000-03-19 18:20:38 +00:00
|
|
|
Datum const_val;
|
2000-05-28 20:33:28 +00:00
|
|
|
bool has_nonconst_input = false;
|
|
|
|
bool has_null_input = false;
|
2000-03-19 18:20:38 +00:00
|
|
|
bool const_is_null;
|
|
|
|
bool isDone;
|
|
|
|
|
|
|
|
/*
|
2000-05-28 20:33:28 +00:00
|
|
|
* Check for constant inputs and especially constant-NULL inputs.
|
2000-03-19 18:20:38 +00:00
|
|
|
*/
|
|
|
|
foreach(arg, args)
|
|
|
|
{
|
2000-05-28 20:33:28 +00:00
|
|
|
if (IsA(lfirst(arg), Const))
|
|
|
|
has_null_input |= ((Const *) lfirst(arg))->constisnull;
|
|
|
|
else
|
|
|
|
has_nonconst_input = true;
|
2000-03-19 18:20:38 +00:00
|
|
|
}
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-05-28 20:33:28 +00:00
|
|
|
/*
|
|
|
|
* If the function is strict and has a constant-NULL input, it will
|
|
|
|
* never be called at all, so we can replace the call by a NULL
|
|
|
|
* constant even if there are other inputs that aren't constant.
|
|
|
|
* Otherwise, we can only simplify if all inputs are constants.
|
|
|
|
* We can skip the function lookup if neither case applies.
|
|
|
|
*/
|
|
|
|
if (has_nonconst_input && !has_null_input)
|
|
|
|
return NULL;
|
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* Get the function procedure's OID and look to see whether it is
|
|
|
|
* marked proiscachable.
|
2000-05-28 20:33:28 +00:00
|
|
|
*
|
|
|
|
* XXX would it be better to take the result type from the pg_proc tuple,
|
|
|
|
* rather than the Oper or Func node?
|
2000-03-19 18:20:38 +00:00
|
|
|
*/
|
|
|
|
if (expr->opType == OP_EXPR)
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Oper *oper = (Oper *) expr->oper;
|
2000-03-19 18:20:38 +00:00
|
|
|
|
|
|
|
replace_opid(oper); /* OK to scribble on input to this extent */
|
|
|
|
funcid = oper->opid;
|
|
|
|
result_typeid = oper->opresulttype;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Func *func = (Func *) expr->oper;
|
2000-03-19 18:20:38 +00:00
|
|
|
|
|
|
|
funcid = func->funcid;
|
|
|
|
result_typeid = func->functype;
|
|
|
|
}
|
|
|
|
/* Someday lsyscache.c might provide a function for this */
|
|
|
|
func_tuple = SearchSysCacheTuple(PROCOID,
|
|
|
|
ObjectIdGetDatum(funcid),
|
|
|
|
0, 0, 0);
|
|
|
|
if (!HeapTupleIsValid(func_tuple))
|
|
|
|
elog(ERROR, "Function OID %u does not exist", funcid);
|
|
|
|
funcform = (Form_pg_proc) GETSTRUCT(func_tuple);
|
2000-04-12 17:17:23 +00:00
|
|
|
if (!funcform->proiscachable)
|
2000-03-19 18:20:38 +00:00
|
|
|
return NULL;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* Also check to make sure it doesn't return a set.
|
|
|
|
*/
|
|
|
|
if (funcform->proretset)
|
|
|
|
return NULL;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-05-28 20:33:28 +00:00
|
|
|
/*
|
|
|
|
* Now that we know if the function is strict, we can finish the
|
|
|
|
* checks for simplifiable inputs that we started above.
|
|
|
|
*/
|
|
|
|
if (funcform->proisstrict && has_null_input)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* It's strict and has NULL input, so must produce NULL output.
|
|
|
|
* Return a NULL constant of the right type.
|
|
|
|
*/
|
|
|
|
resultType = typeidType(result_typeid);
|
|
|
|
return (Expr *) makeConst(result_typeid, typeLen(resultType),
|
|
|
|
(Datum) 0, true,
|
|
|
|
typeByVal(resultType),
|
|
|
|
false, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Otherwise, can simplify only if all inputs are constants.
|
|
|
|
* (For a non-strict function, constant NULL inputs are treated
|
|
|
|
* the same as constant non-NULL inputs.)
|
|
|
|
*/
|
|
|
|
if (has_nonconst_input)
|
|
|
|
return NULL;
|
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* OK, looks like we can simplify this operator/function.
|
|
|
|
*
|
|
|
|
* We use the executor's routine ExecEvalExpr() to avoid duplication of
|
|
|
|
* code and ensure we get the same result as the executor would get.
|
|
|
|
*
|
2000-04-12 17:17:23 +00:00
|
|
|
* Build a new Expr node containing the already-simplified arguments. The
|
|
|
|
* only other setup needed here is the replace_opid() that we already
|
2000-03-19 18:20:38 +00:00
|
|
|
* did for the OP_EXPR case.
|
|
|
|
*/
|
|
|
|
newexpr = makeNode(Expr);
|
|
|
|
newexpr->typeOid = expr->typeOid;
|
|
|
|
newexpr->opType = expr->opType;
|
|
|
|
newexpr->oper = expr->oper;
|
|
|
|
newexpr->args = args;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* It is OK to pass econtext = NULL because none of the ExecEvalExpr()
|
|
|
|
* code used in this situation will use econtext. That might seem
|
2000-04-12 17:17:23 +00:00
|
|
|
* fortuitous, but it's not so unreasonable --- a constant expression
|
|
|
|
* does not depend on context, by definition, n'est ce pas?
|
2000-03-19 18:20:38 +00:00
|
|
|
*/
|
|
|
|
const_val = ExecEvalExpr((Node *) newexpr, NULL,
|
|
|
|
&const_is_null, &isDone);
|
|
|
|
Assert(isDone); /* if this isn't set, we blew it... */
|
|
|
|
pfree(newexpr);
|
2000-04-12 17:17:23 +00:00
|
|
|
|
2000-03-19 18:20:38 +00:00
|
|
|
/*
|
|
|
|
* Make the constant result node.
|
|
|
|
*/
|
|
|
|
resultType = typeidType(result_typeid);
|
|
|
|
return (Expr *) makeConst(result_typeid, typeLen(resultType),
|
|
|
|
const_val, const_is_null,
|
|
|
|
typeByVal(resultType),
|
|
|
|
false, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-08-09 00:51:26 +00:00
|
|
|
/*
|
1999-06-19 03:41:45 +00:00
|
|
|
* Standard expression-tree walking support
|
|
|
|
*
|
|
|
|
* We used to have near-duplicate code in many different routines that
|
|
|
|
* understood how to recurse through an expression node tree. That was
|
|
|
|
* a pain to maintain, and we frequently had bugs due to some particular
|
|
|
|
* routine neglecting to support a particular node type. In most cases,
|
|
|
|
* these routines only actually care about certain node types, and don't
|
|
|
|
* care about other types except insofar as they have to recurse through
|
|
|
|
* non-primitive node types. Therefore, we now provide generic tree-walking
|
1999-08-09 00:51:26 +00:00
|
|
|
* logic to consolidate the redundant "boilerplate" code. There are
|
|
|
|
* two versions: expression_tree_walker() and expression_tree_mutator().
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*--------------------
|
1999-06-19 03:41:45 +00:00
|
|
|
* expression_tree_walker() is designed to support routines that traverse
|
|
|
|
* a tree in a read-only fashion (although it will also work for routines
|
1999-08-09 00:51:26 +00:00
|
|
|
* that modify nodes in-place but never add/delete/replace nodes).
|
|
|
|
* A walker routine should look like this:
|
1999-06-19 03:41:45 +00:00
|
|
|
*
|
|
|
|
* bool my_walker (Node *node, my_struct *context)
|
|
|
|
* {
|
|
|
|
* if (node == NULL)
|
|
|
|
* return false;
|
|
|
|
* // check for nodes that special work is required for, eg:
|
|
|
|
* if (IsA(node, Var))
|
|
|
|
* {
|
|
|
|
* ... do special actions for Var nodes
|
|
|
|
* }
|
|
|
|
* else if (IsA(node, ...))
|
|
|
|
* {
|
|
|
|
* ... do special actions for other node types
|
|
|
|
* }
|
|
|
|
* // for any node type not specially processed, do:
|
|
|
|
* return expression_tree_walker(node, my_walker, (void *) context);
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* The "context" argument points to a struct that holds whatever context
|
1999-08-09 00:51:26 +00:00
|
|
|
* information the walker routine needs --- it can be used to return data
|
|
|
|
* gathered by the walker, too. This argument is not touched by
|
1999-06-19 03:41:45 +00:00
|
|
|
* expression_tree_walker, but it is passed down to recursive sub-invocations
|
|
|
|
* of my_walker. The tree walk is started from a setup routine that
|
|
|
|
* fills in the appropriate context struct, calls my_walker with the top-level
|
|
|
|
* node of the tree, and then examines the results.
|
|
|
|
*
|
|
|
|
* The walker routine should return "false" to continue the tree walk, or
|
|
|
|
* "true" to abort the walk and immediately return "true" to the top-level
|
2000-04-12 17:17:23 +00:00
|
|
|
* caller. This can be used to short-circuit the traversal if the walker
|
|
|
|
* has found what it came for. "false" is returned to the top-level caller
|
1999-06-21 01:18:02 +00:00
|
|
|
* iff no invocation of the walker returned "true".
|
1999-06-19 03:41:45 +00:00
|
|
|
*
|
|
|
|
* The node types handled by expression_tree_walker include all those
|
|
|
|
* normally found in target lists and qualifier clauses during the planning
|
|
|
|
* stage. In particular, it handles List nodes since a cnf-ified qual clause
|
|
|
|
* will have List structure at the top level, and it handles TargetEntry nodes
|
|
|
|
* so that a scan of a target list can be handled without additional code.
|
|
|
|
* (But only the "expr" part of a TargetEntry is examined, unless the walker
|
|
|
|
* chooses to process TargetEntry nodes specially.)
|
|
|
|
*
|
|
|
|
* expression_tree_walker will handle a SUBPLAN_EXPR node by recursing into
|
|
|
|
* the args and slink->oper lists (which belong to the outer plan), but it
|
|
|
|
* will *not* visit the inner plan, since that's typically what expression
|
|
|
|
* tree walkers want. A walker that wants to visit the subplan can force
|
1999-08-25 23:21:43 +00:00
|
|
|
* appropriate behavior by recognizing subplan expression nodes and doing
|
|
|
|
* the right thing.
|
1999-06-19 03:41:45 +00:00
|
|
|
*
|
1999-06-21 01:18:02 +00:00
|
|
|
* Bare SubLink nodes (without a SUBPLAN_EXPR) are handled by recursing into
|
|
|
|
* the "lefthand" argument list only. (A bare SubLink should be seen only if
|
|
|
|
* the tree has not yet been processed by subselect.c.) Again, this can be
|
|
|
|
* overridden by the walker, but it seems to be the most useful default
|
|
|
|
* behavior.
|
1999-06-19 03:41:45 +00:00
|
|
|
*--------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
2000-04-12 17:17:23 +00:00
|
|
|
expression_tree_walker(Node *node, bool (*walker) (), void *context)
|
1999-06-19 03:41:45 +00:00
|
|
|
{
|
|
|
|
List *temp;
|
|
|
|
|
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* The walker has already visited the current node, and so we need
|
|
|
|
* only recurse into any sub-nodes it has.
|
1999-06-19 03:41:45 +00:00
|
|
|
*
|
2000-04-12 17:17:23 +00:00
|
|
|
* We assume that the walker is not interested in List nodes per se, so
|
|
|
|
* when we expect a List we just recurse directly to self without
|
1999-06-19 03:41:45 +00:00
|
|
|
* bothering to call the walker.
|
|
|
|
*/
|
|
|
|
if (node == NULL)
|
|
|
|
return false;
|
|
|
|
switch (nodeTag(node))
|
|
|
|
{
|
|
|
|
case T_Ident:
|
|
|
|
case T_Const:
|
|
|
|
case T_Var:
|
|
|
|
case T_Param:
|
|
|
|
/* primitive node types with no subnodes */
|
|
|
|
break;
|
|
|
|
case T_Expr:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Expr *expr = (Expr *) node;
|
1999-08-25 23:21:43 +00:00
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
if (expr->opType == SUBPLAN_EXPR)
|
|
|
|
{
|
1999-08-25 23:21:43 +00:00
|
|
|
/* recurse to the SubLink node (skipping SubPlan!) */
|
|
|
|
if (walker((Node *) ((SubPlan *) expr->oper)->sublink,
|
|
|
|
context))
|
1999-06-19 03:41:45 +00:00
|
|
|
return true;
|
|
|
|
}
|
1999-08-25 23:21:43 +00:00
|
|
|
/* for all Expr node types, examine args list */
|
|
|
|
if (expression_tree_walker((Node *) expr->args,
|
|
|
|
walker, context))
|
|
|
|
return true;
|
1999-06-19 03:41:45 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_Aggref:
|
|
|
|
return walker(((Aggref *) node)->target, context);
|
|
|
|
case T_Iter:
|
|
|
|
return walker(((Iter *) node)->iterexpr, context);
|
|
|
|
case T_ArrayRef:
|
|
|
|
{
|
|
|
|
ArrayRef *aref = (ArrayRef *) node;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
/* recurse directly for upper/lower array index lists */
|
|
|
|
if (expression_tree_walker((Node *) aref->refupperindexpr,
|
|
|
|
walker, context))
|
|
|
|
return true;
|
|
|
|
if (expression_tree_walker((Node *) aref->reflowerindexpr,
|
|
|
|
walker, context))
|
|
|
|
return true;
|
|
|
|
/* walker must see the refexpr and refassgnexpr, however */
|
|
|
|
if (walker(aref->refexpr, context))
|
|
|
|
return true;
|
|
|
|
if (walker(aref->refassgnexpr, context))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
2000-02-20 21:32:16 +00:00
|
|
|
case T_RelabelType:
|
|
|
|
return walker(((RelabelType *) node)->arg, context);
|
1999-06-19 03:41:45 +00:00
|
|
|
case T_CaseExpr:
|
|
|
|
{
|
|
|
|
CaseExpr *caseexpr = (CaseExpr *) node;
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
/* we assume walker doesn't care about CaseWhens, either */
|
|
|
|
foreach(temp, caseexpr->args)
|
|
|
|
{
|
|
|
|
CaseWhen *when = (CaseWhen *) lfirst(temp);
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-06-19 03:41:45 +00:00
|
|
|
Assert(IsA(when, CaseWhen));
|
|
|
|
if (walker(when->expr, context))
|
|
|
|
return true;
|
|
|
|
if (walker(when->result, context))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
/* caseexpr->arg should be null, but we'll check it anyway */
|
|
|
|
if (walker(caseexpr->arg, context))
|
|
|
|
return true;
|
|
|
|
if (walker(caseexpr->defresult, context))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
1999-06-21 01:18:02 +00:00
|
|
|
case T_SubLink:
|
1999-08-25 23:21:43 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
SubLink *sublink = (SubLink *) node;
|
1999-08-25 23:21:43 +00:00
|
|
|
|
2000-04-12 17:17:23 +00:00
|
|
|
/*
|
|
|
|
* If the SubLink has already been processed by
|
|
|
|
* subselect.c, it will have lefthand=NIL, and we only
|
|
|
|
* need to look at the oper list. Otherwise we only need
|
|
|
|
* to look at lefthand (the Oper nodes in the oper list
|
|
|
|
* are deemed uninteresting).
|
1999-08-25 23:21:43 +00:00
|
|
|
*/
|
|
|
|
if (sublink->lefthand)
|
|
|
|
return walker((Node *) sublink->lefthand, context);
|
|
|
|
else
|
|
|
|
return walker((Node *) sublink->oper, context);
|
|
|
|
}
|
|
|
|
break;
|
1999-06-19 03:41:45 +00:00
|
|
|
case T_List:
|
|
|
|
foreach(temp, (List *) node)
|
|
|
|
{
|
|
|
|
if (walker((Node *) lfirst(temp), context))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_TargetEntry:
|
|
|
|
return walker(((TargetEntry *) node)->expr, context);
|
|
|
|
default:
|
|
|
|
elog(ERROR, "expression_tree_walker: Unexpected node type %d",
|
|
|
|
nodeTag(node));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
/*--------------------
|
|
|
|
* expression_tree_mutator() is designed to support routines that make a
|
|
|
|
* modified copy of an expression tree, with some nodes being added,
|
|
|
|
* removed, or replaced by new subtrees. The original tree is (normally)
|
|
|
|
* not changed. Each recursion level is responsible for returning a copy of
|
|
|
|
* (or appropriately modified substitute for) the subtree it is handed.
|
|
|
|
* A mutator routine should look like this:
|
|
|
|
*
|
|
|
|
* Node * my_mutator (Node *node, my_struct *context)
|
|
|
|
* {
|
|
|
|
* if (node == NULL)
|
|
|
|
* return NULL;
|
|
|
|
* // check for nodes that special work is required for, eg:
|
|
|
|
* if (IsA(node, Var))
|
|
|
|
* {
|
|
|
|
* ... create and return modified copy of Var node
|
|
|
|
* }
|
|
|
|
* else if (IsA(node, ...))
|
|
|
|
* {
|
|
|
|
* ... do special transformations of other node types
|
|
|
|
* }
|
|
|
|
* // for any node type not specially processed, do:
|
|
|
|
* return expression_tree_mutator(node, my_mutator, (void *) context);
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* The "context" argument points to a struct that holds whatever context
|
|
|
|
* information the mutator routine needs --- it can be used to return extra
|
|
|
|
* data gathered by the mutator, too. This argument is not touched by
|
|
|
|
* expression_tree_mutator, but it is passed down to recursive sub-invocations
|
|
|
|
* of my_mutator. The tree walk is started from a setup routine that
|
|
|
|
* fills in the appropriate context struct, calls my_mutator with the
|
|
|
|
* top-level node of the tree, and does any required post-processing.
|
|
|
|
*
|
|
|
|
* Each level of recursion must return an appropriately modified Node.
|
|
|
|
* If expression_tree_mutator() is called, it will make an exact copy
|
|
|
|
* of the given Node, but invoke my_mutator() to copy the sub-node(s)
|
|
|
|
* of that Node. In this way, my_mutator() has full control over the
|
|
|
|
* copying process but need not directly deal with expression trees
|
|
|
|
* that it has no interest in.
|
|
|
|
*
|
|
|
|
* Just as for expression_tree_walker, the node types handled by
|
|
|
|
* expression_tree_mutator include all those normally found in target lists
|
|
|
|
* and qualifier clauses during the planning stage.
|
|
|
|
*
|
|
|
|
* expression_tree_mutator will handle a SUBPLAN_EXPR node by recursing into
|
|
|
|
* the args and slink->oper lists (which belong to the outer plan), but it
|
|
|
|
* will simply copy the link to the inner plan, since that's typically what
|
|
|
|
* expression tree mutators want. A mutator that wants to modify the subplan
|
1999-08-25 23:21:43 +00:00
|
|
|
* can force appropriate behavior by recognizing subplan expression nodes
|
|
|
|
* and doing the right thing.
|
1999-08-09 00:51:26 +00:00
|
|
|
*
|
|
|
|
* Bare SubLink nodes (without a SUBPLAN_EXPR) are handled by recursing into
|
|
|
|
* the "lefthand" argument list only. (A bare SubLink should be seen only if
|
|
|
|
* the tree has not yet been processed by subselect.c.) Again, this can be
|
|
|
|
* overridden by the mutator, but it seems to be the most useful default
|
|
|
|
* behavior.
|
|
|
|
*--------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
Node *
|
2000-04-12 17:17:23 +00:00
|
|
|
expression_tree_mutator(Node *node, Node *(*mutator) (), void *context)
|
1999-08-09 00:51:26 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
1999-08-09 00:51:26 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* The mutator has already decided not to modify the current node, but
|
|
|
|
* we must call the mutator for any sub-nodes.
|
1999-08-09 00:51:26 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#define FLATCOPY(newnode, node, nodetype) \
|
|
|
|
( (newnode) = makeNode(nodetype), \
|
|
|
|
memcpy((newnode), (node), sizeof(nodetype)) )
|
|
|
|
|
2000-04-12 17:17:23 +00:00
|
|
|
#define CHECKFLATCOPY(newnode, node, nodetype) \
|
1999-08-09 00:51:26 +00:00
|
|
|
( AssertMacro(IsA((node), nodetype)), \
|
|
|
|
(newnode) = makeNode(nodetype), \
|
|
|
|
memcpy((newnode), (node), sizeof(nodetype)) )
|
|
|
|
|
|
|
|
#define MUTATE(newfield, oldfield, fieldtype) \
|
|
|
|
( (newfield) = (fieldtype) mutator((Node *) (oldfield), context) )
|
|
|
|
|
|
|
|
if (node == NULL)
|
|
|
|
return NULL;
|
|
|
|
switch (nodeTag(node))
|
|
|
|
{
|
|
|
|
case T_Ident:
|
|
|
|
case T_Const:
|
|
|
|
case T_Var:
|
|
|
|
case T_Param:
|
|
|
|
/* primitive node types with no subnodes */
|
|
|
|
return (Node *) copyObject(node);
|
|
|
|
case T_Expr:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Expr *expr = (Expr *) node;
|
|
|
|
Expr *newnode;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
FLATCOPY(newnode, expr, Expr);
|
|
|
|
|
|
|
|
if (expr->opType == SUBPLAN_EXPR)
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
SubLink *oldsublink = ((SubPlan *) expr->oper)->sublink;
|
|
|
|
SubPlan *newsubplan;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
/* flat-copy the oper node, which is a SubPlan */
|
|
|
|
CHECKFLATCOPY(newsubplan, expr->oper, SubPlan);
|
|
|
|
newnode->oper = (Node *) newsubplan;
|
|
|
|
/* likewise its SubLink node */
|
|
|
|
CHECKFLATCOPY(newsubplan->sublink, oldsublink, SubLink);
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* transform args list (params to be passed to
|
|
|
|
* subplan)
|
|
|
|
*/
|
1999-08-09 00:51:26 +00:00
|
|
|
MUTATE(newnode->args, expr->args, List *);
|
|
|
|
/* transform sublink's oper list as well */
|
2000-04-12 17:17:23 +00:00
|
|
|
MUTATE(newsubplan->sublink->oper, oldsublink->oper, List *);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* but not the subplan itself, which is referenced
|
|
|
|
* as-is
|
|
|
|
*/
|
1999-08-09 00:51:26 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* for other Expr node types, just transform args
|
|
|
|
* list, linking to original oper node (OK?)
|
1999-08-09 00:51:26 +00:00
|
|
|
*/
|
|
|
|
MUTATE(newnode->args, expr->args, List *);
|
|
|
|
}
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_Aggref:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Aggref *aggref = (Aggref *) node;
|
|
|
|
Aggref *newnode;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
FLATCOPY(newnode, aggref, Aggref);
|
|
|
|
MUTATE(newnode->target, aggref->target, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_Iter:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
Iter *iter = (Iter *) node;
|
|
|
|
Iter *newnode;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
FLATCOPY(newnode, iter, Iter);
|
|
|
|
MUTATE(newnode->iterexpr, iter->iterexpr, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_ArrayRef:
|
|
|
|
{
|
|
|
|
ArrayRef *arrayref = (ArrayRef *) node;
|
|
|
|
ArrayRef *newnode;
|
|
|
|
|
|
|
|
FLATCOPY(newnode, arrayref, ArrayRef);
|
|
|
|
MUTATE(newnode->refupperindexpr, arrayref->refupperindexpr,
|
|
|
|
List *);
|
|
|
|
MUTATE(newnode->reflowerindexpr, arrayref->reflowerindexpr,
|
|
|
|
List *);
|
|
|
|
MUTATE(newnode->refexpr, arrayref->refexpr,
|
|
|
|
Node *);
|
|
|
|
MUTATE(newnode->refassgnexpr, arrayref->refassgnexpr,
|
|
|
|
Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
2000-02-20 21:32:16 +00:00
|
|
|
case T_RelabelType:
|
|
|
|
{
|
|
|
|
RelabelType *relabel = (RelabelType *) node;
|
|
|
|
RelabelType *newnode;
|
|
|
|
|
|
|
|
FLATCOPY(newnode, relabel, RelabelType);
|
|
|
|
MUTATE(newnode->arg, relabel->arg, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
1999-08-09 00:51:26 +00:00
|
|
|
case T_CaseExpr:
|
|
|
|
{
|
|
|
|
CaseExpr *caseexpr = (CaseExpr *) node;
|
|
|
|
CaseExpr *newnode;
|
|
|
|
|
|
|
|
FLATCOPY(newnode, caseexpr, CaseExpr);
|
|
|
|
MUTATE(newnode->args, caseexpr->args, List *);
|
|
|
|
/* caseexpr->arg should be null, but we'll check it anyway */
|
|
|
|
MUTATE(newnode->arg, caseexpr->arg, Node *);
|
|
|
|
MUTATE(newnode->defresult, caseexpr->defresult, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_CaseWhen:
|
|
|
|
{
|
|
|
|
CaseWhen *casewhen = (CaseWhen *) node;
|
|
|
|
CaseWhen *newnode;
|
|
|
|
|
|
|
|
FLATCOPY(newnode, casewhen, CaseWhen);
|
|
|
|
MUTATE(newnode->expr, casewhen->expr, Node *);
|
|
|
|
MUTATE(newnode->result, casewhen->result, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_SubLink:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A "bare" SubLink (note we will not come here if we
|
|
|
|
* found a SUBPLAN_EXPR node above it). Transform the
|
|
|
|
* lefthand side, but not the oper list nor the subquery.
|
1999-08-09 00:51:26 +00:00
|
|
|
*/
|
2000-04-12 17:17:23 +00:00
|
|
|
SubLink *sublink = (SubLink *) node;
|
|
|
|
SubLink *newnode;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
FLATCOPY(newnode, sublink, SubLink);
|
|
|
|
MUTATE(newnode->lefthand, sublink->lefthand, List *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_List:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We assume the mutator isn't interested in the list
|
|
|
|
* nodes per se, so just invoke it on each list element.
|
|
|
|
* NOTE: this would fail badly on a list with integer
|
|
|
|
* elements!
|
1999-08-09 00:51:26 +00:00
|
|
|
*/
|
|
|
|
List *resultlist = NIL;
|
|
|
|
List *temp;
|
|
|
|
|
|
|
|
foreach(temp, (List *) node)
|
|
|
|
{
|
|
|
|
resultlist = lappend(resultlist,
|
|
|
|
mutator((Node *) lfirst(temp),
|
|
|
|
context));
|
|
|
|
}
|
|
|
|
return (Node *) resultlist;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case T_TargetEntry:
|
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We mutate the expression, but not the resdom, by
|
|
|
|
* default.
|
|
|
|
*/
|
|
|
|
TargetEntry *targetentry = (TargetEntry *) node;
|
|
|
|
TargetEntry *newnode;
|
1999-08-09 00:51:26 +00:00
|
|
|
|
|
|
|
FLATCOPY(newnode, targetentry, TargetEntry);
|
|
|
|
MUTATE(newnode->expr, targetentry->expr, Node *);
|
|
|
|
return (Node *) newnode;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
elog(ERROR, "expression_tree_mutator: Unexpected node type %d",
|
|
|
|
nodeTag(node));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* can't get here, but keep compiler happy */
|
|
|
|
return NULL;
|
|
|
|
}
|