⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 spi.c

📁 PostgreSQL 8.1.4的源码 适用于Linux下的开源数据库系统
💻 C
📖 第 1 页 / 共 3 页
字号:
/*------------------------------------------------------------------------- * * spi.c *				Server Programming Interface * * Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * * * IDENTIFICATION *	  $PostgreSQL: pgsql/src/backend/executor/spi.c,v 1.144.2.1 2005/11/22 18:23:09 momjian Exp $ * *------------------------------------------------------------------------- */#include "postgres.h"#include "access/printtup.h"#include "catalog/heap.h"#include "commands/trigger.h"#include "executor/spi_priv.h"#include "tcop/tcopprot.h"#include "utils/lsyscache.h"#include "utils/memutils.h"#include "utils/typcache.h"uint32		SPI_processed = 0;Oid			SPI_lastoid = InvalidOid;SPITupleTable *SPI_tuptable = NULL;int			SPI_result;static _SPI_connection *_SPI_stack = NULL;static _SPI_connection *_SPI_current = NULL;static int	_SPI_stack_depth = 0;		/* allocated size of _SPI_stack */static int	_SPI_connected = -1;static int	_SPI_curid = -1;static void _SPI_prepare_plan(const char *src, _SPI_plan *plan);static int _SPI_execute_plan(_SPI_plan *plan,				  Datum *Values, const char *Nulls,				  Snapshot snapshot, Snapshot crosscheck_snapshot,				  bool read_only, long tcount);static int	_SPI_pquery(QueryDesc *queryDesc, long tcount);static void _SPI_error_callback(void *arg);static void _SPI_cursor_operation(Portal portal, bool forward, long count,					  DestReceiver *dest);static _SPI_plan *_SPI_copy_plan(_SPI_plan *plan, int location);static int	_SPI_begin_call(bool execmem);static int	_SPI_end_call(bool procmem);static MemoryContext _SPI_execmem(void);static MemoryContext _SPI_procmem(void);static bool _SPI_checktuples(void);/* =================== interface functions =================== */intSPI_connect(void){	int			newdepth;	/*	 * When procedure called by Executor _SPI_curid expected to be equal to	 * _SPI_connected	 */	if (_SPI_curid != _SPI_connected)		return SPI_ERROR_CONNECT;	if (_SPI_stack == NULL)	{		if (_SPI_connected != -1 || _SPI_stack_depth != 0)			elog(ERROR, "SPI stack corrupted");		newdepth = 16;		_SPI_stack = (_SPI_connection *)			MemoryContextAlloc(TopTransactionContext,							   newdepth * sizeof(_SPI_connection));		_SPI_stack_depth = newdepth;	}	else	{		if (_SPI_stack_depth <= 0 || _SPI_stack_depth <= _SPI_connected)			elog(ERROR, "SPI stack corrupted");		if (_SPI_stack_depth == _SPI_connected + 1)		{			newdepth = _SPI_stack_depth * 2;			_SPI_stack = (_SPI_connection *)				repalloc(_SPI_stack,						 newdepth * sizeof(_SPI_connection));			_SPI_stack_depth = newdepth;		}	}	/*	 * We're entering procedure where _SPI_curid == _SPI_connected - 1	 */	_SPI_connected++;	Assert(_SPI_connected >= 0 && _SPI_connected < _SPI_stack_depth);	_SPI_current = &(_SPI_stack[_SPI_connected]);	_SPI_current->processed = 0;	_SPI_current->lastoid = InvalidOid;	_SPI_current->tuptable = NULL;	_SPI_current->procCxt = NULL;		/* in case we fail to create 'em */	_SPI_current->execCxt = NULL;	_SPI_current->connectSubid = GetCurrentSubTransactionId();	/*	 * Create memory contexts for this procedure	 *	 * XXX it would be better to use PortalContext as the parent context, but	 * we may not be inside a portal (consider deferred-trigger execution).	 * Perhaps CurTransactionContext would do?	For now it doesn't matter	 * because we clean up explicitly in AtEOSubXact_SPI().	 */	_SPI_current->procCxt = AllocSetContextCreate(TopTransactionContext,												  "SPI Proc",												  ALLOCSET_DEFAULT_MINSIZE,												  ALLOCSET_DEFAULT_INITSIZE,												  ALLOCSET_DEFAULT_MAXSIZE);	_SPI_current->execCxt = AllocSetContextCreate(TopTransactionContext,												  "SPI Exec",												  ALLOCSET_DEFAULT_MINSIZE,												  ALLOCSET_DEFAULT_INITSIZE,												  ALLOCSET_DEFAULT_MAXSIZE);	/* ... and switch to procedure's context */	_SPI_current->savedcxt = MemoryContextSwitchTo(_SPI_current->procCxt);	return SPI_OK_CONNECT;}intSPI_finish(void){	int			res;	res = _SPI_begin_call(false);		/* live in procedure memory */	if (res < 0)		return res;	/* Restore memory context as it was before procedure call */	MemoryContextSwitchTo(_SPI_current->savedcxt);	/* Release memory used in procedure call */	MemoryContextDelete(_SPI_current->execCxt);	_SPI_current->execCxt = NULL;	MemoryContextDelete(_SPI_current->procCxt);	_SPI_current->procCxt = NULL;	/*	 * Reset result variables, especially SPI_tuptable which is probably	 * pointing at a just-deleted tuptable	 */	SPI_processed = 0;	SPI_lastoid = InvalidOid;	SPI_tuptable = NULL;	/*	 * After _SPI_begin_call _SPI_connected == _SPI_curid. Now we are closing	 * connection to SPI and returning to upper Executor and so _SPI_connected	 * must be equal to _SPI_curid.	 */	_SPI_connected--;	_SPI_curid--;	if (_SPI_connected == -1)		_SPI_current = NULL;	else		_SPI_current = &(_SPI_stack[_SPI_connected]);	return SPI_OK_FINISH;}/* * Clean up SPI state at transaction commit or abort. */voidAtEOXact_SPI(bool isCommit){	/*	 * Note that memory contexts belonging to SPI stack entries will be freed	 * automatically, so we can ignore them here.  We just need to restore our	 * static variables to initial state.	 */	if (isCommit && _SPI_connected != -1)		ereport(WARNING,				(errcode(ERRCODE_WARNING),				 errmsg("transaction left non-empty SPI stack"),				 errhint("Check for missing \"SPI_finish\" calls.")));	_SPI_current = _SPI_stack = NULL;	_SPI_stack_depth = 0;	_SPI_connected = _SPI_curid = -1;	SPI_processed = 0;	SPI_lastoid = InvalidOid;	SPI_tuptable = NULL;}/* * Clean up SPI state at subtransaction commit or abort. * * During commit, there shouldn't be any unclosed entries remaining from * the current subtransaction; we emit a warning if any are found. */voidAtEOSubXact_SPI(bool isCommit, SubTransactionId mySubid){	bool		found = false;	while (_SPI_connected >= 0)	{		_SPI_connection *connection = &(_SPI_stack[_SPI_connected]);		if (connection->connectSubid != mySubid)			break;				/* couldn't be any underneath it either */		found = true;		/*		 * Release procedure memory explicitly (see note in SPI_connect)		 */		if (connection->execCxt)		{			MemoryContextDelete(connection->execCxt);			connection->execCxt = NULL;		}		if (connection->procCxt)		{			MemoryContextDelete(connection->procCxt);			connection->procCxt = NULL;		}		/*		 * Pop the stack entry and reset global variables.	Unlike		 * SPI_finish(), we don't risk switching to memory contexts that might		 * be already gone.		 */		_SPI_connected--;		_SPI_curid = _SPI_connected;		if (_SPI_connected == -1)			_SPI_current = NULL;		else			_SPI_current = &(_SPI_stack[_SPI_connected]);		SPI_processed = 0;		SPI_lastoid = InvalidOid;		SPI_tuptable = NULL;	}	if (found && isCommit)		ereport(WARNING,				(errcode(ERRCODE_WARNING),				 errmsg("subtransaction left non-empty SPI stack"),				 errhint("Check for missing \"SPI_finish\" calls.")));}/* Pushes SPI stack to allow recursive SPI calls */voidSPI_push(void){	_SPI_curid++;}/* Pops SPI stack to allow recursive SPI calls */voidSPI_pop(void){	_SPI_curid--;}/* Restore state of SPI stack after aborting a subtransaction */voidSPI_restore_connection(void){	Assert(_SPI_connected >= 0);	_SPI_curid = _SPI_connected - 1;}/* Parse, plan, and execute a query string */intSPI_execute(const char *src, bool read_only, long tcount){	_SPI_plan	plan;	int			res;	if (src == NULL || tcount < 0)		return SPI_ERROR_ARGUMENT;	res = _SPI_begin_call(true);	if (res < 0)		return res;	plan.plancxt = NULL;		/* doesn't have own context */	plan.query = src;	plan.nargs = 0;	plan.argtypes = NULL;	_SPI_prepare_plan(src, &plan);	res = _SPI_execute_plan(&plan, NULL, NULL,							InvalidSnapshot, InvalidSnapshot,							read_only, tcount);	_SPI_end_call(true);	return res;}/* Obsolete version of SPI_execute */intSPI_exec(const char *src, long tcount){	return SPI_execute(src, false, tcount);}/* Execute a previously prepared plan */intSPI_execute_plan(void *plan, Datum *Values, const char *Nulls,				 bool read_only, long tcount){	int			res;	if (plan == NULL || tcount < 0)		return SPI_ERROR_ARGUMENT;	if (((_SPI_plan *) plan)->nargs > 0 && Values == NULL)		return SPI_ERROR_PARAM;	res = _SPI_begin_call(true);	if (res < 0)		return res;	res = _SPI_execute_plan((_SPI_plan *) plan,							Values, Nulls,							InvalidSnapshot, InvalidSnapshot,							read_only, tcount);	_SPI_end_call(true);	return res;}/* Obsolete version of SPI_execute_plan */intSPI_execp(void *plan, Datum *Values, const char *Nulls, long tcount){	return SPI_execute_plan(plan, Values, Nulls, false, tcount);}/* * SPI_execute_snapshot -- identical to SPI_execute_plan, except that we allow * the caller to specify exactly which snapshots to use.  This is currently * not documented in spi.sgml because it is only intended for use by RI * triggers. * * Passing snapshot == InvalidSnapshot will select the normal behavior of * fetching a new snapshot for each query. */intSPI_execute_snapshot(void *plan,					 Datum *Values, const char *Nulls,					 Snapshot snapshot, Snapshot crosscheck_snapshot,					 bool read_only, long tcount){	int			res;	if (plan == NULL || tcount < 0)		return SPI_ERROR_ARGUMENT;	if (((_SPI_plan *) plan)->nargs > 0 && Values == NULL)		return SPI_ERROR_PARAM;	res = _SPI_begin_call(true);	if (res < 0)		return res;	res = _SPI_execute_plan((_SPI_plan *) plan,							Values, Nulls,							snapshot, crosscheck_snapshot,							read_only, tcount);	_SPI_end_call(true);	return res;}void *SPI_prepare(const char *src, int nargs, Oid *argtypes){	_SPI_plan	plan;	_SPI_plan  *result;	if (src == NULL || nargs < 0 || (nargs > 0 && argtypes == NULL))	{		SPI_result = SPI_ERROR_ARGUMENT;		return NULL;	}	SPI_result = _SPI_begin_call(true);	if (SPI_result < 0)		return NULL;	plan.plancxt = NULL;		/* doesn't have own context */	plan.query = src;	plan.nargs = nargs;	plan.argtypes = argtypes;	_SPI_prepare_plan(src, &plan);	/* copy plan to procedure context */	result = _SPI_copy_plan(&plan, _SPI_CPLAN_PROCXT);	_SPI_end_call(true);	return (void *) result;}void *SPI_saveplan(void *plan){	_SPI_plan  *newplan;	if (plan == NULL)	{		SPI_result = SPI_ERROR_ARGUMENT;		return NULL;	}	SPI_result = _SPI_begin_call(false);		/* don't change context */	if (SPI_result < 0)		return NULL;	newplan = _SPI_copy_plan((_SPI_plan *) plan, _SPI_CPLAN_TOPCXT);	_SPI_curid--;	SPI_result = 0;	return (void *) newplan;}intSPI_freeplan(void *plan){	_SPI_plan  *spiplan = (_SPI_plan *) plan;	if (plan == NULL)		return SPI_ERROR_ARGUMENT;	MemoryContextDelete(spiplan->plancxt);	return 0;}HeapTupleSPI_copytuple(HeapTuple tuple){	MemoryContext oldcxt = NULL;	HeapTuple	ctuple;	if (tuple == NULL)	{		SPI_result = SPI_ERROR_ARGUMENT;		return NULL;	}	if (_SPI_curid + 1 == _SPI_connected)		/* connected */	{		if (_SPI_current != &(_SPI_stack[_SPI_curid + 1]))			elog(ERROR, "SPI stack corrupted");		oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);	}	ctuple = heap_copytuple(tuple);	if (oldcxt)		MemoryContextSwitchTo(oldcxt);	return ctuple;}HeapTupleHeaderSPI_returntuple(HeapTuple tuple, TupleDesc tupdesc){	MemoryContext oldcxt = NULL;	HeapTupleHeader dtup;	if (tuple == NULL || tupdesc == NULL)	{		SPI_result = SPI_ERROR_ARGUMENT;		return NULL;	}	/* For RECORD results, make sure a typmod has been assigned */	if (tupdesc->tdtypeid == RECORDOID &&		tupdesc->tdtypmod < 0)		assign_record_type_typmod(tupdesc);	if (_SPI_curid + 1 == _SPI_connected)		/* connected */	{		if (_SPI_current != &(_SPI_stack[_SPI_curid + 1]))			elog(ERROR, "SPI stack corrupted");		oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);	}	dtup = (HeapTupleHeader) palloc(tuple->t_len);	memcpy((char *) dtup, (char *) tuple->t_data, tuple->t_len);	HeapTupleHeaderSetDatumLength(dtup, tuple->t_len);	HeapTupleHeaderSetTypeId(dtup, tupdesc->tdtypeid);	HeapTupleHeaderSetTypMod(dtup, tupdesc->tdtypmod);	if (oldcxt)		MemoryContextSwitchTo(oldcxt);	return dtup;}HeapTupleSPI_modifytuple(Relation rel, HeapTuple tuple, int natts, int *attnum,				Datum *Values, const char *Nulls){	MemoryContext oldcxt = NULL;	HeapTuple	mtuple;	int			numberOfAttributes;	Datum	   *v;	char	   *n;	int			i;	if (rel == NULL || tuple == NULL || natts < 0 || attnum == NULL || Values == NULL)	{		SPI_result = SPI_ERROR_ARGUMENT;		return NULL;	}	if (_SPI_curid + 1 == _SPI_connected)		/* connected */	{		if (_SPI_current != &(_SPI_stack[_SPI_curid + 1]))			elog(ERROR, "SPI stack corrupted");		oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);	}	SPI_result = 0;	numberOfAttributes = rel->rd_att->natts;	v = (Datum *) palloc(numberOfAttributes * sizeof(Datum));	n = (char *) palloc(numberOfAttributes * sizeof(char));	/* fetch old values and nulls */	heap_deformtuple(tuple, rel->rd_att, v, n);	/* replace values and nulls */	for (i = 0; i < natts; i++)	{		if (attnum[i] <= 0 || attnum[i] > numberOfAttributes)			break;		v[attnum[i] - 1] = Values[i];		n[attnum[i] - 1] = (Nulls && Nulls[i] == 'n') ? 'n' : ' ';	}	if (i == natts)				/* no errors in *attnum */	{		mtuple = heap_formtuple(rel->rd_att, v, n);		/*		 * copy the identification info of the old tuple: t_ctid, t_self, and		 * OID (if any)		 */		mtuple->t_data->t_ctid = tuple->t_data->t_ctid;		mtuple->t_self = tuple->t_self;		mtuple->t_tableOid = tuple->t_tableOid;		if (rel->rd_att->tdhasoid)			HeapTupleSetOid(mtuple, HeapTupleGetOid(tuple));	}	else	{		mtuple = NULL;		SPI_result = SPI_ERROR_NOATTRIBUTE;	}	pfree(v);	pfree(n);	if (oldcxt)		MemoryContextSwitchTo(oldcxt);	return mtuple;}intSPI_fnumber(TupleDesc tupdesc, const char *fname){	int			res;	Form_pg_attribute sysatt;

⌨️ 快捷键说明

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