preptlist.c

来自「PostgreSQL7.4.6 for Linux」· C语言 代码 · 共 285 行

C
285
字号
/*------------------------------------------------------------------------- * * preptlist.c *	  Routines to preprocess the parse tree target list * * This module takes care of altering the query targetlist as needed for * INSERT, UPDATE, and DELETE queries.	For INSERT and UPDATE queries, * the targetlist must contain an entry for each attribute of the target * relation in the correct order.  For both UPDATE and DELETE queries, * we need a junk targetlist entry holding the CTID attribute --- the * executor relies on this to find the tuple to be replaced/deleted. * * * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * * IDENTIFICATION *	  $Header: /cvsroot/pgsql/src/backend/optimizer/prep/preptlist.c,v 1.65 2003/08/11 23:04:49 tgl Exp $ * *------------------------------------------------------------------------- */#include "postgres.h"#include "access/heapam.h"#include "catalog/pg_type.h"#include "nodes/makefuncs.h"#include "optimizer/prep.h"#include "parser/parsetree.h"#include "parser/parse_coerce.h"static List *expand_targetlist(List *tlist, int command_type,				  Index result_relation, List *range_table);/* * preprocess_targetlist *	  Driver for preprocessing the parse tree targetlist. * *	  Returns the new targetlist. */List *preprocess_targetlist(List *tlist,					  int command_type,					  Index result_relation,					  List *range_table){	/*	 * Sanity check: if there is a result relation, it'd better be a real	 * relation not a subquery.  Else parser or rewriter messed up.	 */	if (result_relation)	{		RangeTblEntry *rte = rt_fetch(result_relation, range_table);		if (rte->subquery != NULL || rte->relid == InvalidOid)			elog(ERROR, "subquery cannot be result relation");	}	/*	 * for heap_formtuple to work, the targetlist must match the exact	 * order of the attributes. We also need to fill in any missing	 * attributes.							-ay 10/94	 */	if (command_type == CMD_INSERT || command_type == CMD_UPDATE)		tlist = expand_targetlist(tlist, command_type,								  result_relation, range_table);	/*	 * for "update" and "delete" queries, add ctid of the result relation	 * into the target list so that the ctid will propagate through	 * execution and ExecutePlan() will be able to identify the right	 * tuple to replace or delete.	This extra field is marked "junk" so	 * that it is not stored back into the tuple.	 */	if (command_type == CMD_UPDATE || command_type == CMD_DELETE)	{		Resdom	   *resdom;		Var		   *var;		resdom = makeResdom(length(tlist) + 1,							TIDOID,							-1,							pstrdup("ctid"),							true);		var = makeVar(result_relation, SelfItemPointerAttributeNumber,					  TIDOID, -1, 0);		/*		 * For an UPDATE, expand_targetlist already created a fresh tlist.		 * For DELETE, better do a listCopy so that we don't destructively		 * modify the original tlist (is this really necessary?).		 */		if (command_type == CMD_DELETE)			tlist = listCopy(tlist);		tlist = lappend(tlist, makeTargetEntry(resdom, (Expr *) var));	}	return tlist;}/***************************************************************************** * *		TARGETLIST EXPANSION * *****************************************************************************//* * expand_targetlist *	  Given a target list as generated by the parser and a result relation, *	  add targetlist entries for any missing attributes, and ensure the *	  non-junk attributes appear in proper field order. * * NOTE: if you are tempted to put more processing here, consider whether * it shouldn't go in the rewriter's rewriteTargetList() instead. */static List *expand_targetlist(List *tlist, int command_type,				  Index result_relation, List *range_table){	List	   *new_tlist = NIL;	Relation	rel;	int			attrno,				numattrs;	/*	 * The rewriter should have already ensured that the TLEs are in	 * correct order; but we have to insert TLEs for any missing	 * attributes.	 *	 * Scan the tuple description in the relation's relcache entry to make	 * sure we have all the user attributes in the right order.	 */	rel = heap_open(getrelid(result_relation, range_table), AccessShareLock);	numattrs = RelationGetNumberOfAttributes(rel);	for (attrno = 1; attrno <= numattrs; attrno++)	{		Form_pg_attribute att_tup = rel->rd_att->attrs[attrno - 1];		TargetEntry *new_tle = NULL;		if (tlist != NIL)		{			TargetEntry *old_tle = (TargetEntry *) lfirst(tlist);			Resdom	   *resdom = old_tle->resdom;			if (!resdom->resjunk && resdom->resno == attrno)			{				new_tle = old_tle;				tlist = lnext(tlist);			}		}		if (new_tle == NULL)		{			/*			 * Didn't find a matching tlist entry, so make one.			 *			 * For INSERT, generate a NULL constant.  (We assume the rewriter			 * would have inserted any available default value.) Also, if			 * the column isn't dropped, apply any domain constraints that			 * might exist --- this is to catch domain NOT NULL.			 *			 * For UPDATE, generate a Var reference to the existing value of			 * the attribute, so that it gets copied to the new tuple. But			 * generate a NULL for dropped columns (we want to drop any			 * old values).			 *			 * When generating a NULL constant for a dropped column, we label			 * it INT4 (any other guaranteed-to-exist datatype would do as			 * well).  We can't label it with the dropped column's			 * datatype since that might not exist anymore.  It does not			 * really matter what we claim the type is, since NULL is NULL			 * --- its representation is datatype-independent.	This could			 * perhaps confuse code comparing the finished plan to the			 * target relation, however.			 */			Oid			atttype = att_tup->atttypid;			int32		atttypmod = att_tup->atttypmod;			Node	   *new_expr;			switch (command_type)			{				case CMD_INSERT:					if (!att_tup->attisdropped)					{						new_expr = (Node *) makeConst(atttype,													  att_tup->attlen,													  (Datum) 0,													  true,		/* isnull */													  att_tup->attbyval);						new_expr = coerce_to_domain(new_expr,													InvalidOid,													atttype,													COERCE_IMPLICIT_CAST);					}					else					{						/* Insert NULL for dropped column */						new_expr = (Node *) makeConst(INT4OID,													  sizeof(int32),													  (Datum) 0,													  true,		/* isnull */													  true /* byval */ );						/* label resdom with INT4, too */						atttype = INT4OID;						atttypmod = -1;					}					break;				case CMD_UPDATE:					if (!att_tup->attisdropped)					{						new_expr = (Node *) makeVar(result_relation,													attrno,													atttype,													atttypmod,													0);					}					else					{						/* Insert NULL for dropped column */						new_expr = (Node *) makeConst(INT4OID,													  sizeof(int32),													  (Datum) 0,													  true,		/* isnull */													  true /* byval */ );						/* label resdom with INT4, too */						atttype = INT4OID;						atttypmod = -1;					}					break;				default:					elog(ERROR, "unrecognized command_type: %d",						 (int) command_type);					new_expr = NULL;	/* keep compiler quiet */					break;			}			new_tle = makeTargetEntry(makeResdom(attrno,												 atttype,												 atttypmod,									  pstrdup(NameStr(att_tup->attname)),												 false),									  (Expr *) new_expr);		}		new_tlist = lappend(new_tlist, new_tle);	}	/*	 * The remaining tlist entries should be resjunk; append them all to	 * the end of the new tlist, making sure they have resnos higher than	 * the last real attribute.  (Note: although the rewriter already did	 * such renumbering, we have to do it again here in case we are doing	 * an UPDATE in a table with dropped columns, or an inheritance child	 * table with extra columns.)	 */	while (tlist)	{		TargetEntry *old_tle = (TargetEntry *) lfirst(tlist);		Resdom	   *resdom = old_tle->resdom;		if (!resdom->resjunk)			elog(ERROR, "targetlist is not sorted correctly");		/* Get the resno right, but don't copy unnecessarily */		if (resdom->resno != attrno)		{			resdom = (Resdom *) copyObject((Node *) resdom);			resdom->resno = attrno;			old_tle = makeTargetEntry(resdom, old_tle->expr);		}		new_tlist = lappend(new_tlist, old_tle);		attrno++;		tlist = lnext(tlist);	}	heap_close(rel, AccessShareLock);	return new_tlist;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?