Commit a393fbf9 authored by Tom Lane's avatar Tom Lane

Restructure error handling as recently discussed. It is now really

possible to trap an error inside a function rather than letting it
propagate out to PostgresMain.  You still have to use AbortCurrentTransaction
to clean up, but at least the error handling itself will cooperate.
parent 94f8f63f
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/bootstrap/bootstrap.c,v 1.189 2004/07/21 20:34:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/bootstrap/bootstrap.c,v 1.190 2004/07/31 00:45:30 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -16,7 +16,6 @@ ...@@ -16,7 +16,6 @@
#include <unistd.h> #include <unistd.h>
#include <signal.h> #include <signal.h>
#include <setjmp.h>
#ifdef HAVE_GETOPT_H #ifdef HAVE_GETOPT_H
#include <getopt.h> #include <getopt.h>
#endif #endif
...@@ -458,15 +457,6 @@ BootstrapMain(int argc, char *argv[]) ...@@ -458,15 +457,6 @@ BootstrapMain(int argc, char *argv[])
for (i = 0; i < HASHTABLESIZE; ++i) for (i = 0; i < HASHTABLESIZE; ++i)
hashtable[i] = NULL; hashtable[i] = NULL;
/*
* abort processing resumes here (this is probably dead code?)
*/
if (sigsetjmp(Warn_restart, 1) != 0)
{
Warnings++;
AbortCurrentTransaction();
}
/* /*
* Process bootstrap input. * Process bootstrap input.
* *
......
...@@ -8,13 +8,12 @@ ...@@ -8,13 +8,12 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/copy.c,v 1.227 2004/06/16 01:26:42 tgl Exp $ * $PostgreSQL: pgsql/src/backend/commands/copy.c,v 1.228 2004/07/31 00:45:31 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "postgres.h" #include "postgres.h"
#include <errno.h>
#include <unistd.h> #include <unistd.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <netinet/in.h> #include <netinet/in.h>
...@@ -130,6 +129,9 @@ static StringInfoData line_buf; ...@@ -130,6 +129,9 @@ static StringInfoData line_buf;
static bool line_buf_converted; static bool line_buf_converted;
/* non-export function prototypes */ /* non-export function prototypes */
static void DoCopyTo(Relation rel, List *attnumlist, bool binary, bool oids,
char *delim, char *null_print, bool csv_mode, char *quote,
char *escape, List *force_quote_atts, bool fe_copy);
static void CopyTo(Relation rel, List *attnumlist, bool binary, bool oids, static void CopyTo(Relation rel, List *attnumlist, bool binary, bool oids,
char *delim, char *null_print, bool csv_mode, char *quote, char *escape, char *delim, char *null_print, bool csv_mode, char *quote, char *escape,
List *force_quote_atts); List *force_quote_atts);
...@@ -688,6 +690,7 @@ DoCopy(const CopyStmt *stmt) ...@@ -688,6 +690,7 @@ DoCopy(const CopyStmt *stmt)
ListCell *option; ListCell *option;
List *attnamelist = stmt->attlist; List *attnamelist = stmt->attlist;
List *attnumlist; List *attnumlist;
bool fe_copy = false;
bool binary = false; bool binary = false;
bool oids = false; bool oids = false;
bool csv_mode = false; bool csv_mode = false;
...@@ -1062,7 +1065,7 @@ DoCopy(const CopyStmt *stmt) ...@@ -1062,7 +1065,7 @@ DoCopy(const CopyStmt *stmt)
if (pipe) if (pipe)
{ {
if (whereToSendOutput == Remote) if (whereToSendOutput == Remote)
SendCopyBegin(binary, list_length(attnumlist)); fe_copy = true;
else else
copy_file = stdout; copy_file = stdout;
} }
...@@ -1099,8 +1102,9 @@ DoCopy(const CopyStmt *stmt) ...@@ -1099,8 +1102,9 @@ DoCopy(const CopyStmt *stmt)
errmsg("\"%s\" is a directory", filename))); errmsg("\"%s\" is a directory", filename)));
} }
} }
CopyTo(rel, attnumlist, binary, oids, delim, null_print, csv_mode,
quote, escape, force_quote_atts); DoCopyTo(rel, attnumlist, binary, oids, delim, null_print, csv_mode,
quote, escape, force_quote_atts, fe_copy);
} }
if (!pipe) if (!pipe)
...@@ -1112,8 +1116,6 @@ DoCopy(const CopyStmt *stmt) ...@@ -1112,8 +1116,6 @@ DoCopy(const CopyStmt *stmt)
errmsg("could not write to file \"%s\": %m", errmsg("could not write to file \"%s\": %m",
filename))); filename)));
} }
else if (whereToSendOutput == Remote && !is_from)
SendCopyEnd(binary);
pfree(attribute_buf.data); pfree(attribute_buf.data);
pfree(line_buf.data); pfree(line_buf.data);
...@@ -1127,6 +1129,39 @@ DoCopy(const CopyStmt *stmt) ...@@ -1127,6 +1129,39 @@ DoCopy(const CopyStmt *stmt)
} }
/*
* This intermediate routine just exists to localize the effects of setjmp
* so we don't need to plaster a lot of variables with "volatile".
*/
static void
DoCopyTo(Relation rel, List *attnumlist, bool binary, bool oids,
char *delim, char *null_print, bool csv_mode, char *quote,
char *escape, List *force_quote_atts, bool fe_copy)
{
PG_TRY();
{
if (fe_copy)
SendCopyBegin(binary, list_length(attnumlist));
CopyTo(rel, attnumlist, binary, oids, delim, null_print, csv_mode,
quote, escape, force_quote_atts);
if (fe_copy)
SendCopyEnd(binary);
}
PG_CATCH();
{
/*
* Make sure we turn off old-style COPY OUT mode upon error.
* It is okay to do this in all cases, since it does nothing
* if the mode is not on.
*/
pq_endcopyout(true);
PG_RE_THROW();
}
PG_END_TRY();
}
/* /*
* Copy from relation TO file. * Copy from relation TO file.
*/ */
......
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/portalcmds.c,v 1.29 2004/07/17 03:28:47 tgl Exp $ * $PostgreSQL: pgsql/src/backend/commands/portalcmds.c,v 1.30 2004/07/31 00:45:31 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -259,8 +259,18 @@ PortalCleanup(Portal portal) ...@@ -259,8 +259,18 @@ PortalCleanup(Portal portal)
/* We must make the portal's resource owner current */ /* We must make the portal's resource owner current */
saveResourceOwner = CurrentResourceOwner; saveResourceOwner = CurrentResourceOwner;
PG_TRY();
{
CurrentResourceOwner = portal->resowner; CurrentResourceOwner = portal->resowner;
ExecutorEnd(queryDesc); ExecutorEnd(queryDesc);
}
PG_CATCH();
{
/* Ensure CurrentResourceOwner is restored on error */
CurrentResourceOwner = saveResourceOwner;
PG_RE_THROW();
}
PG_END_TRY();
CurrentResourceOwner = saveResourceOwner; CurrentResourceOwner = saveResourceOwner;
} }
} }
...@@ -317,15 +327,17 @@ PersistHoldablePortal(Portal portal) ...@@ -317,15 +327,17 @@ PersistHoldablePortal(Portal portal)
portal->status = PORTAL_ACTIVE; portal->status = PORTAL_ACTIVE;
/* /*
* Set global portal context pointers. * Set up global portal context pointers.
*/ */
saveActivePortal = ActivePortal; saveActivePortal = ActivePortal;
ActivePortal = portal;
saveResourceOwner = CurrentResourceOwner; saveResourceOwner = CurrentResourceOwner;
CurrentResourceOwner = portal->resowner;
savePortalContext = PortalContext; savePortalContext = PortalContext;
PortalContext = PortalGetHeapMemory(portal);
saveQueryContext = QueryContext; saveQueryContext = QueryContext;
PG_TRY();
{
ActivePortal = portal;
CurrentResourceOwner = portal->resowner;
PortalContext = PortalGetHeapMemory(portal);
QueryContext = portal->queryContext; QueryContext = portal->queryContext;
MemoryContextSwitchTo(PortalContext); MemoryContextSwitchTo(PortalContext);
...@@ -386,16 +398,23 @@ PersistHoldablePortal(Portal portal) ...@@ -386,16 +398,23 @@ PersistHoldablePortal(Portal portal)
pfree(tup); pfree(tup);
} }
} }
}
PG_CATCH();
{
/* Uncaught error while executing portal: mark it dead */
portal->status = PORTAL_FAILED;
MemoryContextSwitchTo(oldcxt); /* Restore global vars and propagate error */
ActivePortal = saveActivePortal;
CurrentResourceOwner = saveResourceOwner;
PortalContext = savePortalContext;
QueryContext = saveQueryContext;
/* PG_RE_THROW();
* We can now release any subsidiary memory of the portal's heap }
* context; we'll never use it again. The executor already dropped PG_END_TRY();
* its context, but this will clean up anything that glommed onto the
* portal's heap via PortalContext. MemoryContextSwitchTo(oldcxt);
*/
MemoryContextDeleteChildren(PortalGetHeapMemory(portal));
/* Mark portal not active */ /* Mark portal not active */
portal->status = PORTAL_READY; portal->status = PORTAL_READY;
...@@ -404,4 +423,12 @@ PersistHoldablePortal(Portal portal) ...@@ -404,4 +423,12 @@ PersistHoldablePortal(Portal portal)
CurrentResourceOwner = saveResourceOwner; CurrentResourceOwner = saveResourceOwner;
PortalContext = savePortalContext; PortalContext = savePortalContext;
QueryContext = saveQueryContext; QueryContext = saveQueryContext;
/*
* We can now release any subsidiary memory of the portal's heap
* context; we'll never use it again. The executor already dropped
* its context, but this will clean up anything that glommed onto the
* portal's heap via PortalContext.
*/
MemoryContextDeleteChildren(PortalGetHeapMemory(portal));
} }
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/vacuum.c,v 1.284 2004/07/21 22:31:21 tgl Exp $ * $PostgreSQL: pgsql/src/backend/commands/vacuum.c,v 1.285 2004/07/31 00:45:31 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -228,14 +228,13 @@ void ...@@ -228,14 +228,13 @@ void
vacuum(VacuumStmt *vacstmt) vacuum(VacuumStmt *vacstmt)
{ {
const char *stmttype = vacstmt->vacuum ? "VACUUM" : "ANALYZE"; const char *stmttype = vacstmt->vacuum ? "VACUUM" : "ANALYZE";
MemoryContext anl_context = NULL;
TransactionId initialOldestXmin = InvalidTransactionId; TransactionId initialOldestXmin = InvalidTransactionId;
TransactionId initialFreezeLimit = InvalidTransactionId; TransactionId initialFreezeLimit = InvalidTransactionId;
bool all_rels, volatile MemoryContext anl_context = NULL;
volatile bool all_rels,
in_outer_xact, in_outer_xact,
use_own_xacts; use_own_xacts;
List *relations; List *relations;
ListCell *cur;
if (vacstmt->verbose) if (vacstmt->verbose)
elevel = INFO; elevel = INFO;
...@@ -267,10 +266,6 @@ vacuum(VacuumStmt *vacstmt) ...@@ -267,10 +266,6 @@ vacuum(VacuumStmt *vacstmt)
in_outer_xact = IsInTransactionChain((void *) vacstmt); in_outer_xact = IsInTransactionChain((void *) vacstmt);
} }
/* Turn vacuum cost accounting on or off */
VacuumCostActive = (VacuumCostNaptime > 0);
VacuumCostBalance = 0;
/* /*
* Send info about dead objects to the statistics collector * Send info about dead objects to the statistics collector
*/ */
...@@ -377,6 +372,14 @@ vacuum(VacuumStmt *vacstmt) ...@@ -377,6 +372,14 @@ vacuum(VacuumStmt *vacstmt)
CommitTransactionCommand(); CommitTransactionCommand();
} }
/* Turn vacuum cost accounting on or off */
PG_TRY();
{
ListCell *cur;
VacuumCostActive = (VacuumCostNaptime > 0);
VacuumCostBalance = 0;
/* /*
* Loop to process each selected relation. * Loop to process each selected relation.
*/ */
...@@ -428,6 +431,17 @@ vacuum(VacuumStmt *vacstmt) ...@@ -428,6 +431,17 @@ vacuum(VacuumStmt *vacstmt)
} }
} }
} }
}
PG_CATCH();
{
/* Make sure cost accounting is turned off after error */
VacuumCostActive = false;
PG_RE_THROW();
}
PG_END_TRY();
/* Turn off vacuum cost accounting */
VacuumCostActive = false;
/* /*
* Finish up processing. * Finish up processing.
...@@ -475,9 +489,6 @@ vacuum(VacuumStmt *vacstmt) ...@@ -475,9 +489,6 @@ vacuum(VacuumStmt *vacstmt)
if (anl_context) if (anl_context)
MemoryContextDelete(anl_context); MemoryContextDelete(anl_context);
/* Turn off vacuum cost accounting */
VacuumCostActive = false;
} }
/* /*
......
...@@ -37,7 +37,7 @@ ...@@ -37,7 +37,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/postmaster/bgwriter.c,v 1.3 2004/06/03 02:08:03 tgl Exp $ * $PostgreSQL: pgsql/src/backend/postmaster/bgwriter.c,v 1.4 2004/07/31 00:45:33 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -57,6 +57,7 @@ ...@@ -57,6 +57,7 @@
#include "storage/smgr.h" #include "storage/smgr.h"
#include "tcop/tcopprot.h" #include "tcop/tcopprot.h"
#include "utils/guc.h" #include "utils/guc.h"
#include "utils/memutils.h"
/*---------- /*----------
...@@ -153,6 +154,8 @@ static void ReqShutdownHandler(SIGNAL_ARGS); ...@@ -153,6 +154,8 @@ static void ReqShutdownHandler(SIGNAL_ARGS);
void void
BackgroundWriterMain(void) BackgroundWriterMain(void)
{ {
sigjmp_buf local_sigjmp_buf;
Assert(BgWriterShmem != NULL); Assert(BgWriterShmem != NULL);
BgWriterShmem->bgwriter_pid = MyProcPid; BgWriterShmem->bgwriter_pid = MyProcPid;
am_bg_writer = true; am_bg_writer = true;
...@@ -201,19 +204,19 @@ BackgroundWriterMain(void) ...@@ -201,19 +204,19 @@ BackgroundWriterMain(void)
/* /*
* If an exception is encountered, processing resumes here. * If an exception is encountered, processing resumes here.
*
* See notes in postgres.c about the design of this coding.
*/ */
if (sigsetjmp(Warn_restart, 1) != 0) if (sigsetjmp(local_sigjmp_buf, 1) != 0)
{ {
/* /* Since not using PG_TRY, must reset error stack by hand */
* Make sure we're not interrupted while cleaning up. Also forget error_context_stack = NULL;
* any pending QueryCancel request, since we're aborting anyway.
* Force InterruptHoldoffCount to a known state in case we /* Prevent interrupts while cleaning up */
* ereport'd from inside a holdoff section. HOLD_INTERRUPTS();
*/
ImmediateInterruptOK = false; /* Report the error to the server log */
QueryCancelPending = false; EmitErrorReport();
InterruptHoldoffCount = 1;
CritSectionCount = 0; /* should be unnecessary, but... */
/* /*
* These operations are really just a minimal subset of * These operations are really just a minimal subset of
...@@ -224,12 +227,6 @@ BackgroundWriterMain(void) ...@@ -224,12 +227,6 @@ BackgroundWriterMain(void)
AbortBufferIO(); AbortBufferIO();
UnlockBuffers(); UnlockBuffers();
/*
* Clear flag to indicate that we got out of error recovery mode
* successfully. (Flag was set in elog.c before longjmp().)
*/
InError = false;
/* Warn any waiting backends that the checkpoint failed. */ /* Warn any waiting backends that the checkpoint failed. */
if (ckpt_active) if (ckpt_active)
{ {
...@@ -242,8 +239,13 @@ BackgroundWriterMain(void) ...@@ -242,8 +239,13 @@ BackgroundWriterMain(void)
} }
/* /*
* Exit interrupt holdoff section we implicitly established above. * Now return to normal top-level context and clear ErrorContext
* for next time.
*/ */
MemoryContextSwitchTo(TopMemoryContext);
FlushErrorState();
/* Now we can allow interrupts again */
RESUME_INTERRUPTS(); RESUME_INTERRUPTS();
/* /*
...@@ -255,7 +257,8 @@ BackgroundWriterMain(void) ...@@ -255,7 +257,8 @@ BackgroundWriterMain(void)
pg_usleep(1000000L); pg_usleep(1000000L);
} }
Warn_restart_ready = true; /* we can now handle ereport(ERROR) */ /* We can now handle ereport(ERROR) */
PG_exception_stack = &local_sigjmp_buf;
/* /*
* Unblock signals (they were blocked when the postmaster forked us) * Unblock signals (they were blocked when the postmaster forked us)
......
...@@ -37,7 +37,7 @@ ...@@ -37,7 +37,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/postmaster/postmaster.c,v 1.416 2004/07/27 01:46:03 tgl Exp $ * $PostgreSQL: pgsql/src/backend/postmaster/postmaster.c,v 1.417 2004/07/31 00:45:33 tgl Exp $
* *
* NOTES * NOTES
* *
...@@ -73,7 +73,6 @@ ...@@ -73,7 +73,6 @@
#include <ctype.h> #include <ctype.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/socket.h> #include <sys/socket.h>
#include <errno.h>
#include <fcntl.h> #include <fcntl.h>
#include <sys/param.h> #include <sys/param.h>
#include <netinet/in.h> #include <netinet/in.h>
...@@ -3226,6 +3225,11 @@ StartChildProcess(int xlop) ...@@ -3226,6 +3225,11 @@ StartChildProcess(int xlop)
/* Lose the postmaster's on-exit routines and port connections */ /* Lose the postmaster's on-exit routines and port connections */
on_exit_reset(); on_exit_reset();
/* Release postmaster's working memory context */
MemoryContextSwitchTo(TopMemoryContext);
MemoryContextDelete(PostmasterContext);
PostmasterContext = NULL;
BootstrapMain(ac, av); BootstrapMain(ac, av);
ExitPostmaster(0); ExitPostmaster(0);
} }
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/storage/large_object/inv_api.c,v 1.103 2004/07/28 14:23:29 tgl Exp $ * $PostgreSQL: pgsql/src/backend/storage/large_object/inv_api.c,v 1.104 2004/07/31 00:45:34 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -54,6 +54,8 @@ open_lo_relation(void) ...@@ -54,6 +54,8 @@ open_lo_relation(void)
/* Arrange for the top xact to own these relation references */ /* Arrange for the top xact to own these relation references */
currentOwner = CurrentResourceOwner; currentOwner = CurrentResourceOwner;
PG_TRY();
{
CurrentResourceOwner = TopTransactionResourceOwner; CurrentResourceOwner = TopTransactionResourceOwner;
/* Use RowExclusiveLock since we might either read or write */ /* Use RowExclusiveLock since we might either read or write */
...@@ -61,7 +63,14 @@ open_lo_relation(void) ...@@ -61,7 +63,14 @@ open_lo_relation(void)
lo_heap_r = heap_openr(LargeObjectRelationName, RowExclusiveLock); lo_heap_r = heap_openr(LargeObjectRelationName, RowExclusiveLock);
if (lo_index_r == NULL) if (lo_index_r == NULL)
lo_index_r = index_openr(LargeObjectLOidPNIndex); lo_index_r = index_openr(LargeObjectLOidPNIndex);
}
PG_CATCH();
{
/* Ensure CurrentResourceOwner is restored on error */
CurrentResourceOwner = currentOwner;
PG_RE_THROW();
}
PG_END_TRY();
CurrentResourceOwner = currentOwner; CurrentResourceOwner = currentOwner;
} }
...@@ -82,13 +91,22 @@ close_lo_relation(bool isCommit) ...@@ -82,13 +91,22 @@ close_lo_relation(bool isCommit)
ResourceOwner currentOwner; ResourceOwner currentOwner;
currentOwner = CurrentResourceOwner; currentOwner = CurrentResourceOwner;
PG_TRY();
{
CurrentResourceOwner = TopTransactionResourceOwner; CurrentResourceOwner = TopTransactionResourceOwner;
if (lo_index_r) if (lo_index_r)
index_close(lo_index_r); index_close(lo_index_r);
if (lo_heap_r) if (lo_heap_r)
heap_close(lo_heap_r, NoLock); heap_close(lo_heap_r, NoLock);
}
PG_CATCH();
{
/* Ensure CurrentResourceOwner is restored on error */
CurrentResourceOwner = currentOwner;
PG_RE_THROW();
}
PG_END_TRY();
CurrentResourceOwner = currentOwner; CurrentResourceOwner = currentOwner;
} }
lo_heap_r = NULL; lo_heap_r = NULL;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/tcop/postgres.c,v 1.426 2004/07/28 22:05:46 tgl Exp $ * $PostgreSQL: pgsql/src/backend/tcop/postgres.c,v 1.427 2004/07/31 00:45:36 tgl Exp $
* *
* NOTES * NOTES
* this is the "main" module of the postgres backend and * this is the "main" module of the postgres backend and
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
#include <signal.h> #include <signal.h>
#include <fcntl.h> #include <fcntl.h>
#include <sys/socket.h> #include <sys/socket.h>
#include <errno.h>
#if HAVE_SYS_SELECT_H #if HAVE_SYS_SELECT_H
#include <sys/select.h> #include <sys/select.h>
#endif #endif
...@@ -77,12 +76,6 @@ const char *debug_query_string; /* for pgmonitor and ...@@ -77,12 +76,6 @@ const char *debug_query_string; /* for pgmonitor and
/* Note: whereToSendOutput is initialized for the bootstrap/standalone case */ /* Note: whereToSendOutput is initialized for the bootstrap/standalone case */
CommandDest whereToSendOutput = Debug; CommandDest whereToSendOutput = Debug;
/* note: these declarations had better match tcopprot.h */
sigjmp_buf Warn_restart;
bool Warn_restart_ready = false;
bool InError = false;
/* flag for logging end of session */ /* flag for logging end of session */
bool Log_disconnections = false; bool Log_disconnections = false;
...@@ -1876,7 +1869,7 @@ quickdie(SIGNAL_ARGS) ...@@ -1876,7 +1869,7 @@ quickdie(SIGNAL_ARGS)
/* /*
* Ideally this should be ereport(FATAL), but then we'd not get * Ideally this should be ereport(FATAL), but then we'd not get
* control back (perhaps could fix by doing local sigsetjmp?) * control back...
*/ */
ereport(WARNING, ereport(WARNING,
(errcode(ERRCODE_CRASH_SHUTDOWN), (errcode(ERRCODE_CRASH_SHUTDOWN),
...@@ -1962,10 +1955,9 @@ StatementCancelHandler(SIGNAL_ARGS) ...@@ -1962,10 +1955,9 @@ StatementCancelHandler(SIGNAL_ARGS)
int save_errno = errno; int save_errno = errno;
/* /*
* Don't joggle the elbow of proc_exit, nor an already-in-progress * Don't joggle the elbow of proc_exit
* abort
*/ */
if (!proc_exit_inprogress && !InError) if (!proc_exit_inprogress)
{ {
InterruptPending = true; InterruptPending = true;
QueryCancelPending = true; QueryCancelPending = true;
...@@ -2148,7 +2140,6 @@ usage(const char *progname) ...@@ -2148,7 +2140,6 @@ usage(const char *progname)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* PostgresMain * PostgresMain
* postgres main loop -- all backends, interactive or otherwise start here * postgres main loop -- all backends, interactive or otherwise start here
...@@ -2175,6 +2166,7 @@ PostgresMain(int argc, char *argv[], const char *username) ...@@ -2175,6 +2166,7 @@ PostgresMain(int argc, char *argv[], const char *username)
int firstchar; int firstchar;
char stack_base; char stack_base;
StringInfoData input_message; StringInfoData input_message;
sigjmp_buf local_sigjmp_buf;
volatile bool send_rfq = true; volatile bool send_rfq = true;
/* /*
...@@ -2772,50 +2764,61 @@ PostgresMain(int argc, char *argv[], const char *username) ...@@ -2772,50 +2764,61 @@ PostgresMain(int argc, char *argv[], const char *username)
* *
* If an exception is encountered, processing resumes here so we abort * If an exception is encountered, processing resumes here so we abort
* the current transaction and start a new one. * the current transaction and start a new one.
*
* You might wonder why this isn't coded as an infinite loop around
* a PG_TRY construct. The reason is that this is the bottom of the
* exception stack, and so with PG_TRY there would be no exception
* handler in force at all during the CATCH part. By leaving the
* outermost setjmp always active, we have at least some chance of
* recovering from an error during error recovery. (If we get into
* an infinite loop thereby, it will soon be stopped by overflow of
* elog.c's internal state stack.)
*/ */
if (sigsetjmp(Warn_restart, 1) != 0) if (sigsetjmp(local_sigjmp_buf, 1) != 0)
{ {
/* /*
* NOTE: if you are tempted to add more code in this if-block, * NOTE: if you are tempted to add more code in this if-block,
* consider the probability that it should be in * consider the high probability that it should be in
* AbortTransaction() instead. * AbortTransaction() instead. The only stuff done directly here
* * should be stuff that is guaranteed to apply *only* for outer-level
* Make sure we're not interrupted while cleaning up. Also forget * error recovery, such as adjusting the FE/BE protocol status.
* any pending QueryCancel request, since we're aborting anyway. */
* Force InterruptHoldoffCount to a known state in case we
* ereport'd from inside a holdoff section. /* Since not using PG_TRY, must reset error stack by hand */
error_context_stack = NULL;
/* Prevent interrupts while cleaning up */
HOLD_INTERRUPTS();
/*
* Forget any pending QueryCancel request, since we're returning
* to the idle loop anyway, and cancel the statement timer if running.
*/ */
ImmediateInterruptOK = false;
QueryCancelPending = false; QueryCancelPending = false;
InterruptHoldoffCount = 1;
CritSectionCount = 0; /* should be unnecessary, but... */
disable_sig_alarm(true); disable_sig_alarm(true);
QueryCancelPending = false; /* again in case timeout occurred */ QueryCancelPending = false; /* again in case timeout occurred */
/*
* Turn off these interrupts too. This is only needed here and not
* in other exception-catching places since these interrupts are
* only enabled while we wait for client input.
*/
DisableNotifyInterrupt(); DisableNotifyInterrupt();
DisableCatchupInterrupt(); DisableCatchupInterrupt();
debug_query_string = NULL;
/* Report the error to the client and/or server log */
EmitErrorReport();
/* /*
* If there's an active portal, mark it as failed * Make sure debug_query_string gets reset before we possibly clobber
* the storage it points at.
*/ */
if (ActivePortal) debug_query_string = NULL;
ActivePortal->status = PORTAL_FAILED;
/* /*
* Make sure we are in a valid memory context during recovery. * Abort the current transaction in order to recover.
*
* We use ErrorContext in hopes that it will have some free space
* even if we're otherwise up against it...
*/ */
MemoryContextSwitchTo(ErrorContext);
/* Make sure we are using a sane ResourceOwner, too */
CurrentResourceOwner = CurTransactionResourceOwner;
/* Do the recovery */
ereport(DEBUG2,
(errmsg_internal("AbortCurrentTransaction")));
AbortCurrentTransaction(); AbortCurrentTransaction();
/* /*
...@@ -2823,23 +2826,9 @@ PostgresMain(int argc, char *argv[], const char *username) ...@@ -2823,23 +2826,9 @@ PostgresMain(int argc, char *argv[], const char *username)
* for next time. * for next time.
*/ */
MemoryContextSwitchTo(TopMemoryContext); MemoryContextSwitchTo(TopMemoryContext);
MemoryContextResetAndDeleteChildren(ErrorContext); FlushErrorState();
ActivePortal = NULL;
PortalContext = NULL;
QueryContext = NULL; QueryContext = NULL;
/*
* Clear flag to indicate that we got out of error recovery mode
* successfully. (Flag was set in elog.c before longjmp().)
*/
InError = false;
xact_started = false;
/*
* Clear flag that causes accounting for cost based vacuum.
*/
VacuumCostActive = false;
/* /*
* If we were handling an extended-query-protocol message, * If we were handling an extended-query-protocol message,
* initiate skip till next Sync. This also causes us not to issue * initiate skip till next Sync. This also causes us not to issue
...@@ -2848,13 +2837,15 @@ PostgresMain(int argc, char *argv[], const char *username) ...@@ -2848,13 +2837,15 @@ PostgresMain(int argc, char *argv[], const char *username)
if (doing_extended_query_message) if (doing_extended_query_message)
ignore_till_sync = true; ignore_till_sync = true;
/* /* We don't have a transaction command open anymore */
* Exit interrupt holdoff section we implicitly established above. xact_started = false;
*/
/* Now we can allow interrupts again */
RESUME_INTERRUPTS(); RESUME_INTERRUPTS();
} }
Warn_restart_ready = true; /* we can now handle ereport(ERROR) */ /* We can now handle ereport(ERROR) */
PG_exception_stack = &local_sigjmp_buf;
PG_SETMASK(&UnBlockSig); PG_SETMASK(&UnBlockSig);
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/tcop/pquery.c,v 1.81 2004/07/17 03:29:00 tgl Exp $ * $PostgreSQL: pgsql/src/backend/tcop/pquery.c,v 1.82 2004/07/31 00:45:36 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -246,13 +246,15 @@ PortalStart(Portal portal, ParamListInfo params) ...@@ -246,13 +246,15 @@ PortalStart(Portal portal, ParamListInfo params)
AssertState(portal->status == PORTAL_NEW); /* else extra PortalStart */ AssertState(portal->status == PORTAL_NEW); /* else extra PortalStart */
/* /*
* Set global portal context pointers. (Should we set QueryContext?) * Set up global portal context pointers. (Should we set QueryContext?)
*/ */
saveActivePortal = ActivePortal; saveActivePortal = ActivePortal;
ActivePortal = portal;
saveResourceOwner = CurrentResourceOwner; saveResourceOwner = CurrentResourceOwner;
CurrentResourceOwner = portal->resowner;
savePortalContext = PortalContext; savePortalContext = PortalContext;
PG_TRY();
{
ActivePortal = portal;
CurrentResourceOwner = portal->resowner;
PortalContext = PortalGetHeapMemory(portal); PortalContext = PortalGetHeapMemory(portal);
oldContext = MemoryContextSwitchTo(PortalGetHeapMemory(portal)); oldContext = MemoryContextSwitchTo(PortalGetHeapMemory(portal));
...@@ -334,6 +336,20 @@ PortalStart(Portal portal, ParamListInfo params) ...@@ -334,6 +336,20 @@ PortalStart(Portal portal, ParamListInfo params)
portal->tupDesc = NULL; portal->tupDesc = NULL;
break; break;
} }
}
PG_CATCH();
{
/* Uncaught error while executing portal: mark it dead */
portal->status = PORTAL_FAILED;
/* Restore global vars and propagate error */
ActivePortal = saveActivePortal;
CurrentResourceOwner = saveResourceOwner;
PortalContext = savePortalContext;
PG_RE_THROW();
}
PG_END_TRY();
MemoryContextSwitchTo(oldContext); MemoryContextSwitchTo(oldContext);
...@@ -449,15 +465,17 @@ PortalRun(Portal portal, long count, ...@@ -449,15 +465,17 @@ PortalRun(Portal portal, long count,
portal->status = PORTAL_ACTIVE; portal->status = PORTAL_ACTIVE;
/* /*
* Set global portal context pointers. * Set up global portal context pointers.
*/ */
saveActivePortal = ActivePortal; saveActivePortal = ActivePortal;
ActivePortal = portal;
saveResourceOwner = CurrentResourceOwner; saveResourceOwner = CurrentResourceOwner;
CurrentResourceOwner = portal->resowner;
savePortalContext = PortalContext; savePortalContext = PortalContext;
PortalContext = PortalGetHeapMemory(portal);
saveQueryContext = QueryContext; saveQueryContext = QueryContext;
PG_TRY();
{
ActivePortal = portal;
CurrentResourceOwner = portal->resowner;
PortalContext = PortalGetHeapMemory(portal);
QueryContext = portal->queryContext; QueryContext = portal->queryContext;
oldContext = MemoryContextSwitchTo(PortalContext); oldContext = MemoryContextSwitchTo(PortalContext);
...@@ -534,6 +552,21 @@ PortalRun(Portal portal, long count, ...@@ -534,6 +552,21 @@ PortalRun(Portal portal, long count,
result = false; /* keep compiler quiet */ result = false; /* keep compiler quiet */
break; break;
} }
}
PG_CATCH();
{
/* Uncaught error while executing portal: mark it dead */
portal->status = PORTAL_FAILED;
/* Restore global vars and propagate error */
ActivePortal = saveActivePortal;
CurrentResourceOwner = saveResourceOwner;
PortalContext = savePortalContext;
QueryContext = saveQueryContext;
PG_RE_THROW();
}
PG_END_TRY();
MemoryContextSwitchTo(oldContext); MemoryContextSwitchTo(oldContext);
...@@ -970,15 +1003,17 @@ PortalRunFetch(Portal portal, ...@@ -970,15 +1003,17 @@ PortalRunFetch(Portal portal,
portal->status = PORTAL_ACTIVE; portal->status = PORTAL_ACTIVE;
/* /*
* Set global portal context pointers. * Set up global portal context pointers.
*/ */
saveActivePortal = ActivePortal; saveActivePortal = ActivePortal;
ActivePortal = portal;
saveResourceOwner = CurrentResourceOwner; saveResourceOwner = CurrentResourceOwner;
CurrentResourceOwner = portal->resowner;
savePortalContext = PortalContext; savePortalContext = PortalContext;
PortalContext = PortalGetHeapMemory(portal);
saveQueryContext = QueryContext; saveQueryContext = QueryContext;
PG_TRY();
{
ActivePortal = portal;
CurrentResourceOwner = portal->resowner;
PortalContext = PortalGetHeapMemory(portal);
QueryContext = portal->queryContext; QueryContext = portal->queryContext;
oldContext = MemoryContextSwitchTo(PortalContext); oldContext = MemoryContextSwitchTo(PortalContext);
...@@ -994,6 +1029,21 @@ PortalRunFetch(Portal portal, ...@@ -994,6 +1029,21 @@ PortalRunFetch(Portal portal,
result = 0; /* keep compiler quiet */ result = 0; /* keep compiler quiet */
break; break;
} }
}
PG_CATCH();
{
/* Uncaught error while executing portal: mark it dead */
portal->status = PORTAL_FAILED;
/* Restore global vars and propagate error */
ActivePortal = saveActivePortal;
CurrentResourceOwner = saveResourceOwner;
PortalContext = savePortalContext;
QueryContext = saveQueryContext;
PG_RE_THROW();
}
PG_END_TRY();
MemoryContextSwitchTo(oldContext); MemoryContextSwitchTo(oldContext);
......
...@@ -37,14 +37,13 @@ ...@@ -37,14 +37,13 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/utils/error/elog.c,v 1.143 2004/07/28 22:05:46 tgl Exp $ * $PostgreSQL: pgsql/src/backend/utils/error/elog.c,v 1.144 2004/07/31 00:45:38 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "postgres.h" #include "postgres.h"
#include <fcntl.h> #include <fcntl.h>
#include <errno.h>
#include <time.h> #include <time.h>
#include <unistd.h> #include <unistd.h>
#include <signal.h> #include <signal.h>
...@@ -67,6 +66,8 @@ ...@@ -67,6 +66,8 @@
/* Global variables */ /* Global variables */
ErrorContextCallback *error_context_stack = NULL; ErrorContextCallback *error_context_stack = NULL;
sigjmp_buf *PG_exception_stack = NULL;
/* GUC parameters */ /* GUC parameters */
PGErrorVerbosity Log_error_verbosity = PGERROR_VERBOSE; PGErrorVerbosity Log_error_verbosity = PGERROR_VERBOSE;
char *Log_line_prefix = NULL; /* format for extra log line info */ char *Log_line_prefix = NULL; /* format for extra log line info */
...@@ -82,33 +83,6 @@ static void write_syslog(int level, const char *line); ...@@ -82,33 +83,6 @@ static void write_syslog(int level, const char *line);
static void write_eventlog(int level, const char *line); static void write_eventlog(int level, const char *line);
#endif #endif
/*
* ErrorData holds the data accumulated during any one ereport() cycle.
* Any non-NULL pointers must point to palloc'd data in ErrorContext.
* (The const pointers are an exception; we assume they point at non-freeable
* constant strings.)
*/
typedef struct ErrorData
{
int elevel; /* error level */
bool output_to_server; /* will report to server log? */
bool output_to_client; /* will report to client? */
bool show_funcname; /* true to force funcname inclusion */
const char *filename; /* __FILE__ of ereport() call */
int lineno; /* __LINE__ of ereport() call */
const char *funcname; /* __func__ of ereport() call */
int sqlerrcode; /* encoded ERRSTATE */
char *message; /* primary error message */
char *detail; /* detail error message */
char *hint; /* hint message */
char *context; /* context message */
int cursorpos; /* cursor index into query string */
int internalpos; /* cursor index into internalquery */
char *internalquery; /* text of internally-generated query */
int saved_errno; /* errno at entry */
} ErrorData;
/* We provide a small stack of ErrorData records for re-entrant cases */ /* We provide a small stack of ErrorData records for re-entrant cases */
#define ERRORDATA_STACK_SIZE 5 #define ERRORDATA_STACK_SIZE 5
...@@ -166,7 +140,7 @@ errstart(int elevel, const char *filename, int lineno, ...@@ -166,7 +140,7 @@ errstart(int elevel, const char *filename, int lineno,
/* /*
* Convert initialization errors into fatal errors. This is probably * Convert initialization errors into fatal errors. This is probably
* redundant, because Warn_restart_ready won't be set anyway. * redundant, because PG_exception_stack will still be null anyway.
*/ */
if (elevel == ERROR && IsInitProcessingMode()) if (elevel == ERROR && IsInitProcessingMode())
elevel = FATAL; elevel = FATAL;
...@@ -257,20 +231,13 @@ errstart(int elevel, const char *filename, int lineno, ...@@ -257,20 +231,13 @@ errstart(int elevel, const char *filename, int lineno,
} }
if (++errordata_stack_depth >= ERRORDATA_STACK_SIZE) if (++errordata_stack_depth >= ERRORDATA_STACK_SIZE)
{ {
/* Wups, stack not big enough */
int i;
elevel = Max(elevel, ERROR);
/* /*
* Don't forget any FATAL/PANIC status on the stack (see comments * Wups, stack not big enough. We treat this as a PANIC condition
* in errfinish) * because it suggests an infinite loop of errors during error
* recovery.
*/ */
for (i = 0; i < errordata_stack_depth; i++) errordata_stack_depth = -1; /* make room on stack */
elevel = Max(elevel, errordata[i].elevel); ereport(PANIC, (errmsg_internal("ERRORDATA_STACK_SIZE exceeded")));
/* Clear the stack and try again */
errordata_stack_depth = -1;
ereport(elevel, (errmsg_internal("ERRORDATA_STACK_SIZE exceeded")));
} }
/* Initialize data for this error frame */ /* Initialize data for this error frame */
...@@ -331,23 +298,90 @@ errfinish(int dummy,...) ...@@ -331,23 +298,90 @@ errfinish(int dummy,...)
econtext = econtext->previous) econtext = econtext->previous)
(*econtext->callback) (econtext->arg); (*econtext->callback) (econtext->arg);
/* Send to server log, if enabled */ /*
if (edata->output_to_server) * If the error level is ERROR or more, we are not going to return to
send_message_to_server_log(edata); * caller; therefore, if there is any stacked error already in
* progress it will be lost. This is more or less okay, except we do
* not want to have a FATAL or PANIC error downgraded because the
* reporting process was interrupted by a lower-grade error. So check
* the stack and make sure we panic if panic is warranted.
*/
if (elevel >= ERROR)
{
int i;
for (i = 0; i <= errordata_stack_depth; i++)
elevel = Max(elevel, errordata[i].elevel);
}
/*
* Check some other reasons for treating ERROR as FATAL:
*
* 1. we have no handler to pass the error to (implies we are in
* the postmaster or in backend startup).
*
* 2. ExitOnAnyError mode switch is set (initdb uses this).
*
* 3. the error occurred after proc_exit has begun to run. (It's
* proc_exit's responsibility to see that this doesn't turn into
* infinite recursion!)
*/
if (elevel == ERROR)
{
if (PG_exception_stack == NULL ||
ExitOnAnyError ||
proc_exit_inprogress)
elevel = FATAL;
else
{
/*
* Otherwise we can pass the error off to the current handler.
* Printing it and popping the stack is the responsibility of
* the handler.
*
* We do some minimal cleanup before longjmp'ing so that handlers
* can execute in a reasonably sane state.
*/
/* This is just in case the error came while waiting for input */
ImmediateInterruptOK = false;
/* /*
* Abort any old-style COPY OUT in progress when an error is detected. * Reset InterruptHoldoffCount in case we ereport'd from inside an
* interrupt holdoff section. (We assume here that no handler
* will itself be inside a holdoff section. If necessary, such
* a handler could save and restore InterruptHoldoffCount for
* itself, but this should make life easier for most.)
*/
InterruptHoldoffCount = 0;
CritSectionCount = 0; /* should be unnecessary, but... */
/*
* Note that we leave CurrentMemoryContext set to ErrorContext.
* The handler should reset it to something else soon.
*/
recursion_depth--;
PG_RE_THROW();
}
}
/*
* If we are doing FATAL or PANIC, abort any old-style COPY OUT in
* progress, so that we can report the message before dying. (Without
* this, pq_putmessage will refuse to send the message at all, which
* is what we want for NOTICE messages, but not for fatal exits.)
* This hack is necessary because of poor design of old-style copy * This hack is necessary because of poor design of old-style copy
* protocol. Note we must do this even if client is fool enough to * protocol. Note we must do this even if client is fool enough to
* have set client_min_messages above ERROR, so don't look at * have set client_min_messages above FATAL, so don't look at
* output_to_client. * output_to_client.
*/ */
if (elevel >= ERROR && whereToSendOutput == Remote) if (elevel >= FATAL && whereToSendOutput == Remote)
pq_endcopyout(true); pq_endcopyout(true);
/* Send to client, if enabled */ /* Emit the message to the right places */
if (edata->output_to_client) EmitErrorReport();
send_message_to_frontend(edata);
/* Now free up subsidiary data attached to stack entry, and release it */ /* Now free up subsidiary data attached to stack entry, and release it */
if (edata->message) if (edata->message)
...@@ -361,41 +395,20 @@ errfinish(int dummy,...) ...@@ -361,41 +395,20 @@ errfinish(int dummy,...)
if (edata->internalquery) if (edata->internalquery)
pfree(edata->internalquery); pfree(edata->internalquery);
MemoryContextSwitchTo(oldcontext);
errordata_stack_depth--; errordata_stack_depth--;
/* Exit error-handling context */
MemoryContextSwitchTo(oldcontext);
recursion_depth--; recursion_depth--;
/* /*
* If the error level is ERROR or more, we are not going to return to * Perform error recovery action as specified by elevel.
* caller; therefore, if there is any stacked error already in
* progress it will be lost. This is more or less okay, except we do
* not want to have a FATAL or PANIC error downgraded because the
* reporting process was interrupted by a lower-grade error. So check
* the stack and make sure we panic if panic is warranted.
*/ */
if (elevel >= ERROR) if (elevel == FATAL)
{ {
int i;
for (i = 0; i <= errordata_stack_depth; i++)
elevel = Max(elevel, errordata[i].elevel);
/*
* Also, be sure to reset the stack to empty. We do not clear
* ErrorContext here, though; PostgresMain does that later on.
*/
errordata_stack_depth = -1;
recursion_depth = 0;
error_context_stack = NULL;
}
/* /*
* Perform error recovery action as specified by elevel. * For a FATAL error, we let proc_exit clean up and exit.
*/ */
if (elevel == ERROR || elevel == FATAL)
{
/* Prevent immediate interrupt while entering error recovery */
ImmediateInterruptOK = false; ImmediateInterruptOK = false;
/* /*
...@@ -403,36 +416,9 @@ errfinish(int dummy,...) ...@@ -403,36 +416,9 @@ errfinish(int dummy,...)
* disconnect on receiving it, so don't send any more to the * disconnect on receiving it, so don't send any more to the
* client. * client.
*/ */
if (!Warn_restart_ready && whereToSendOutput == Remote) if (PG_exception_stack == NULL && whereToSendOutput == Remote)
whereToSendOutput = None; whereToSendOutput = None;
/*
* For a FATAL error, we let proc_exit clean up and exit.
*
* There are several other cases in which we treat ERROR as FATAL and
* go directly to proc_exit:
*
* 1. ExitOnAnyError mode switch is set (initdb uses this).
*
* 2. we have not yet entered the main backend loop (ie, we are in
* the postmaster or in backend startup); we have noplace to
* recover.
*
* 3. the error occurred after proc_exit has begun to run. (It's
* proc_exit's responsibility to see that this doesn't turn into
* infinite recursion!)
*
* In the last case, we exit with nonzero exit code to indicate that
* something's pretty wrong. We also want to exit with nonzero
* exit code if not running under the postmaster (for example, if
* we are being run from the initdb script, we'd better return an
* error status).
*/
if (elevel == FATAL ||
ExitOnAnyError ||
!Warn_restart_ready ||
proc_exit_inprogress)
{
/* /*
* fflush here is just to improve the odds that we get to see * fflush here is just to improve the odds that we get to see
* the error message, in case things are so hosed that * the error message, in case things are so hosed that
...@@ -442,20 +428,15 @@ errfinish(int dummy,...) ...@@ -442,20 +428,15 @@ errfinish(int dummy,...)
*/ */
fflush(stdout); fflush(stdout);
fflush(stderr); fflush(stderr);
proc_exit(proc_exit_inprogress || !IsUnderPostmaster);
}
/*
* Guard against infinite loop from errors during error recovery.
*/
if (InError)
ereport(PANIC, (errmsg("error during error recovery, giving up")));
InError = true;
/* /*
* Otherwise we can return to the main loop in postgres.c. * If proc_exit is already running, we exit with nonzero exit code to
* indicate that something's pretty wrong. We also want to exit with
* nonzero exit code if not running under the postmaster (for example,
* if we are being run from the initdb script, we'd better return an
* error status).
*/ */
siglongjmp(Warn_restart, 1); proc_exit(proc_exit_inprogress || !IsUnderPostmaster);
} }
if (elevel >= PANIC) if (elevel >= PANIC)
...@@ -923,6 +904,168 @@ elog_finish(int elevel, const char *fmt,...) ...@@ -923,6 +904,168 @@ elog_finish(int elevel, const char *fmt,...)
errfinish(0); errfinish(0);
} }
/*
* Actual output of the top-of-stack error message
*
* In the ereport(ERROR) case this is called from PostgresMain (or not at all,
* if the error is caught by somebody). For all other severity levels this
* is called by errfinish.
*/
void
EmitErrorReport(void)
{
ErrorData *edata = &errordata[errordata_stack_depth];
MemoryContext oldcontext;
recursion_depth++;
CHECK_STACK_DEPTH();
oldcontext = MemoryContextSwitchTo(ErrorContext);
/* Send to server log, if enabled */
if (edata->output_to_server)
send_message_to_server_log(edata);
/* Send to client, if enabled */
if (edata->output_to_client)
send_message_to_frontend(edata);
MemoryContextSwitchTo(oldcontext);
recursion_depth--;
}
/*
* CopyErrorData --- obtain a copy of the topmost error stack entry
*
* This is only for use in error handler code. The data is copied into the
* current memory context, so callers should always switch away from
* ErrorContext first; otherwise it will be lost when FlushErrorState is done.
*/
ErrorData *
CopyErrorData(void)
{
ErrorData *edata = &errordata[errordata_stack_depth];
ErrorData *newedata;
/*
* we don't increment recursion_depth because out-of-memory here does
* not indicate a problem within the error subsystem.
*/
CHECK_STACK_DEPTH();
Assert(CurrentMemoryContext != ErrorContext);
/* Copy the struct itself */
newedata = (ErrorData *) palloc(sizeof(ErrorData));
memcpy(newedata, edata, sizeof(ErrorData));
/* Make copies of separately-allocated fields */
if (newedata->message)
newedata->message = pstrdup(newedata->message);
if (newedata->detail)
newedata->detail = pstrdup(newedata->detail);
if (newedata->hint)
newedata->hint = pstrdup(newedata->hint);
if (newedata->context)
newedata->context = pstrdup(newedata->context);
if (newedata->internalquery)
newedata->internalquery = pstrdup(newedata->internalquery);
return newedata;
}
/*
* FreeErrorData --- free the structure returned by CopyErrorData.
*
* Error handlers should use this in preference to assuming they know all
* the separately-allocated fields.
*/
void
FreeErrorData(ErrorData *edata)
{
if (edata->message)
pfree(edata->message);
if (edata->detail)
pfree(edata->detail);
if (edata->hint)
pfree(edata->hint);
if (edata->context)
pfree(edata->context);
if (edata->internalquery)
pfree(edata->internalquery);
pfree(edata);
}
/*
* FlushErrorState --- flush the error state after error recovery
*
* This should be called by an error handler after it's done processing
* the error; or as soon as it's done CopyErrorData, if it intends to
* do stuff that is likely to provoke another error. You are not "out" of
* the error subsystem until you have done this.
*/
void
FlushErrorState(void)
{
/*
* Reset stack to empty. The only case where it would be more than
* one deep is if we serviced an error that interrupted construction
* of another message. We assume control escaped out of that
* message construction and won't ever go back.
*/
errordata_stack_depth = -1;
recursion_depth = 0;
/* Delete all data in ErrorContext */
MemoryContextResetAndDeleteChildren(ErrorContext);
}
/*
* ReThrowError --- re-throw a previously copied error
*
* A handler can do CopyErrorData/FlushErrorState to get out of the error
* subsystem, then do some processing, and finally ReThrowError to re-throw
* the original error. This is slower than just PG_RE_THROW() but should
* be used if the "some processing" is likely to incur another error.
*/
void
ReThrowError(ErrorData *edata)
{
ErrorData *newedata;
Assert(edata->elevel == ERROR);
/* Push the data back into the error context */
recursion_depth++;
MemoryContextSwitchTo(ErrorContext);
if (++errordata_stack_depth >= ERRORDATA_STACK_SIZE)
{
/*
* Wups, stack not big enough. We treat this as a PANIC condition
* because it suggests an infinite loop of errors during error
* recovery.
*/
errordata_stack_depth = -1; /* make room on stack */
ereport(PANIC, (errmsg_internal("ERRORDATA_STACK_SIZE exceeded")));
}
newedata = &errordata[errordata_stack_depth];
memcpy(newedata, edata, sizeof(ErrorData));
/* Make copies of separately-allocated fields */
if (newedata->message)
newedata->message = pstrdup(newedata->message);
if (newedata->detail)
newedata->detail = pstrdup(newedata->detail);
if (newedata->hint)
newedata->hint = pstrdup(newedata->hint);
if (newedata->context)
newedata->context = pstrdup(newedata->context);
if (newedata->internalquery)
newedata->internalquery = pstrdup(newedata->internalquery);
recursion_depth--;
PG_RE_THROW();
}
/* /*
* Initialization of error output file * Initialization of error output file
......
$PostgreSQL: pgsql/src/backend/utils/resowner/README,v 1.1 2004/07/17 03:30:10 tgl Exp $ $PostgreSQL: pgsql/src/backend/utils/resowner/README,v 1.2 2004/07/31 00:45:40 tgl Exp $
Notes about resource owners Notes about resource owners
--------------------------- ---------------------------
...@@ -72,3 +72,7 @@ CurrentResourceOwner must point to the same resource owner that was current ...@@ -72,3 +72,7 @@ CurrentResourceOwner must point to the same resource owner that was current
when the buffer, lock, or cache reference was acquired. It would be possible when the buffer, lock, or cache reference was acquired. It would be possible
to relax this restriction given additional bookkeeping effort, but at present to relax this restriction given additional bookkeeping effort, but at present
there seems no need. there seems no need.
Code that transiently changes CurrentResourceOwner should generally use a
PG_TRY construct to ensure that the previous value of CurrentResourceOwner
is restored if control is lost during an error exit.
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/utils/resowner/resowner.c,v 1.1 2004/07/17 03:30:10 tgl Exp $ * $PostgreSQL: pgsql/src/backend/utils/resowner/resowner.c,v 1.2 2004/07/31 00:45:40 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -99,6 +99,13 @@ typedef struct ResourceReleaseCallbackItem ...@@ -99,6 +99,13 @@ typedef struct ResourceReleaseCallbackItem
static ResourceReleaseCallbackItem *ResourceRelease_callbacks = NULL; static ResourceReleaseCallbackItem *ResourceRelease_callbacks = NULL;
/* Internal routines */
static void ResourceOwnerReleaseInternal(ResourceOwner owner,
ResourceReleasePhase phase,
bool isCommit,
bool isTopLevel);
/***************************************************************************** /*****************************************************************************
* EXPORTED ROUTINES * * EXPORTED ROUTINES *
*****************************************************************************/ *****************************************************************************/
...@@ -161,6 +168,29 @@ ResourceOwnerRelease(ResourceOwner owner, ...@@ -161,6 +168,29 @@ ResourceOwnerRelease(ResourceOwner owner,
ResourceReleasePhase phase, ResourceReleasePhase phase,
bool isCommit, bool isCommit,
bool isTopLevel) bool isTopLevel)
{
/* Rather than PG_TRY at every level of recursion, set it up once */
ResourceOwner save;
save = CurrentResourceOwner;
PG_TRY();
{
ResourceOwnerReleaseInternal(owner, phase, isCommit, isTopLevel);
}
PG_CATCH();
{
CurrentResourceOwner = save;
PG_RE_THROW();
}
PG_END_TRY();
CurrentResourceOwner = save;
}
static void
ResourceOwnerReleaseInternal(ResourceOwner owner,
ResourceReleasePhase phase,
bool isCommit,
bool isTopLevel)
{ {
ResourceOwner child; ResourceOwner child;
ResourceOwner save; ResourceOwner save;
...@@ -168,11 +198,12 @@ ResourceOwnerRelease(ResourceOwner owner, ...@@ -168,11 +198,12 @@ ResourceOwnerRelease(ResourceOwner owner,
/* Recurse to handle descendants */ /* Recurse to handle descendants */
for (child = owner->firstchild; child != NULL; child = child->nextchild) for (child = owner->firstchild; child != NULL; child = child->nextchild)
ResourceOwnerRelease(child, phase, isCommit, isTopLevel); ResourceOwnerReleaseInternal(child, phase, isCommit, isTopLevel);
/* /*
* Make CurrentResourceOwner point to me, so that ReleaseBuffer etc * Make CurrentResourceOwner point to me, so that ReleaseBuffer etc
* don't get confused. * don't get confused. We needn't PG_TRY here because the outermost
* level will fix it on error abort.
*/ */
save = CurrentResourceOwner; save = CurrentResourceOwner;
CurrentResourceOwner = owner; CurrentResourceOwner = owner;
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/tcop/tcopprot.h,v 1.68 2004/07/28 22:05:47 tgl Exp $ * $PostgreSQL: pgsql/src/include/tcop/tcopprot.h,v 1.69 2004/07/31 00:45:41 tgl Exp $
* *
* OLD COMMENTS * OLD COMMENTS
* This file was created so that other c files could get the two * This file was created so that other c files could get the two
...@@ -19,17 +19,12 @@ ...@@ -19,17 +19,12 @@
#ifndef TCOPPROT_H #ifndef TCOPPROT_H
#define TCOPPROT_H #define TCOPPROT_H
#include <setjmp.h>
#include "executor/execdesc.h" #include "executor/execdesc.h"
#include "nodes/params.h" #include "nodes/params.h"
#include "tcop/dest.h" #include "tcop/dest.h"
#include "utils/guc.h" #include "utils/guc.h"
extern DLLIMPORT sigjmp_buf Warn_restart;
extern bool Warn_restart_ready;
extern bool InError;
extern CommandDest whereToSendOutput; extern CommandDest whereToSendOutput;
extern DLLIMPORT const char *debug_query_string; extern DLLIMPORT const char *debug_query_string;
extern int max_stack_depth; extern int max_stack_depth;
......
...@@ -7,13 +7,15 @@ ...@@ -7,13 +7,15 @@
* Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/utils/elog.h,v 1.70 2004/07/06 19:51:59 momjian Exp $ * $PostgreSQL: pgsql/src/include/utils/elog.h,v 1.71 2004/07/31 00:45:43 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#ifndef ELOG_H #ifndef ELOG_H
#define ELOG_H #define ELOG_H
#include <setjmp.h>
/* Error level codes */ /* Error level codes */
#define DEBUG5 10 /* Debugging messages, in categories of #define DEBUG5 10 /* Debugging messages, in categories of
* decreasing detail. */ * decreasing detail. */
...@@ -168,6 +170,93 @@ typedef struct ErrorContextCallback ...@@ -168,6 +170,93 @@ typedef struct ErrorContextCallback
extern DLLIMPORT ErrorContextCallback *error_context_stack; extern DLLIMPORT ErrorContextCallback *error_context_stack;
/*----------
* API for catching ereport(ERROR) exits. Use these macros like so:
*
* PG_TRY();
* {
* ... code that might throw ereport(ERROR) ...
* }
* PG_CATCH();
* {
* ... error recovery code ...
* }
* PG_END_TRY();
*
* (The braces are not actually necessary, but are recommended so that
* pg_indent will indent the construct nicely.) The error recovery code
* can optionally do PG_RE_THROW() to propagate the same error outwards.
*
* Note: while the system will correctly propagate any new ereport(ERROR)
* occurring in the recovery section, there is a small limit on the number
* of levels this will work for. It's best to keep the error recovery
* section simple enough that it can't generate any new errors, at least
* not before popping the error stack.
*----------
*/
#define PG_TRY() \
do { \
sigjmp_buf *save_exception_stack = PG_exception_stack; \
ErrorContextCallback *save_context_stack = error_context_stack; \
sigjmp_buf local_sigjmp_buf; \
if (sigsetjmp(local_sigjmp_buf, 1) == 0) \
{ \
PG_exception_stack = &local_sigjmp_buf
#define PG_CATCH() \
} \
else \
{ \
PG_exception_stack = save_exception_stack; \
error_context_stack = save_context_stack
#define PG_END_TRY() \
} \
PG_exception_stack = save_exception_stack; \
error_context_stack = save_context_stack; \
} while (0)
#define PG_RE_THROW() \
siglongjmp(*PG_exception_stack, 1)
extern DLLIMPORT sigjmp_buf *PG_exception_stack;
/* Stuff that error handlers might want to use */
/*
* ErrorData holds the data accumulated during any one ereport() cycle.
* Any non-NULL pointers must point to palloc'd data.
* (The const pointers are an exception; we assume they point at non-freeable
* constant strings.)
*/
typedef struct ErrorData
{
int elevel; /* error level */
bool output_to_server; /* will report to server log? */
bool output_to_client; /* will report to client? */
bool show_funcname; /* true to force funcname inclusion */
const char *filename; /* __FILE__ of ereport() call */
int lineno; /* __LINE__ of ereport() call */
const char *funcname; /* __func__ of ereport() call */
int sqlerrcode; /* encoded ERRSTATE */
char *message; /* primary error message */
char *detail; /* detail error message */
char *hint; /* hint message */
char *context; /* context message */
int cursorpos; /* cursor index into query string */
int internalpos; /* cursor index into internalquery */
char *internalquery; /* text of internally-generated query */
int saved_errno; /* errno at entry */
} ErrorData;
extern void EmitErrorReport(void);
extern ErrorData *CopyErrorData(void);
extern void FreeErrorData(ErrorData *edata);
extern void FlushErrorState(void);
extern void ReThrowError(ErrorData *edata);
/* GUC-configurable parameters */ /* GUC-configurable parameters */
typedef enum typedef enum
......
...@@ -33,16 +33,15 @@ ...@@ -33,16 +33,15 @@
* ENHANCEMENTS, OR MODIFICATIONS. * ENHANCEMENTS, OR MODIFICATIONS.
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plperl/plperl.c,v 1.47 2004/07/21 20:45:54 momjian Exp $ * $PostgreSQL: pgsql/src/pl/plperl/plperl.c,v 1.48 2004/07/31 00:45:44 tgl Exp $
* *
**********************************************************************/ **********************************************************************/
#include "postgres.h" #include "postgres.h"
/* system stuff */ /* system stuff */
#include <unistd.h>
#include <fcntl.h> #include <fcntl.h>
#include <setjmp.h> #include <unistd.h>
/* postgreSQL stuff */ /* postgreSQL stuff */
#include "access/heapam.h" #include "access/heapam.h"
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* procedural language * procedural language
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.77 2004/06/06 00:41:28 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_comp.c,v 1.78 2004/07/31 00:45:46 tgl Exp $
* *
* This software is copyrighted by Jan Wieck - Hamburg. * This software is copyrighted by Jan Wieck - Hamburg.
* *
...@@ -38,7 +38,6 @@ ...@@ -38,7 +38,6 @@
#include "plpgsql.h" #include "plpgsql.h"
#include <ctype.h> #include <ctype.h>
#include <setjmp.h>
#include "pl.tab.h" #include "pl.tab.h"
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* procedural language * procedural language
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.107 2004/06/09 19:08:19 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpgsql/src/pl_exec.c,v 1.108 2004/07/31 00:45:46 tgl Exp $
* *
* This software is copyrighted by Jan Wieck - Hamburg. * This software is copyrighted by Jan Wieck - Hamburg.
* *
...@@ -39,7 +39,6 @@ ...@@ -39,7 +39,6 @@
#include "pl.tab.h" #include "pl.tab.h"
#include <ctype.h> #include <ctype.h>
#include <setjmp.h>
#include "access/heapam.h" #include "access/heapam.h"
#include "catalog/pg_proc.h" #include "catalog/pg_proc.h"
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
* MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. * MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/plpython/plpython.c,v 1.49 2004/06/06 00:41:28 tgl Exp $ * $PostgreSQL: pgsql/src/pl/plpython/plpython.c,v 1.50 2004/07/31 00:45:52 tgl Exp $
* *
********************************************************************* *********************************************************************
*/ */
...@@ -39,7 +39,6 @@ ...@@ -39,7 +39,6 @@
/* system stuff */ /* system stuff */
#include <unistd.h> #include <unistd.h>
#include <fcntl.h> #include <fcntl.h>
#include <setjmp.h>
/* postgreSQL stuff */ /* postgreSQL stuff */
#include "access/heapam.h" #include "access/heapam.h"
...@@ -58,7 +57,6 @@ ...@@ -58,7 +57,6 @@
#include <Python.h> #include <Python.h>
#include <compile.h> #include <compile.h>
#include <eval.h> #include <eval.h>
#include "plpython.h"
/* convert Postgresql Datum or tuple into a PyObject. /* convert Postgresql Datum or tuple into a PyObject.
* input to Python. Tuples are converted to dictionary * input to Python. Tuples are converted to dictionary
...@@ -192,11 +190,6 @@ static void PLy_init_all(void); ...@@ -192,11 +190,6 @@ static void PLy_init_all(void);
static void PLy_init_interp(void); static void PLy_init_interp(void);
static void PLy_init_plpy(void); static void PLy_init_plpy(void);
/* error handler. collects the current Python exception, if any,
* and appends it to the error and sends it to elog
*/
static void PLy_elog(int, const char *,...);
/* call PyErr_SetString with a vprint interface /* call PyErr_SetString with a vprint interface
*/ */
static void static void
...@@ -209,6 +202,11 @@ static char *PLy_procedure_name(PLyProcedure *); ...@@ -209,6 +202,11 @@ static char *PLy_procedure_name(PLyProcedure *);
/* some utility functions /* some utility functions
*/ */
static void PLy_elog(int, const char *,...);
static char *PLy_traceback(int *);
static char *PLy_vprintf(const char *fmt, va_list ap);
static char *PLy_printf(const char *fmt,...);
static void *PLy_malloc(size_t); static void *PLy_malloc(size_t);
static void *PLy_realloc(void *, size_t); static void *PLy_realloc(void *, size_t);
static void PLy_free(void *); static void PLy_free(void *);
...@@ -259,18 +257,26 @@ static PyObject *PLyString_FromString(const char *); ...@@ -259,18 +257,26 @@ static PyObject *PLyString_FromString(const char *);
/* global data /* global data
*/ */
static int PLy_first_call = 1; static int PLy_first_call = 1;
static volatile int PLy_call_level = 0;
/* /*
* Last function called by postgres backend * Last function called by postgres backend
*
* XXX replace this with errcontext mechanism
*/ */
static PLyProcedure *PLy_last_procedure = NULL; static PLyProcedure *PLy_last_procedure = NULL;
/* this gets modified in plpython_call_handler and PLy_elog. /*
* test it any old where, but do NOT modify it anywhere except * When a callback from Python into PG incurs an error, we temporarily store
* those two functions * the error information here, and return NULL to the Python interpreter.
*/ * Any further callback attempts immediately fail, and when the Python
static volatile int PLy_restart_in_progress = 0; * interpreter returns to the calling function, we re-throw the error (even if
* Python thinks it trapped the error and doesn't return NULL). Eventually
* this ought to be improved to let Python code really truly trap the error,
* but that's more of a change from the pre-7.5 semantics than I have time for
* now --- it will only be possible if the callback query is executed inside a
* subtransaction.
*/
static ErrorData *PLy_error_in_progress = NULL;
static PyObject *PLy_interp_globals = NULL; static PyObject *PLy_interp_globals = NULL;
static PyObject *PLy_interp_safe_globals = NULL; static PyObject *PLy_interp_safe_globals = NULL;
...@@ -293,13 +299,6 @@ static char PLy_result_doc[] = { ...@@ -293,13 +299,6 @@ static char PLy_result_doc[] = {
}; };
#if DEBUG_EXC
volatile int exc_save_calls = 0;
volatile int exc_restore_calls = 0;
volatile int func_enter_calls = 0;
volatile int func_leave_calls = 0;
#endif
/* /*
* the function definitions * the function definitions
*/ */
...@@ -324,44 +323,16 @@ perm_fmgr_info(Oid functionId, FmgrInfo *finfo) ...@@ -324,44 +323,16 @@ perm_fmgr_info(Oid functionId, FmgrInfo *finfo)
Datum Datum
plpython_call_handler(PG_FUNCTION_ARGS) plpython_call_handler(PG_FUNCTION_ARGS)
{ {
DECLARE_EXC();
Datum retval; Datum retval;
PLyProcedure *volatile proc = NULL; PLyProcedure *volatile proc = NULL;
enter();
PLy_init_all(); PLy_init_all();
if (SPI_connect() != SPI_OK_CONNECT) if (SPI_connect() != SPI_OK_CONNECT)
elog(ERROR, "could not connect to SPI manager"); elog(ERROR, "could not connect to SPI manager");
CALL_LEVEL_INC(); PG_TRY();
SAVE_EXC();
if (TRAP_EXC())
{
RESTORE_EXC();
CALL_LEVEL_DEC();
if (PLy_call_level == 0)
{
PLy_restart_in_progress = 0;
PyErr_Clear();
}
else
PLy_restart_in_progress += 1;
if (proc)
{ {
/* note: Py_DECREF needs braces around it, as of 2003/08 */
Py_DECREF(proc->me);
}
RERAISE_EXC();
}
/*
* elog(DEBUG3, "PLy_restart_in_progress is %d",
* PLy_restart_in_progress);
*/
if (CALLED_AS_TRIGGER(fcinfo)) if (CALLED_AS_TRIGGER(fcinfo))
{ {
TriggerData *tdata = (TriggerData *) fcinfo->context; TriggerData *tdata = (TriggerData *) fcinfo->context;
...@@ -377,12 +348,20 @@ plpython_call_handler(PG_FUNCTION_ARGS) ...@@ -377,12 +348,20 @@ plpython_call_handler(PG_FUNCTION_ARGS)
proc = PLy_procedure_get(fcinfo, InvalidOid); proc = PLy_procedure_get(fcinfo, InvalidOid);
retval = PLy_function_handler(fcinfo, proc); retval = PLy_function_handler(fcinfo, proc);
} }
}
CALL_LEVEL_DEC(); PG_CATCH();
RESTORE_EXC(); {
if (proc)
{
/* note: Py_DECREF needs braces around it, as of 2003/08 */
Py_DECREF(proc->me);
}
PyErr_Clear();
PG_RE_THROW();
}
PG_END_TRY();
Py_DECREF(proc->me); Py_DECREF(proc->me);
refc(proc->me);
return retval; return retval;
} }
...@@ -397,42 +376,27 @@ plpython_call_handler(PG_FUNCTION_ARGS) ...@@ -397,42 +376,27 @@ plpython_call_handler(PG_FUNCTION_ARGS)
* BEFORE the event and is ROW level. postgres expects the function * BEFORE the event and is ROW level. postgres expects the function
* to take no arguments and return an argument of type trigger. * to take no arguments and return an argument of type trigger.
*/ */
HeapTuple static HeapTuple
PLy_trigger_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) PLy_trigger_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
{ {
DECLARE_EXC();
HeapTuple rv = NULL; HeapTuple rv = NULL;
PyObject *volatile plargs = NULL; PyObject *volatile plargs = NULL;
PyObject *volatile plrv = NULL; PyObject *volatile plrv = NULL;
enter(); PG_TRY();
SAVE_EXC();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_XDECREF(plargs);
Py_XDECREF(plrv);
RERAISE_EXC();
}
plargs = PLy_trigger_build_args(fcinfo, proc, &rv); plargs = PLy_trigger_build_args(fcinfo, proc, &rv);
plrv = PLy_procedure_call(proc, "TD", plargs); plrv = PLy_procedure_call(proc, "TD", plargs);
Assert(plrv != NULL);
Assert(!PLy_error_in_progress);
/* /*
* Disconnect from SPI manager * Disconnect from SPI manager
*/ */
if (SPI_finish() != SPI_OK_FINISH) if (SPI_finish() != SPI_OK_FINISH)
elog(ERROR, "SPI_finish failed"); elog(ERROR, "SPI_finish failed");
if (plrv == NULL)
elog(FATAL, "PLy_procedure_call returned NULL");
if (PLy_restart_in_progress)
elog(FATAL, "restart in progress not expected");
/* /*
* return of None means we're happy with the tuple * return of None means we're happy with the tuple
*/ */
...@@ -466,20 +430,26 @@ PLy_trigger_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) ...@@ -466,20 +430,26 @@ PLy_trigger_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
elog(ERROR, "expected return to be \"SKIP\" or \"MODIFY\""); elog(ERROR, "expected return to be \"SKIP\" or \"MODIFY\"");
} }
} }
}
PG_CATCH();
{
Py_XDECREF(plargs);
Py_XDECREF(plrv);
PG_RE_THROW();
}
PG_END_TRY();
Py_DECREF(plargs); Py_DECREF(plargs);
Py_DECREF(plrv); Py_DECREF(plrv);
RESTORE_EXC();
return rv; return rv;
} }
HeapTuple static HeapTuple
PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata, PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata,
HeapTuple otup) HeapTuple otup)
{ {
DECLARE_EXC();
PyObject *volatile plntup; PyObject *volatile plntup;
PyObject *volatile plkeys; PyObject *volatile plkeys;
PyObject *volatile platt; PyObject *volatile platt;
...@@ -500,29 +470,8 @@ PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata, ...@@ -500,29 +470,8 @@ PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata,
modvalues = NULL; modvalues = NULL;
modnulls = NULL; modnulls = NULL;
enter(); PG_TRY();
SAVE_EXC();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_XDECREF(plntup);
Py_XDECREF(plkeys);
Py_XDECREF(platt);
Py_XDECREF(plval);
Py_XDECREF(plstr);
if (modnulls)
pfree(modnulls);
if (modvalues)
pfree(modvalues);
if (modattrs)
pfree(modattrs);
RERAISE_EXC();
}
if ((plntup = PyDict_GetItemString(pltd, "new")) == NULL) if ((plntup = PyDict_GetItemString(pltd, "new")) == NULL)
elog(ERROR, "TD[\"new\"] deleted, unable to modify tuple"); elog(ERROR, "TD[\"new\"] deleted, unable to modify tuple");
if (!PyDict_Check(plntup)) if (!PyDict_Check(plntup))
...@@ -585,31 +534,42 @@ PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata, ...@@ -585,31 +534,42 @@ PLy_modify_tuple(PLyProcedure * proc, PyObject * pltd, TriggerData *tdata,
rtup = SPI_modifytuple(tdata->tg_relation, otup, natts, rtup = SPI_modifytuple(tdata->tg_relation, otup, natts,
modattrs, modvalues, modnulls); modattrs, modvalues, modnulls);
if (rtup == NULL)
elog(ERROR, "SPI_modifytuple failed -- error %d", SPI_result);
}
PG_CATCH();
{
Py_XDECREF(plntup);
Py_XDECREF(plkeys);
Py_XDECREF(platt);
Py_XDECREF(plval);
Py_XDECREF(plstr);
/* if (modnulls)
* FIXME -- these leak if not explicitly pfree'd by other elog calls,
* no? (No, I think, but might as well leave the pfrees here...)
*/
pfree(modattrs);
pfree(modvalues);
pfree(modnulls); pfree(modnulls);
if (modvalues)
pfree(modvalues);
if (modattrs)
pfree(modattrs);
if (rtup == NULL) PG_RE_THROW();
elog(ERROR, "SPI_modifytuple failed -- error %d", SPI_result); }
PG_END_TRY();
Py_DECREF(plntup); Py_DECREF(plntup);
Py_DECREF(plkeys); Py_DECREF(plkeys);
RESTORE_EXC(); pfree(modattrs);
pfree(modvalues);
pfree(modnulls);
return rtup; return rtup;
} }
PyObject * static PyObject *
PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple *rv) PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple *rv)
{ {
DECLARE_EXC(); TriggerData *tdata = (TriggerData *) fcinfo->context;
TriggerData *tdata;
PyObject *pltname, PyObject *pltname,
*pltevent, *pltevent,
*pltwhen, *pltwhen,
...@@ -621,20 +581,8 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple * ...@@ -621,20 +581,8 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple *
PyObject *volatile pltdata = NULL; PyObject *volatile pltdata = NULL;
char *stroid; char *stroid;
enter(); PG_TRY();
SAVE_EXC();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_XDECREF(pltdata);
RERAISE_EXC();
}
tdata = (TriggerData *) fcinfo->context;
pltdata = PyDict_New(); pltdata = PyDict_New();
if (!pltdata) if (!pltdata)
PLy_elog(ERROR, "could not build arguments for trigger procedure"); PLy_elog(ERROR, "could not build arguments for trigger procedure");
...@@ -767,8 +715,13 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple * ...@@ -767,8 +715,13 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple *
} }
PyDict_SetItemString(pltdata, "args", pltargs); PyDict_SetItemString(pltdata, "args", pltargs);
Py_DECREF(pltargs); Py_DECREF(pltargs);
}
RESTORE_EXC(); PG_CATCH();
{
Py_XDECREF(pltdata);
PG_RE_THROW();
}
PG_END_TRY();
return pltdata; return pltdata;
} }
...@@ -777,37 +730,23 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple * ...@@ -777,37 +730,23 @@ PLy_trigger_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc, HeapTuple *
/* function handler and friends /* function handler and friends
*/ */
Datum static Datum
PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
{ {
DECLARE_EXC();
Datum rv; Datum rv;
PyObject *volatile plargs = NULL; PyObject *volatile plargs = NULL;
PyObject *volatile plrv = NULL; PyObject *volatile plrv = NULL;
PyObject *volatile plrv_so = NULL; PyObject *volatile plrv_so = NULL;
char *plrv_sc; char *plrv_sc;
enter(); PG_TRY();
/*
* setup to catch elog in while building function arguments, and
* DECREF the plargs if the function call fails
*/
SAVE_EXC();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_XDECREF(plargs);
Py_XDECREF(plrv);
Py_XDECREF(plrv_so);
RERAISE_EXC();
}
plargs = PLy_function_build_args(fcinfo, proc); plargs = PLy_function_build_args(fcinfo, proc);
plrv = PLy_procedure_call(proc, "args", plargs); plrv = PLy_procedure_call(proc, "args", plargs);
Assert(plrv != NULL);
Assert(!PLy_error_in_progress);
/* /*
* Disconnect from SPI manager and then create the return values datum * Disconnect from SPI manager and then create the return values datum
* (if the input function does a palloc for it this must not be * (if the input function does a palloc for it this must not be
...@@ -817,28 +756,9 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) ...@@ -817,28 +756,9 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
if (SPI_finish() != SPI_OK_FINISH) if (SPI_finish() != SPI_OK_FINISH)
elog(ERROR, "SPI_finish failed"); elog(ERROR, "SPI_finish failed");
if (plrv == NULL)
{
elog(FATAL, "PLy_procedure_call returned NULL");
#ifdef NOT_USED
if (!PLy_restart_in_progress)
PLy_elog(ERROR, "function \"%s\" failed", proc->proname);
/* /*
* FIXME is this dead code? i'm pretty sure it is for unnested * convert the python PyObject to a postgresql Datum
* calls, but not for nested calls
*/ */
RAISE_EXC(1);
#endif
}
/*
* convert the python PyObject to a postgresql Datum FIXME returning a
* NULL, ie PG_RETURN_NULL() blows the backend to small messy bits...
* it this a bug or expected? so just call with the string value of
* None for now
*/
if (plrv == Py_None) if (plrv == Py_None)
{ {
fcinfo->isnull = true; fcinfo->isnull = true;
...@@ -855,7 +775,16 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) ...@@ -855,7 +775,16 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
Int32GetDatum(-1)); Int32GetDatum(-1));
} }
RESTORE_EXC(); }
PG_CATCH();
{
Py_XDECREF(plargs);
Py_XDECREF(plrv);
Py_XDECREF(plrv_so);
PG_RE_THROW();
}
PG_END_TRY();
Py_XDECREF(plargs); Py_XDECREF(plargs);
Py_DECREF(plrv); Py_DECREF(plrv);
...@@ -864,56 +793,49 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc) ...@@ -864,56 +793,49 @@ PLy_function_handler(FunctionCallInfo fcinfo, PLyProcedure * proc)
return rv; return rv;
} }
PyObject * static PyObject *
PLy_procedure_call(PLyProcedure * proc, char *kargs, PyObject * vargs) PLy_procedure_call(PLyProcedure * proc, char *kargs, PyObject * vargs)
{ {
PyObject *rv; PyObject *rv;
PLyProcedure *current; PLyProcedure *current;
enter();
current = PLy_last_procedure; current = PLy_last_procedure;
PLy_last_procedure = proc; PLy_last_procedure = proc;
PyDict_SetItemString(proc->globals, kargs, vargs); PyDict_SetItemString(proc->globals, kargs, vargs);
rv = PyEval_EvalCode((PyCodeObject *) proc->code, proc->globals, proc->globals); rv = PyEval_EvalCode((PyCodeObject *) proc->code,
proc->globals, proc->globals);
PLy_last_procedure = current; PLy_last_procedure = current;
/*
* If there was an error in a PG callback, propagate that
* no matter what Python claims about its success.
*/
if (PLy_error_in_progress)
{
ErrorData *edata = PLy_error_in_progress;
PLy_error_in_progress = NULL;
ReThrowError(edata);
}
if ((rv == NULL) || (PyErr_Occurred())) if ((rv == NULL) || (PyErr_Occurred()))
{ {
Py_XDECREF(rv); Py_XDECREF(rv);
if (!PLy_restart_in_progress)
PLy_elog(ERROR, "function \"%s\" failed", proc->proname); PLy_elog(ERROR, "function \"%s\" failed", proc->proname);
RAISE_EXC(1);
} }
return rv; return rv;
} }
PyObject * static PyObject *
PLy_function_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc) PLy_function_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc)
{ {
DECLARE_EXC();
PyObject *volatile arg = NULL; PyObject *volatile arg = NULL;
PyObject *volatile args = NULL; PyObject *volatile args = NULL;
int i; int i;
enter(); PG_TRY();
/*
* FIXME -- if the setjmp setup is expensive, add the arg and args
* field to the procedure struct and cleanup at the start of the next
* call
*/
SAVE_EXC();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_XDECREF(arg);
Py_XDECREF(args);
RERAISE_EXC();
}
args = PyList_New(proc->nargs); args = PyList_New(proc->nargs);
for (i = 0; i < proc->nargs; i++) for (i = 0; i < proc->nargs; i++)
{ {
...@@ -976,8 +898,15 @@ PLy_function_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc) ...@@ -976,8 +898,15 @@ PLy_function_build_args(FunctionCallInfo fcinfo, PLyProcedure * proc)
*/ */
PyList_SetItem(args, i, arg); PyList_SetItem(args, i, arg);
} }
}
PG_CATCH();
{
Py_XDECREF(arg);
Py_XDECREF(args);
RESTORE_EXC(); PG_RE_THROW();
}
PG_END_TRY();
return args; return args;
} }
...@@ -1002,8 +931,6 @@ PLy_procedure_get(FunctionCallInfo fcinfo, Oid tgreloid) ...@@ -1002,8 +931,6 @@ PLy_procedure_get(FunctionCallInfo fcinfo, Oid tgreloid)
PLyProcedure *proc = NULL; PLyProcedure *proc = NULL;
int rv; int rv;
enter();
fn_oid = fcinfo->flinfo->fn_oid; fn_oid = fcinfo->flinfo->fn_oid;
procTup = SearchSysCache(PROCOID, procTup = SearchSysCache(PROCOID,
ObjectIdGetDatum(fn_oid), ObjectIdGetDatum(fn_oid),
...@@ -1023,8 +950,6 @@ PLy_procedure_get(FunctionCallInfo fcinfo, Oid tgreloid) ...@@ -1023,8 +950,6 @@ PLy_procedure_get(FunctionCallInfo fcinfo, Oid tgreloid)
if (!PyCObject_Check(plproc)) if (!PyCObject_Check(plproc))
elog(FATAL, "expected a PyCObject, didn't get one"); elog(FATAL, "expected a PyCObject, didn't get one");
mark();
proc = PyCObject_AsVoidPtr(plproc); proc = PyCObject_AsVoidPtr(plproc);
if (proc->me != plproc) if (proc->me != plproc)
elog(FATAL, "proc->me != plproc"); elog(FATAL, "proc->me != plproc");
...@@ -1051,7 +976,6 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid, ...@@ -1051,7 +976,6 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid,
{ {
char procName[NAMEDATALEN + 256]; char procName[NAMEDATALEN + 256];
DECLARE_EXC();
Form_pg_proc procStruct; Form_pg_proc procStruct;
PLyProcedure *volatile proc; PLyProcedure *volatile proc;
char *volatile procSource = NULL; char *volatile procSource = NULL;
...@@ -1060,8 +984,6 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid, ...@@ -1060,8 +984,6 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid,
int i, int i,
rv; rv;
enter();
procStruct = (Form_pg_proc) GETSTRUCT(procTup); procStruct = (Form_pg_proc) GETSTRUCT(procTup);
if (OidIsValid(tgreloid)) if (OidIsValid(tgreloid))
...@@ -1092,16 +1014,8 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid, ...@@ -1092,16 +1014,8 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid,
proc->code = proc->statics = NULL; proc->code = proc->statics = NULL;
proc->globals = proc->me = NULL; proc->globals = proc->me = NULL;
SAVE_EXC(); PG_TRY();
if (TRAP_EXC())
{ {
RESTORE_EXC();
PLy_procedure_delete(proc);
if (procSource)
pfree(procSource);
RERAISE_EXC();
}
/* /*
* get information required for output conversion of the return value, * get information required for output conversion of the return value,
* but only if this isn't a trigger. * but only if this isn't a trigger.
...@@ -1185,20 +1099,26 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid, ...@@ -1185,20 +1099,26 @@ PLy_procedure_create(FunctionCallInfo fcinfo, Oid tgreloid,
proc->me = PyCObject_FromVoidPtr(proc, NULL); proc->me = PyCObject_FromVoidPtr(proc, NULL);
PyDict_SetItemString(PLy_procedure_cache, key, proc->me); PyDict_SetItemString(PLy_procedure_cache, key, proc->me);
}
PG_CATCH();
{
PLy_procedure_delete(proc);
if (procSource)
pfree(procSource);
RESTORE_EXC(); PG_RE_THROW();
}
PG_END_TRY();
return proc; return proc;
} }
void static void
PLy_procedure_compile(PLyProcedure * proc, const char *src) PLy_procedure_compile(PLyProcedure * proc, const char *src)
{ {
PyObject *crv = NULL; PyObject *crv = NULL;
char *msrc; char *msrc;
enter();
proc->globals = PyDict_Copy(PLy_interp_globals); proc->globals = PyDict_Copy(PLy_interp_globals);
/* /*
...@@ -1238,7 +1158,7 @@ PLy_procedure_compile(PLyProcedure * proc, const char *src) ...@@ -1238,7 +1158,7 @@ PLy_procedure_compile(PLyProcedure * proc, const char *src)
PLy_elog(ERROR, "could not compile function \"%s\"", proc->proname); PLy_elog(ERROR, "could not compile function \"%s\"", proc->proname);
} }
char * static char *
PLy_procedure_munge_source(const char *name, const char *src) PLy_procedure_munge_source(const char *name, const char *src)
{ {
char *mrc, char *mrc,
...@@ -1247,8 +1167,6 @@ PLy_procedure_munge_source(const char *name, const char *src) ...@@ -1247,8 +1167,6 @@ PLy_procedure_munge_source(const char *name, const char *src)
size_t mlen, size_t mlen,
plen; plen;
enter();
/* /*
* room for function source and the def statement * room for function source and the def statement
*/ */
...@@ -1281,13 +1199,11 @@ PLy_procedure_munge_source(const char *name, const char *src) ...@@ -1281,13 +1199,11 @@ PLy_procedure_munge_source(const char *name, const char *src)
return mrc; return mrc;
} }
void static void
PLy_procedure_delete(PLyProcedure * proc) PLy_procedure_delete(PLyProcedure * proc)
{ {
int i; int i;
enter();
Py_XDECREF(proc->code); Py_XDECREF(proc->code);
Py_XDECREF(proc->statics); Py_XDECREF(proc->statics);
Py_XDECREF(proc->globals); Py_XDECREF(proc->globals);
...@@ -1304,20 +1220,16 @@ PLy_procedure_delete(PLyProcedure * proc) ...@@ -1304,20 +1220,16 @@ PLy_procedure_delete(PLyProcedure * proc)
if (proc->args[i].out.r.atts) if (proc->args[i].out.r.atts)
PLy_free(proc->args[i].out.r.atts); PLy_free(proc->args[i].out.r.atts);
} }
leave();
} }
/* conversion functions. remember output from python is /* conversion functions. remember output from python is
* input to postgresql, and vis versa. * input to postgresql, and vis versa.
*/ */
void static void
PLy_input_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc) PLy_input_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc)
{ {
int i; int i;
enter();
if (arg->is_rowtype == 0) if (arg->is_rowtype == 0)
elog(ERROR, "PLyTypeInfo struct is initialized for a Datum"); elog(ERROR, "PLyTypeInfo struct is initialized for a Datum");
...@@ -1347,13 +1259,11 @@ PLy_input_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc) ...@@ -1347,13 +1259,11 @@ PLy_input_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc)
} }
} }
void static void
PLy_output_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc) PLy_output_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc)
{ {
int i; int i;
enter();
if (arg->is_rowtype == 0) if (arg->is_rowtype == 0)
elog(ERROR, "PLyTypeInfo struct is initialized for a Datum"); elog(ERROR, "PLyTypeInfo struct is initialized for a Datum");
...@@ -1381,41 +1291,35 @@ PLy_output_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc) ...@@ -1381,41 +1291,35 @@ PLy_output_tuple_funcs(PLyTypeInfo * arg, TupleDesc desc)
} }
} }
void static void
PLy_output_datum_func(PLyTypeInfo * arg, HeapTuple typeTup) PLy_output_datum_func(PLyTypeInfo * arg, HeapTuple typeTup)
{ {
enter();
if (arg->is_rowtype > 0) if (arg->is_rowtype > 0)
elog(ERROR, "PLyTypeInfo struct is initialized for a Tuple"); elog(ERROR, "PLyTypeInfo struct is initialized for a Tuple");
arg->is_rowtype = 0; arg->is_rowtype = 0;
PLy_output_datum_func2(&(arg->out.d), typeTup); PLy_output_datum_func2(&(arg->out.d), typeTup);
} }
void static void
PLy_output_datum_func2(PLyObToDatum * arg, HeapTuple typeTup) PLy_output_datum_func2(PLyObToDatum * arg, HeapTuple typeTup)
{ {
Form_pg_type typeStruct = (Form_pg_type) GETSTRUCT(typeTup); Form_pg_type typeStruct = (Form_pg_type) GETSTRUCT(typeTup);
enter();
perm_fmgr_info(typeStruct->typinput, &arg->typfunc); perm_fmgr_info(typeStruct->typinput, &arg->typfunc);
arg->typioparam = getTypeIOParam(typeTup); arg->typioparam = getTypeIOParam(typeTup);
arg->typbyval = typeStruct->typbyval; arg->typbyval = typeStruct->typbyval;
} }
void static void
PLy_input_datum_func(PLyTypeInfo * arg, Oid typeOid, HeapTuple typeTup) PLy_input_datum_func(PLyTypeInfo * arg, Oid typeOid, HeapTuple typeTup)
{ {
enter();
if (arg->is_rowtype > 0) if (arg->is_rowtype > 0)
elog(ERROR, "PLyTypeInfo struct is initialized for Tuple"); elog(ERROR, "PLyTypeInfo struct is initialized for Tuple");
arg->is_rowtype = 0; arg->is_rowtype = 0;
PLy_input_datum_func2(&(arg->in.d), typeOid, typeTup); PLy_input_datum_func2(&(arg->in.d), typeOid, typeTup);
} }
void static void
PLy_input_datum_func2(PLyDatumToOb * arg, Oid typeOid, HeapTuple typeTup) PLy_input_datum_func2(PLyDatumToOb * arg, Oid typeOid, HeapTuple typeTup)
{ {
Form_pg_type typeStruct = (Form_pg_type) GETSTRUCT(typeTup); Form_pg_type typeStruct = (Form_pg_type) GETSTRUCT(typeTup);
...@@ -1449,7 +1353,7 @@ PLy_input_datum_func2(PLyDatumToOb * arg, Oid typeOid, HeapTuple typeTup) ...@@ -1449,7 +1353,7 @@ PLy_input_datum_func2(PLyDatumToOb * arg, Oid typeOid, HeapTuple typeTup)
} }
} }
void static void
PLy_typeinfo_init(PLyTypeInfo * arg) PLy_typeinfo_init(PLyTypeInfo * arg)
{ {
arg->is_rowtype = -1; arg->is_rowtype = -1;
...@@ -1458,7 +1362,7 @@ PLy_typeinfo_init(PLyTypeInfo * arg) ...@@ -1458,7 +1362,7 @@ PLy_typeinfo_init(PLyTypeInfo * arg)
arg->out.r.atts = NULL; arg->out.r.atts = NULL;
} }
void static void
PLy_typeinfo_dealloc(PLyTypeInfo * arg) PLy_typeinfo_dealloc(PLyTypeInfo * arg)
{ {
if (arg->is_rowtype == 1) if (arg->is_rowtype == 1)
...@@ -1472,24 +1376,20 @@ PLy_typeinfo_dealloc(PLyTypeInfo * arg) ...@@ -1472,24 +1376,20 @@ PLy_typeinfo_dealloc(PLyTypeInfo * arg)
/* assumes that a bool is always returned as a 't' or 'f' /* assumes that a bool is always returned as a 't' or 'f'
*/ */
PyObject * static PyObject *
PLyBool_FromString(const char *src) PLyBool_FromString(const char *src)
{ {
enter();
if (src[0] == 't') if (src[0] == 't')
return PyInt_FromLong(1); return PyInt_FromLong(1);
return PyInt_FromLong(0); return PyInt_FromLong(0);
} }
PyObject * static PyObject *
PLyFloat_FromString(const char *src) PLyFloat_FromString(const char *src)
{ {
double v; double v;
char *eptr; char *eptr;
enter();
errno = 0; errno = 0;
v = strtod(src, &eptr); v = strtod(src, &eptr);
if ((*eptr != '\0') || (errno)) if ((*eptr != '\0') || (errno))
...@@ -1497,14 +1397,12 @@ PLyFloat_FromString(const char *src) ...@@ -1497,14 +1397,12 @@ PLyFloat_FromString(const char *src)
return PyFloat_FromDouble(v); return PyFloat_FromDouble(v);
} }
PyObject * static PyObject *
PLyInt_FromString(const char *src) PLyInt_FromString(const char *src)
{ {
long v; long v;
char *eptr; char *eptr;
enter();
errno = 0; errno = 0;
v = strtol(src, &eptr, 0); v = strtol(src, &eptr, 0);
if ((*eptr != '\0') || (errno)) if ((*eptr != '\0') || (errno))
...@@ -1512,27 +1410,24 @@ PLyInt_FromString(const char *src) ...@@ -1512,27 +1410,24 @@ PLyInt_FromString(const char *src)
return PyInt_FromLong(v); return PyInt_FromLong(v);
} }
PyObject * static PyObject *
PLyLong_FromString(const char *src) PLyLong_FromString(const char *src)
{ {
return PyLong_FromString((char *) src, NULL, 0); return PyLong_FromString((char *) src, NULL, 0);
} }
PyObject * static PyObject *
PLyString_FromString(const char *src) PLyString_FromString(const char *src)
{ {
return PyString_FromString(src); return PyString_FromString(src);
} }
PyObject * static PyObject *
PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc) PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc)
{ {
DECLARE_EXC();
PyObject *volatile dict; PyObject *volatile dict;
int i; int i;
enter();
if (info->is_rowtype != 1) if (info->is_rowtype != 1)
elog(ERROR, "PLyTypeInfo structure describes a datum"); elog(ERROR, "PLyTypeInfo structure describes a datum");
...@@ -1540,15 +1435,8 @@ PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc) ...@@ -1540,15 +1435,8 @@ PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc)
if (dict == NULL) if (dict == NULL)
PLy_elog(ERROR, "could not create tuple dictionary"); PLy_elog(ERROR, "could not create tuple dictionary");
SAVE_EXC(); PG_TRY();
if (TRAP_EXC())
{ {
RESTORE_EXC();
Py_DECREF(dict);
RERAISE_EXC();
}
for (i = 0; i < info->in.r.natts; i++) for (i = 0; i < info->in.r.natts; i++)
{ {
char *key, char *key,
...@@ -1583,8 +1471,13 @@ PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc) ...@@ -1583,8 +1471,13 @@ PLyDict_FromTuple(PLyTypeInfo * info, HeapTuple tuple, TupleDesc desc)
Py_DECREF(value); Py_DECREF(value);
} }
} }
}
RESTORE_EXC(); PG_CATCH();
{
Py_DECREF(dict);
PG_RE_THROW();
}
PG_END_TRY();
return dict; return dict;
} }
...@@ -1747,13 +1640,11 @@ static PyMethodDef PLy_methods[] = { ...@@ -1747,13 +1640,11 @@ static PyMethodDef PLy_methods[] = {
/* plan object methods /* plan object methods
*/ */
PyObject * static PyObject *
PLy_plan_new(void) PLy_plan_new(void)
{ {
PLyPlanObject *ob; PLyPlanObject *ob;
enter();
if ((ob = PyObject_NEW(PLyPlanObject, &PLy_PlanType)) == NULL) if ((ob = PyObject_NEW(PLyPlanObject, &PLy_PlanType)) == NULL)
return NULL; return NULL;
...@@ -1766,13 +1657,11 @@ PLy_plan_new(void) ...@@ -1766,13 +1657,11 @@ PLy_plan_new(void)
} }
void static void
PLy_plan_dealloc(PyObject * arg) PLy_plan_dealloc(PyObject * arg)
{ {
PLyPlanObject *ob = (PLyPlanObject *) arg; PLyPlanObject *ob = (PLyPlanObject *) arg;
enter();
if (ob->plan) if (ob->plan)
SPI_freeplan(ob->plan); SPI_freeplan(ob->plan);
if (ob->types) if (ob->types)
...@@ -1787,18 +1676,16 @@ PLy_plan_dealloc(PyObject * arg) ...@@ -1787,18 +1676,16 @@ PLy_plan_dealloc(PyObject * arg)
} }
PyMem_DEL(arg); PyMem_DEL(arg);
leave();
} }
PyObject * static PyObject *
PLy_plan_getattr(PyObject * self, char *name) PLy_plan_getattr(PyObject * self, char *name)
{ {
return Py_FindMethod(PLy_plan_methods, self, name); return Py_FindMethod(PLy_plan_methods, self, name);
} }
PyObject * static PyObject *
PLy_plan_status(PyObject * self, PyObject * args) PLy_plan_status(PyObject * self, PyObject * args)
{ {
if (PyArg_ParseTuple(args, "")) if (PyArg_ParseTuple(args, ""))
...@@ -1816,13 +1703,11 @@ PLy_plan_status(PyObject * self, PyObject * args) ...@@ -1816,13 +1703,11 @@ PLy_plan_status(PyObject * self, PyObject * args)
/* result object methods /* result object methods
*/ */
PyObject * static PyObject *
PLy_result_new(void) PLy_result_new(void)
{ {
PLyResultObject *ob; PLyResultObject *ob;
enter();
if ((ob = PyObject_NEW(PLyResultObject, &PLy_ResultType)) == NULL) if ((ob = PyObject_NEW(PLyResultObject, &PLy_ResultType)) == NULL)
return NULL; return NULL;
...@@ -1836,13 +1721,11 @@ PLy_result_new(void) ...@@ -1836,13 +1721,11 @@ PLy_result_new(void)
return (PyObject *) ob; return (PyObject *) ob;
} }
void static void
PLy_result_dealloc(PyObject * arg) PLy_result_dealloc(PyObject * arg)
{ {
PLyResultObject *ob = (PLyResultObject *) arg; PLyResultObject *ob = (PLyResultObject *) arg;
enter();
Py_XDECREF(ob->nrows); Py_XDECREF(ob->nrows);
Py_XDECREF(ob->rows); Py_XDECREF(ob->rows);
Py_XDECREF(ob->status); Py_XDECREF(ob->status);
...@@ -1850,7 +1733,7 @@ PLy_result_dealloc(PyObject * arg) ...@@ -1850,7 +1733,7 @@ PLy_result_dealloc(PyObject * arg)
PyMem_DEL(ob); PyMem_DEL(ob);
} }
PyObject * static PyObject *
PLy_result_getattr(PyObject * self, char *attr) PLy_result_getattr(PyObject * self, char *attr)
{ {
return NULL; return NULL;
...@@ -1858,13 +1741,13 @@ PLy_result_getattr(PyObject * self, char *attr) ...@@ -1858,13 +1741,13 @@ PLy_result_getattr(PyObject * self, char *attr)
#ifdef NOT_USED #ifdef NOT_USED
/* Appear to be unused */ /* Appear to be unused */
PyObject * static PyObject *
PLy_result_fetch(PyObject * self, PyObject * args) PLy_result_fetch(PyObject * self, PyObject * args)
{ {
return NULL; return NULL;
} }
PyObject * static PyObject *
PLy_result_nrows(PyObject * self, PyObject * args) PLy_result_nrows(PyObject * self, PyObject * args)
{ {
PLyResultObject *ob = (PLyResultObject *) self; PLyResultObject *ob = (PLyResultObject *) self;
...@@ -1873,7 +1756,7 @@ PLy_result_nrows(PyObject * self, PyObject * args) ...@@ -1873,7 +1756,7 @@ PLy_result_nrows(PyObject * self, PyObject * args)
return ob->nrows; return ob->nrows;
} }
PyObject * static PyObject *
PLy_result_status(PyObject * self, PyObject * args) PLy_result_status(PyObject * self, PyObject * args)
{ {
PLyResultObject *ob = (PLyResultObject *) self; PLyResultObject *ob = (PLyResultObject *) self;
...@@ -1882,7 +1765,8 @@ PLy_result_status(PyObject * self, PyObject * args) ...@@ -1882,7 +1765,8 @@ PLy_result_status(PyObject * self, PyObject * args)
return ob->status; return ob->status;
} }
#endif #endif
int
static int
PLy_result_length(PyObject * arg) PLy_result_length(PyObject * arg)
{ {
PLyResultObject *ob = (PLyResultObject *) arg; PLyResultObject *ob = (PLyResultObject *) arg;
...@@ -1890,7 +1774,7 @@ PLy_result_length(PyObject * arg) ...@@ -1890,7 +1774,7 @@ PLy_result_length(PyObject * arg)
return PyList_Size(ob->rows); return PyList_Size(ob->rows);
} }
PyObject * static PyObject *
PLy_result_item(PyObject * arg, int idx) PLy_result_item(PyObject * arg, int idx)
{ {
PyObject *rv; PyObject *rv;
...@@ -1902,7 +1786,7 @@ PLy_result_item(PyObject * arg, int idx) ...@@ -1902,7 +1786,7 @@ PLy_result_item(PyObject * arg, int idx)
return rv; return rv;
} }
int static int
PLy_result_ass_item(PyObject * arg, int idx, PyObject * item) PLy_result_ass_item(PyObject * arg, int idx, PyObject * item)
{ {
int rv; int rv;
...@@ -1913,7 +1797,7 @@ PLy_result_ass_item(PyObject * arg, int idx, PyObject * item) ...@@ -1913,7 +1797,7 @@ PLy_result_ass_item(PyObject * arg, int idx, PyObject * item)
return rv; return rv;
} }
PyObject * static PyObject *
PLy_result_slice(PyObject * arg, int lidx, int hidx) PLy_result_slice(PyObject * arg, int lidx, int hidx)
{ {
PyObject *rv; PyObject *rv;
...@@ -1926,7 +1810,7 @@ PLy_result_slice(PyObject * arg, int lidx, int hidx) ...@@ -1926,7 +1810,7 @@ PLy_result_slice(PyObject * arg, int lidx, int hidx)
return rv; return rv;
} }
int static int
PLy_result_ass_slice(PyObject * arg, int lidx, int hidx, PyObject * slice) PLy_result_ass_slice(PyObject * arg, int lidx, int hidx, PyObject * slice)
{ {
int rv; int rv;
...@@ -1938,17 +1822,22 @@ PLy_result_ass_slice(PyObject * arg, int lidx, int hidx, PyObject * slice) ...@@ -1938,17 +1822,22 @@ PLy_result_ass_slice(PyObject * arg, int lidx, int hidx, PyObject * slice)
/* SPI interface /* SPI interface
*/ */
PyObject * static PyObject *
PLy_spi_prepare(PyObject * self, PyObject * args) PLy_spi_prepare(PyObject * self, PyObject * args)
{ {
DECLARE_EXC();
PLyPlanObject *plan; PLyPlanObject *plan;
PyObject *list = NULL; PyObject *list = NULL;
PyObject *volatile optr = NULL; PyObject *volatile optr = NULL;
char *query; char *query;
void *tmpplan; void *tmpplan;
MemoryContext oldcontext;
enter(); /* Can't execute more if we have an unhandled error */
if (PLy_error_in_progress)
{
PyErr_SetString(PLy_exc_error, "Transaction aborted.");
return NULL;
}
if (!PyArg_ParseTuple(args, "s|O", &query, &list)) if (!PyArg_ParseTuple(args, "s|O", &query, &list))
{ {
...@@ -1964,24 +1853,12 @@ PLy_spi_prepare(PyObject * self, PyObject * args) ...@@ -1964,24 +1853,12 @@ PLy_spi_prepare(PyObject * self, PyObject * args)
return NULL; return NULL;
} }
if ((plan = (PLyPlanObject *) PLy_plan_new()) == NULL) if ((plan = (PLyPlanObject *) PLy_plan_new()) == NULL)
return NULL; return NULL;
SAVE_EXC(); oldcontext = CurrentMemoryContext;
if (TRAP_EXC()) PG_TRY();
{ {
RESTORE_EXC();
Py_DECREF(plan);
Py_XDECREF(optr);
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_spi_error,
"Unknown error in PLy_spi_prepare");
/* XXX this oughta be replaced with errcontext mechanism */
PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure));
RERAISE_EXC();
}
if (list != NULL) if (list != NULL)
{ {
int nargs, int nargs,
...@@ -2014,11 +1891,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args) ...@@ -2014,11 +1891,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args)
optr = PySequence_GetItem(list, i); optr = PySequence_GetItem(list, i);
if (!PyString_Check(optr)) if (!PyString_Check(optr))
{ elog(ERROR, "Type names must be strings.");
PyErr_SetString(PLy_exc_spi_error,
"Type names must be strings.");
RAISE_EXC(1);
}
sptr = PyString_AsString(optr); sptr = PyString_AsString(optr);
/* XXX should extend this to allow qualified type names */ /* XXX should extend this to allow qualified type names */
typeTup = typenameType(makeTypeName(sptr)); typeTup = typenameType(makeTypeName(sptr));
...@@ -2030,11 +1903,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args) ...@@ -2030,11 +1903,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args)
if (typeStruct->typtype != 'c') if (typeStruct->typtype != 'c')
PLy_output_datum_func(&plan->args[i], typeTup); PLy_output_datum_func(&plan->args[i], typeTup);
else else
{ elog(ERROR, "tuples not handled in plpy.prepare, yet.");
PyErr_SetString(PLy_exc_spi_error,
"tuples not handled in plpy.prepare, yet.");
RAISE_EXC(1);
}
ReleaseSysCache(typeTup); ReleaseSysCache(typeTup);
} }
} }
...@@ -2042,26 +1911,32 @@ PLy_spi_prepare(PyObject * self, PyObject * args) ...@@ -2042,26 +1911,32 @@ PLy_spi_prepare(PyObject * self, PyObject * args)
plan->plan = SPI_prepare(query, plan->nargs, plan->types); plan->plan = SPI_prepare(query, plan->nargs, plan->types);
if (plan->plan == NULL) if (plan->plan == NULL)
{ elog(ERROR, "Unable to prepare plan. SPI_prepare failed -- %s.",
PLy_exception_set(PLy_exc_spi_error,
"Unable to prepare plan. SPI_prepare failed -- %s.",
PLy_spi_error_string(SPI_result)); PLy_spi_error_string(SPI_result));
RAISE_EXC(1);
}
/* transfer plan from procCxt to topCxt */ /* transfer plan from procCxt to topCxt */
tmpplan = plan->plan; tmpplan = plan->plan;
plan->plan = SPI_saveplan(tmpplan); plan->plan = SPI_saveplan(tmpplan);
SPI_freeplan(tmpplan); SPI_freeplan(tmpplan);
if (plan->plan == NULL) if (plan->plan == NULL)
{ elog(ERROR, "Unable to save plan. SPI_saveplan failed -- %s.",
PLy_exception_set(PLy_exc_spi_error,
"Unable to save plan. SPI_saveplan failed -- %s.",
PLy_spi_error_string(SPI_result)); PLy_spi_error_string(SPI_result));
RAISE_EXC(1);
} }
PG_CATCH();
RESTORE_EXC(); {
MemoryContextSwitchTo(oldcontext);
PLy_error_in_progress = CopyErrorData();
FlushErrorState();
Py_DECREF(plan);
Py_XDECREF(optr);
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_spi_error,
"Unknown error in PLy_spi_prepare");
/* XXX this oughta be replaced with errcontext mechanism */
PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure));
return NULL;
}
PG_END_TRY();
return (PyObject *) plan; return (PyObject *) plan;
} }
...@@ -2069,7 +1944,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args) ...@@ -2069,7 +1944,7 @@ PLy_spi_prepare(PyObject * self, PyObject * args)
/* execute(query="select * from foo", limit=5) /* execute(query="select * from foo", limit=5)
* execute(plan=plan, values=(foo, bar), limit=5) * execute(plan=plan, values=(foo, bar), limit=5)
*/ */
PyObject * static PyObject *
PLy_spi_execute(PyObject * self, PyObject * args) PLy_spi_execute(PyObject * self, PyObject * args)
{ {
char *query; char *query;
...@@ -2077,17 +1952,12 @@ PLy_spi_execute(PyObject * self, PyObject * args) ...@@ -2077,17 +1952,12 @@ PLy_spi_execute(PyObject * self, PyObject * args)
PyObject *list = NULL; PyObject *list = NULL;
int limit = 0; int limit = 0;
enter(); /* Can't execute more if we have an unhandled error */
if (PLy_error_in_progress)
#ifdef NOT_USED {
PyErr_SetString(PLy_exc_error, "Transaction aborted.");
/*
* there should - hahaha - be an python exception set so just return
* NULL. FIXME -- is this needed?
*/
if (PLy_restart_in_progress)
return NULL; return NULL;
#endif }
if (PyArg_ParseTuple(args, "s|i", &query, &limit)) if (PyArg_ParseTuple(args, "s|i", &query, &limit))
return PLy_spi_execute_query(query, limit); return PLy_spi_execute_query(query, limit);
...@@ -2096,27 +1966,21 @@ PLy_spi_execute(PyObject * self, PyObject * args) ...@@ -2096,27 +1966,21 @@ PLy_spi_execute(PyObject * self, PyObject * args)
if ((PyArg_ParseTuple(args, "O|Oi", &plan, &list, &limit)) && if ((PyArg_ParseTuple(args, "O|Oi", &plan, &list, &limit)) &&
(is_PLyPlanObject(plan))) (is_PLyPlanObject(plan)))
{ return PLy_spi_execute_plan(plan, list, limit);
PyObject *rv = PLy_spi_execute_plan(plan, list, limit);
return rv;
}
PyErr_SetString(PLy_exc_error, "Expected a query or plan."); PyErr_SetString(PLy_exc_error, "Expected a query or plan.");
return NULL; return NULL;
} }
PyObject * static PyObject *
PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit) PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit)
{ {
DECLARE_EXC();
volatile int nargs; volatile int nargs;
int i, int i,
rv; rv;
PLyPlanObject *plan; PLyPlanObject *plan;
char *nulls; char *nulls;
MemoryContext oldcontext;
enter();
if (list != NULL) if (list != NULL)
{ {
...@@ -2149,31 +2013,9 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit) ...@@ -2149,31 +2013,9 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit)
return NULL; return NULL;
} }
SAVE_EXC(); oldcontext = CurrentMemoryContext;
if (TRAP_EXC()) PG_TRY();
{
RESTORE_EXC();
/*
* cleanup plan->values array
*/
for (i = 0; i < nargs; i++)
{
if (!plan->args[i].out.d.typbyval &&
(plan->values[i] != (Datum) NULL))
{ {
pfree(DatumGetPointer(plan->values[i]));
plan->values[i] = (Datum) NULL;
}
}
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_error,
"Unknown error in PLy_spi_execute_plan");
PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure));
RERAISE_EXC();
}
nulls = palloc(nargs * sizeof(char)); nulls = palloc(nargs * sizeof(char));
for (i = 0; i < nargs; i++) for (i = 0; i < nargs; i++)
...@@ -2189,7 +2031,7 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit) ...@@ -2189,7 +2031,7 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit)
sv = PyString_AsString(so); sv = PyString_AsString(so);
/* /*
* FIXME -- if this can elog, we have leak * FIXME -- if this elogs, we have Python reference leak
*/ */
plan->values[i] = plan->values[i] =
FunctionCall3(&(plan->args[i].out.d.typfunc), FunctionCall3(&(plan->args[i].out.d.typfunc),
...@@ -2213,8 +2055,32 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit) ...@@ -2213,8 +2055,32 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit)
rv = SPI_execp(plan->plan, plan->values, nulls, limit); rv = SPI_execp(plan->plan, plan->values, nulls, limit);
pfree(nulls); pfree(nulls);
}
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
PLy_error_in_progress = CopyErrorData();
FlushErrorState();
/*
* cleanup plan->values array
*/
for (i = 0; i < nargs; i++)
{
if (!plan->args[i].out.d.typbyval &&
(plan->values[i] != (Datum) NULL))
{
pfree(DatumGetPointer(plan->values[i]));
plan->values[i] = (Datum) NULL;
}
}
RESTORE_EXC(); if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_error,
"Unknown error in PLy_spi_execute_plan");
PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure));
return NULL;
}
PG_END_TRY();
for (i = 0; i < nargs; i++) for (i = 0; i < nargs; i++)
{ {
...@@ -2237,25 +2103,30 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit) ...@@ -2237,25 +2103,30 @@ PLy_spi_execute_plan(PyObject * ob, PyObject * list, int limit)
return PLy_spi_execute_fetch_result(SPI_tuptable, SPI_processed, rv); return PLy_spi_execute_fetch_result(SPI_tuptable, SPI_processed, rv);
} }
PyObject * static PyObject *
PLy_spi_execute_query(char *query, int limit) PLy_spi_execute_query(char *query, int limit)
{ {
DECLARE_EXC();
int rv; int rv;
MemoryContext oldcontext;
SAVE_EXC(); oldcontext = CurrentMemoryContext;
if (TRAP_EXC()) PG_TRY();
{ {
RESTORE_EXC(); rv = SPI_exec(query, limit);
if ((!PLy_restart_in_progress) && (!PyErr_Occurred())) }
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
PLy_error_in_progress = CopyErrorData();
FlushErrorState();
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_spi_error, PyErr_SetString(PLy_exc_spi_error,
"Unknown error in PLy_spi_execute_query"); "Unknown error in PLy_spi_execute_query");
PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure)); PLy_elog(WARNING, "in function %s:", PLy_procedure_name(PLy_last_procedure));
RERAISE_EXC(); return NULL;
} }
PG_END_TRY();
rv = SPI_exec(query, limit);
RESTORE_EXC();
if (rv < 0) if (rv < 0)
{ {
PLy_exception_set(PLy_exc_spi_error, PLy_exception_set(PLy_exc_spi_error,
...@@ -2267,12 +2138,11 @@ PLy_spi_execute_query(char *query, int limit) ...@@ -2267,12 +2138,11 @@ PLy_spi_execute_query(char *query, int limit)
return PLy_spi_execute_fetch_result(SPI_tuptable, SPI_processed, rv); return PLy_spi_execute_fetch_result(SPI_tuptable, SPI_processed, rv);
} }
PyObject * static PyObject *
PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status) PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status)
{ {
PLyResultObject *result; PLyResultObject *result;
MemoryContext oldcontext;
enter();
result = (PLyResultObject *) PLy_result_new(); result = (PLyResultObject *) PLy_result_new();
Py_DECREF(result->status); Py_DECREF(result->status);
...@@ -2290,7 +2160,6 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status) ...@@ -2290,7 +2160,6 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status)
} }
else else
{ {
DECLARE_EXC();
PLyTypeInfo args; PLyTypeInfo args;
int i; int i;
...@@ -2298,19 +2167,9 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status) ...@@ -2298,19 +2167,9 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status)
Py_DECREF(result->nrows); Py_DECREF(result->nrows);
result->nrows = PyInt_FromLong(rows); result->nrows = PyInt_FromLong(rows);
SAVE_EXC(); oldcontext = CurrentMemoryContext;
if (TRAP_EXC()) PG_TRY();
{ {
RESTORE_EXC();
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_error,
"Unknown error in PLy_spi_execute_fetch_result");
Py_DECREF(result);
PLy_typeinfo_dealloc(&args);
RERAISE_EXC();
}
if (rows) if (rows)
{ {
Py_DECREF(result->rows); Py_DECREF(result->rows);
...@@ -2328,13 +2187,26 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status) ...@@ -2328,13 +2187,26 @@ PLy_spi_execute_fetch_result(SPITupleTable *tuptable, int rows, int status)
SPI_freetuptable(tuptable); SPI_freetuptable(tuptable);
} }
RESTORE_EXC(); }
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
PLy_error_in_progress = CopyErrorData();
FlushErrorState();
if (!PyErr_Occurred())
PyErr_SetString(PLy_exc_error,
"Unknown error in PLy_spi_execute_fetch_result");
Py_DECREF(result);
PLy_typeinfo_dealloc(&args);
return NULL;
}
PG_END_TRY();
} }
return (PyObject *) result; return (PyObject *) result;
} }
const char * static const char *
PLy_spi_error_string(int code) PLy_spi_error_string(int code)
{ {
switch (code) switch (code)
...@@ -2384,8 +2256,6 @@ plpython_init(void) ...@@ -2384,8 +2256,6 @@ plpython_init(void)
if (!PLy_first_call) if (!PLy_first_call)
return; return;
enter();
if (init_active) if (init_active)
elog(FATAL, "initialization of language module failed"); elog(FATAL, "initialization of language module failed");
init_active = 1; init_active = 1;
...@@ -2400,8 +2270,6 @@ plpython_init(void) ...@@ -2400,8 +2270,6 @@ plpython_init(void)
PLy_elog(ERROR, "could not create procedure cache"); PLy_elog(ERROR, "could not create procedure cache");
PLy_first_call = 0; PLy_first_call = 0;
leave();
} }
static void static void
...@@ -2418,13 +2286,11 @@ PLy_init_all(void) ...@@ -2418,13 +2286,11 @@ PLy_init_all(void)
} }
void static void
PLy_init_interp(void) PLy_init_interp(void)
{ {
PyObject *mainmod; PyObject *mainmod;
enter();
mainmod = PyImport_AddModule("__main__"); mainmod = PyImport_AddModule("__main__");
if ((mainmod == NULL) || (PyErr_Occurred())) if ((mainmod == NULL) || (PyErr_Occurred()))
PLy_elog(ERROR, "could not import \"__main__\" module."); PLy_elog(ERROR, "could not import \"__main__\" module.");
...@@ -2437,7 +2303,7 @@ PLy_init_interp(void) ...@@ -2437,7 +2303,7 @@ PLy_init_interp(void)
PLy_elog(ERROR, "could not initialize globals"); PLy_elog(ERROR, "could not initialize globals");
} }
void static void
PLy_init_plpy(void) PLy_init_plpy(void)
{ {
PyObject *main_mod, PyObject *main_mod,
...@@ -2446,8 +2312,6 @@ PLy_init_plpy(void) ...@@ -2446,8 +2312,6 @@ PLy_init_plpy(void)
PyObject *plpy, PyObject *plpy,
*plpy_dict; *plpy_dict;
enter();
/* /*
* initialize plpy module * initialize plpy module
*/ */
...@@ -2480,60 +2344,55 @@ PLy_init_plpy(void) ...@@ -2480,60 +2344,55 @@ PLy_init_plpy(void)
*/ */
static PyObject *PLy_output(int, PyObject *, PyObject *); static PyObject *PLy_output(int, PyObject *, PyObject *);
PyObject * static PyObject *
PLy_debug(PyObject * self, PyObject * args) PLy_debug(PyObject * self, PyObject * args)
{ {
return PLy_output(DEBUG2, self, args); return PLy_output(DEBUG2, self, args);
} }
PyObject * static PyObject *
PLy_log(PyObject * self, PyObject * args) PLy_log(PyObject * self, PyObject * args)
{ {
return PLy_output(LOG, self, args); return PLy_output(LOG, self, args);
} }
PyObject * static PyObject *
PLy_info(PyObject * self, PyObject * args) PLy_info(PyObject * self, PyObject * args)
{ {
return PLy_output(INFO, self, args); return PLy_output(INFO, self, args);
} }
PyObject * static PyObject *
PLy_notice(PyObject * self, PyObject * args) PLy_notice(PyObject * self, PyObject * args)
{ {
return PLy_output(NOTICE, self, args); return PLy_output(NOTICE, self, args);
} }
PyObject * static PyObject *
PLy_warning(PyObject * self, PyObject * args) PLy_warning(PyObject * self, PyObject * args)
{ {
return PLy_output(WARNING, self, args); return PLy_output(WARNING, self, args);
} }
PyObject * static PyObject *
PLy_error(PyObject * self, PyObject * args) PLy_error(PyObject * self, PyObject * args)
{ {
return PLy_output(ERROR, self, args); return PLy_output(ERROR, self, args);
} }
PyObject * static PyObject *
PLy_fatal(PyObject * self, PyObject * args) PLy_fatal(PyObject * self, PyObject * args)
{ {
return PLy_output(FATAL, self, args); return PLy_output(FATAL, self, args);
} }
PyObject * static PyObject *
PLy_output(volatile int level, PyObject * self, PyObject * args) PLy_output(volatile int level, PyObject * self, PyObject * args)
{ {
DECLARE_EXC();
PyObject *so; PyObject *so;
char *volatile sv; char *volatile sv;
MemoryContext oldcontext;
enter();
if (args == NULL)
elog(WARNING, "args is NULL");
so = PyObject_Str(args); so = PyObject_Str(args);
if ((so == NULL) || ((sv = PyString_AsString(so)) == NULL)) if ((so == NULL) || ((sv = PyString_AsString(so)) == NULL))
...@@ -2542,54 +2401,34 @@ PLy_output(volatile int level, PyObject * self, PyObject * args) ...@@ -2542,54 +2401,34 @@ PLy_output(volatile int level, PyObject * self, PyObject * args)
sv = "Unable to parse error message in `plpy.elog'"; sv = "Unable to parse error message in `plpy.elog'";
} }
/* oldcontext = CurrentMemoryContext;
* returning NULL here causes the python interpreter to bail. when PG_TRY();
* control passes back into plpython_*_handler, we check for python
* exceptions and do the actual elog call. actually PLy_elog.
*/
if (level == ERROR)
{ {
PyErr_SetString(PLy_exc_error, sv); elog(level, "%s", sv);
return NULL;
} }
else if (level >= FATAL) PG_CATCH();
{ {
PyErr_SetString(PLy_exc_fatal, sv); MemoryContextSwitchTo(oldcontext);
return NULL; PLy_error_in_progress = CopyErrorData();
} FlushErrorState();
/*
* ok, this is a WARNING, or LOG message
*
* but just in case DON'T long jump out of the interpreter!
*/
SAVE_EXC();
if (TRAP_EXC())
{
RESTORE_EXC();
Py_XDECREF(so); Py_XDECREF(so);
/* /*
* the real error message should already be written into the * returning NULL here causes the python interpreter to bail. when
* postgresql log, no? whatever, this shouldn't happen so die * control passes back to PLy_procedure_call, we check for PG
* hideously. * exceptions and re-throw the error.
*/ */
elog(FATAL, "elog threw an unknown exception"); PyErr_SetString(PLy_exc_error, sv);
RERAISE_EXC(); return NULL;
} }
PG_END_TRY();
elog(level, "%s", sv);
RESTORE_EXC();
Py_XDECREF(so); Py_XDECREF(so);
Py_INCREF(Py_None);
/* /*
* return a legal object so the interpreter will continue on its merry * return a legal object so the interpreter will continue on its merry
* way * way
*/ */
Py_INCREF(Py_None);
return Py_None; return Py_None;
} }
...@@ -2602,7 +2441,7 @@ PLy_output(volatile int level, PyObject * self, PyObject * args) ...@@ -2602,7 +2441,7 @@ PLy_output(volatile int level, PyObject * self, PyObject * args)
* NB: this returns SQL name, not the internal Python procedure name * NB: this returns SQL name, not the internal Python procedure name
*/ */
char * static char *
PLy_procedure_name(PLyProcedure * proc) PLy_procedure_name(PLyProcedure * proc)
{ {
if (proc == NULL) if (proc == NULL)
...@@ -2613,12 +2452,7 @@ PLy_procedure_name(PLyProcedure * proc) ...@@ -2613,12 +2452,7 @@ PLy_procedure_name(PLyProcedure * proc)
/* output a python traceback/exception via the postgresql elog /* output a python traceback/exception via the postgresql elog
* function. not pretty. * function. not pretty.
*/ */
static void
static char *PLy_traceback(int *);
static char *PLy_vprintf(const char *fmt, va_list ap);
static char *PLy_printf(const char *fmt,...);
void
PLy_exception_set(PyObject * exc, const char *fmt,...) PLy_exception_set(PyObject * exc, const char *fmt,...)
{ {
char buf[1024]; char buf[1024];
...@@ -2631,53 +2465,45 @@ PLy_exception_set(PyObject * exc, const char *fmt,...) ...@@ -2631,53 +2465,45 @@ PLy_exception_set(PyObject * exc, const char *fmt,...)
PyErr_SetString(exc, buf); PyErr_SetString(exc, buf);
} }
void /* Emit a PG error or notice, together with any available info about the
* current Python error. This should be used to propagate Python errors
* into PG.
*/
static void
PLy_elog(int elevel, const char *fmt,...) PLy_elog(int elevel, const char *fmt,...)
{ {
DECLARE_EXC();
va_list ap; va_list ap;
char *xmsg, char *xmsg,
*emsg; *emsg;
int xlevel; int xlevel;
enter();
xmsg = PLy_traceback(&xlevel); xmsg = PLy_traceback(&xlevel);
va_start(ap, fmt); va_start(ap, fmt);
emsg = PLy_vprintf(fmt, ap); emsg = PLy_vprintf(fmt, ap);
va_end(ap); va_end(ap);
SAVE_EXC(); PG_TRY();
if (TRAP_EXC()) {
ereport(elevel,
(errmsg("plpython: %s", emsg),
(xmsg) ? errdetail("%s", xmsg) : 0));
}
PG_CATCH();
{ {
RESTORE_EXC();
mark();
/*
* elog called siglongjmp. cleanup, restore and reraise
*/
PLy_restart_in_progress += 1;
PLy_free(emsg); PLy_free(emsg);
if (xmsg) if (xmsg)
PLy_free(xmsg); PLy_free(xmsg);
RERAISE_EXC(); PG_RE_THROW();
} }
PG_END_TRY();
ereport(elevel,
(errmsg("plpython: %s", emsg),
(xmsg) ? errdetail("%s", xmsg) : 0));
PLy_free(emsg); PLy_free(emsg);
if (xmsg) if (xmsg)
PLy_free(xmsg); PLy_free(xmsg);
leave();
RESTORE_EXC();
} }
char * static char *
PLy_traceback(int *xlevel) PLy_traceback(int *xlevel)
{ {
PyObject *e, PyObject *e,
...@@ -2689,8 +2515,6 @@ PLy_traceback(int *xlevel) ...@@ -2689,8 +2515,6 @@ PLy_traceback(int *xlevel)
*estr, *estr,
*xstr = NULL; *xstr = NULL;
enter();
/* /*
* get the current exception * get the current exception
*/ */
...@@ -2729,12 +2553,10 @@ PLy_traceback(int *xlevel) ...@@ -2729,12 +2553,10 @@ PLy_traceback(int *xlevel)
else else
*xlevel = ERROR; *xlevel = ERROR;
leave();
return xstr; return xstr;
} }
char * static char *
PLy_printf(const char *fmt,...) PLy_printf(const char *fmt,...)
{ {
va_list ap; va_list ap;
...@@ -2746,7 +2568,7 @@ PLy_printf(const char *fmt,...) ...@@ -2746,7 +2568,7 @@ PLy_printf(const char *fmt,...)
return emsg; return emsg;
} }
char * static char *
PLy_vprintf(const char *fmt, va_list ap) PLy_vprintf(const char *fmt, va_list ap)
{ {
size_t blen; size_t blen;
...@@ -2783,7 +2605,7 @@ PLy_vprintf(const char *fmt, va_list ap) ...@@ -2783,7 +2605,7 @@ PLy_vprintf(const char *fmt, va_list ap)
/* some dumb utility functions /* some dumb utility functions
*/ */
void * static void *
PLy_malloc(size_t bytes) PLy_malloc(size_t bytes)
{ {
void *ptr = malloc(bytes); void *ptr = malloc(bytes);
...@@ -2795,7 +2617,7 @@ PLy_malloc(size_t bytes) ...@@ -2795,7 +2617,7 @@ PLy_malloc(size_t bytes)
return ptr; return ptr;
} }
void * static void *
PLy_realloc(void *optr, size_t bytes) PLy_realloc(void *optr, size_t bytes)
{ {
void *nptr = realloc(optr, bytes); void *nptr = realloc(optr, bytes);
...@@ -2809,7 +2631,7 @@ PLy_realloc(void *optr, size_t bytes) ...@@ -2809,7 +2631,7 @@ PLy_realloc(void *optr, size_t bytes)
/* define this away /* define this away
*/ */
void static void
PLy_free(void *ptr) PLy_free(void *ptr)
{ {
free(ptr); free(ptr);
......
/* $PostgreSQL: pgsql/src/pl/plpython/plpython.h,v 1.9 2003/11/29 19:52:13 pgsql Exp $ */
#ifndef PLPYTHON_H
#define PLPYTHON_H
#define DEBUG_EXC 0
#define DEBUG_LEVEL 0
#define DECLARE_N_EXC(N) int rv_##N; sigjmp_buf buf_##N
#define TRAP_N_EXC(N) ((rv_##N = sigsetjmp(Warn_restart, 1)) != 0)
#if !DEBUG_EXC
#define RESTORE_N_EXC(N) memcpy(&Warn_restart, &(buf_##N), sizeof(sigjmp_buf))
#define SAVE_N_EXC(N) memcpy(&(buf_##N), &Warn_restart, sizeof(sigjmp_buf))
#define RERAISE_N_EXC(N) siglongjmp(Warn_restart, rv_##N)
#define RAISE_EXC(V) siglongjmp(Warn_restart, (V))
#else
#define RESTORE_N_EXC(N) do { \
elog(WARNING, "exception (%d,%d) restore at %s:%d",\
PLy_call_level, exc_save_calls, __FUNCTION__, (__LINE__));\
exc_save_calls -= 1; \
memcpy(&Warn_restart, &(buf_##N), sizeof(sigjmp_buf)); } while (0)
#define SAVE_N_EXC(N) do { \
exc_save_calls += 1; \
elog(WARNING, "exception (%d,%d) save at %s:%d", \
PLy_call_level, exc_save_calls, __FUNCTION__, (__LINE__)); \
memcpy(&(buf_##N), &Warn_restart, sizeof(sigjmp_buf)); } while (0)
#define RERAISE_N_EXC(N) do { \
elog(WARNING, "exception (%d,%d) reraise at %s:%d", \
PLy_call_level, exc_save_calls, __FUNCTION__, (__LINE__)); \
siglongjmp(Warn_restart, rv_##N); } while (0)
#define RAISE_EXC(V) do { \
elog(WARNING, "exception (%d,%d) raise at %s:%d", \
PLy_call_level, exc_save_calls, __FUNCTION__, (__LINE__)); \
siglongjmp(Warn_restart, (V)); } while (0)
#endif
#define DECLARE_EXC() DECLARE_N_EXC(save_restart)
#define SAVE_EXC() SAVE_N_EXC(save_restart)
#define RERAISE_EXC() RERAISE_N_EXC(save_restart)
#define RESTORE_EXC() RESTORE_N_EXC(save_restart)
#define TRAP_EXC() TRAP_N_EXC(save_restart)
#if DEBUG_LEVEL
#define CALL_LEVEL_INC() do { PLy_call_level += 1; \
elog(DEBUG4, "level: %d", PLy_call_level); } while (0)
#define CALL_LEVEL_DEC() do { elog(DEBUG4, "level: %d", PLy_call_level); \
PLy_call_level -= 1; } while (0)
#else
#define CALL_LEVEL_INC() do { PLy_call_level += 1; } while (0)
#define CALL_LEVEL_DEC() do { PLy_call_level -= 1; } while (0)
#endif
/* temporary debugging macros
*/
#if DEBUG_LEVEL
#define enter() elog(DEBUG4, "Enter(%d): %s", func_enter_calls++,__FUNCTION__)
#define leave() elog(DEBUG4, "Leave(%d): %s", func_leave_calls++,__FUNCTION__)
#define mark() elog(DEBUG4, "Mark: %s:%d", __FUNCTION__, __LINE__);
#define refc(O) elog(DEBUG4, "Ref<%p>:<%d>:%s:%d", (O), (((O) == NULL) ? -1 : (O)->ob_refcnt), __FUNCTION__, __LINE__)
#else
#define enter()
#define leave()
#define mark()
#define refc(O)
#endif
#endif /* PLPYTHON_H */
...@@ -31,7 +31,7 @@ ...@@ -31,7 +31,7 @@
* ENHANCEMENTS, OR MODIFICATIONS. * ENHANCEMENTS, OR MODIFICATIONS.
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/pl/tcl/pltcl.c,v 1.86 2004/06/06 00:41:28 tgl Exp $ * $PostgreSQL: pgsql/src/pl/tcl/pltcl.c,v 1.87 2004/07/31 00:45:57 tgl Exp $
* *
**********************************************************************/ **********************************************************************/
...@@ -41,7 +41,6 @@ ...@@ -41,7 +41,6 @@
#include <unistd.h> #include <unistd.h>
#include <fcntl.h> #include <fcntl.h>
#include <setjmp.h>
/* Hack to deal with Tcl 8.4 const-ification without losing compatibility */ /* Hack to deal with Tcl 8.4 const-ification without losing compatibility */
#ifndef CONST84 #ifndef CONST84
...@@ -132,8 +131,6 @@ typedef struct pltcl_query_desc ...@@ -132,8 +131,6 @@ typedef struct pltcl_query_desc
**********************************************************************/ **********************************************************************/
static bool pltcl_pm_init_done = false; static bool pltcl_pm_init_done = false;
static bool pltcl_be_init_done = false; static bool pltcl_be_init_done = false;
static int pltcl_call_level = 0;
static int pltcl_restart_in_progress = 0;
static Tcl_Interp *pltcl_hold_interp = NULL; static Tcl_Interp *pltcl_hold_interp = NULL;
static Tcl_Interp *pltcl_norm_interp = NULL; static Tcl_Interp *pltcl_norm_interp = NULL;
static Tcl_Interp *pltcl_safe_interp = NULL; static Tcl_Interp *pltcl_safe_interp = NULL;
...@@ -142,6 +139,19 @@ static Tcl_HashTable *pltcl_norm_query_hash = NULL; ...@@ -142,6 +139,19 @@ static Tcl_HashTable *pltcl_norm_query_hash = NULL;
static Tcl_HashTable *pltcl_safe_query_hash = NULL; static Tcl_HashTable *pltcl_safe_query_hash = NULL;
static FunctionCallInfo pltcl_current_fcinfo = NULL; static FunctionCallInfo pltcl_current_fcinfo = NULL;
/*
* When a callback from Tcl into PG incurs an error, we temporarily store
* the error information here, and return TCL_ERROR to the Tcl interpreter.
* Any further callback attempts immediately fail, and when the Tcl interpreter
* returns to the calling function, we re-throw the error (even if Tcl
* thinks it trapped the error and doesn't return TCL_ERROR). Eventually
* this ought to be improved to let Tcl code really truly trap the error,
* but that's more of a change from the pre-7.5 semantics than I have time
* for now --- it will only be possible if the callback query is executed
* inside a subtransaction.
*/
static ErrorData *pltcl_error_in_progress = NULL;
/********************************************************************** /**********************************************************************
* Forward declarations * Forward declarations
**********************************************************************/ **********************************************************************/
...@@ -397,20 +407,10 @@ pltcl_call_handler(PG_FUNCTION_ARGS) ...@@ -397,20 +407,10 @@ pltcl_call_handler(PG_FUNCTION_ARGS)
FunctionCallInfo save_fcinfo; FunctionCallInfo save_fcinfo;
/************************************************************ /************************************************************
* Initialize interpreters * Initialize interpreters if first time through
************************************************************/ ************************************************************/
pltcl_init_all(); pltcl_init_all();
/************************************************************
* Connect to SPI manager
************************************************************/
if (SPI_connect() != SPI_OK_CONNECT)
elog(ERROR, "could not connect to SPI manager");
/************************************************************
* Keep track about the nesting of Tcl-SPI-Tcl-... calls
************************************************************/
pltcl_call_level++;
/************************************************************ /************************************************************
* Determine if called as function or trigger and * Determine if called as function or trigger and
* call appropriate subhandler * call appropriate subhandler
...@@ -430,8 +430,6 @@ pltcl_call_handler(PG_FUNCTION_ARGS) ...@@ -430,8 +430,6 @@ pltcl_call_handler(PG_FUNCTION_ARGS)
pltcl_current_fcinfo = save_fcinfo; pltcl_current_fcinfo = save_fcinfo;
pltcl_call_level--;
return retval; return retval;
} }
...@@ -461,7 +459,10 @@ pltcl_func_handler(PG_FUNCTION_ARGS) ...@@ -461,7 +459,10 @@ pltcl_func_handler(PG_FUNCTION_ARGS)
int i; int i;
int tcl_rc; int tcl_rc;
Datum retval; Datum retval;
sigjmp_buf save_restart;
/* Connect to SPI manager */
if (SPI_connect() != SPI_OK_CONNECT)
elog(ERROR, "could not connect to SPI manager");
/* Find or compile the function */ /* Find or compile the function */
prodesc = compile_pltcl_function(fcinfo->flinfo->fn_oid, InvalidOid); prodesc = compile_pltcl_function(fcinfo->flinfo->fn_oid, InvalidOid);
...@@ -479,24 +480,11 @@ pltcl_func_handler(PG_FUNCTION_ARGS) ...@@ -479,24 +480,11 @@ pltcl_func_handler(PG_FUNCTION_ARGS)
Tcl_DStringInit(&list_tmp); Tcl_DStringInit(&list_tmp);
Tcl_DStringAppendElement(&tcl_cmd, prodesc->proname); Tcl_DStringAppendElement(&tcl_cmd, prodesc->proname);
/************************************************************
* Catch elog(ERROR) during build of the Tcl command
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
Tcl_DStringFree(&tcl_cmd);
Tcl_DStringFree(&list_tmp);
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
/************************************************************ /************************************************************
* Add all call arguments to the command * Add all call arguments to the command
************************************************************/ ************************************************************/
PG_TRY();
{
for (i = 0; i < prodesc->nargs; i++) for (i = 0; i < prodesc->nargs; i++)
{ {
if (prodesc->arg_is_rowtype[i]) if (prodesc->arg_is_rowtype[i])
...@@ -552,28 +540,41 @@ pltcl_func_handler(PG_FUNCTION_ARGS) ...@@ -552,28 +540,41 @@ pltcl_func_handler(PG_FUNCTION_ARGS)
} }
} }
} }
}
PG_CATCH();
{
Tcl_DStringFree(&tcl_cmd);
Tcl_DStringFree(&list_tmp);
PG_RE_THROW();
}
PG_END_TRY();
Tcl_DStringFree(&list_tmp); Tcl_DStringFree(&list_tmp);
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
/************************************************************ /************************************************************
* Call the Tcl function * Call the Tcl function
*
* We assume no PG error can be thrown directly from this call.
************************************************************/ ************************************************************/
tcl_rc = Tcl_GlobalEval(interp, Tcl_DStringValue(&tcl_cmd)); tcl_rc = Tcl_GlobalEval(interp, Tcl_DStringValue(&tcl_cmd));
Tcl_DStringFree(&tcl_cmd); Tcl_DStringFree(&tcl_cmd);
/************************************************************ /************************************************************
* Check the return code from Tcl and handle * If there was an error in a PG callback, propagate that
* our special restart mechanism to get rid * no matter what Tcl claims about its success.
* of all nested call levels on transaction
* abort.
************************************************************/ ************************************************************/
if (tcl_rc != TCL_OK || pltcl_restart_in_progress) if (pltcl_error_in_progress)
{ {
if (!pltcl_restart_in_progress) ErrorData *edata = pltcl_error_in_progress;
pltcl_error_in_progress = NULL;
ReThrowError(edata);
}
/************************************************************
* Check for errors reported by Tcl itself.
************************************************************/
if (tcl_rc != TCL_OK)
{ {
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
UTF_BEGIN; UTF_BEGIN;
ereport(ERROR, ereport(ERROR,
(errmsg("pltcl: %s", interp->result), (errmsg("pltcl: %s", interp->result),
...@@ -582,30 +583,10 @@ pltcl_func_handler(PG_FUNCTION_ARGS) ...@@ -582,30 +583,10 @@ pltcl_func_handler(PG_FUNCTION_ARGS)
TCL_GLOBAL_ONLY))))); TCL_GLOBAL_ONLY)))));
UTF_END; UTF_END;
} }
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
/************************************************************
* Convert the result value from the Tcl interpreter
* into its PostgreSQL data format and return it.
* Again, the function call could fire an elog and we
* have to count for the current interpreter level we are
* on. The save_restart from above is still good.
************************************************************/
if (sigsetjmp(Warn_restart, 1) != 0)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
/************************************************************ /************************************************************
* Disconnect from SPI manager and then create the return * Disconnect from SPI manager and then create the return
* values datum (if the input function does a palloc for it * value datum (if the input function does a palloc for it
* this must not be allocated in the SPI memory context * this must not be allocated in the SPI memory context
* because SPI_finish would free it). But don't try to call * because SPI_finish would free it). But don't try to call
* the result_in_func if we've been told to return a NULL; * the result_in_func if we've been told to return a NULL;
...@@ -627,11 +608,6 @@ pltcl_func_handler(PG_FUNCTION_ARGS) ...@@ -627,11 +608,6 @@ pltcl_func_handler(PG_FUNCTION_ARGS)
UTF_END; UTF_END;
} }
/************************************************************
* Finally we may restore normal error handling.
************************************************************/
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
return retval; return retval;
} }
...@@ -653,15 +629,15 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -653,15 +629,15 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
Tcl_DString tcl_newtup; Tcl_DString tcl_newtup;
int tcl_rc; int tcl_rc;
int i; int i;
int *modattrs; int *modattrs;
Datum *modvalues; Datum *modvalues;
char *modnulls; char *modnulls;
int ret_numvals; int ret_numvals;
CONST84 char **ret_values; CONST84 char **ret_values;
sigjmp_buf save_restart; /* Connect to SPI manager */
if (SPI_connect() != SPI_OK_CONNECT)
elog(ERROR, "could not connect to SPI manager");
/* Find or compile the function */ /* Find or compile the function */
prodesc = compile_pltcl_function(fcinfo->flinfo->fn_oid, prodesc = compile_pltcl_function(fcinfo->flinfo->fn_oid,
...@@ -681,23 +657,8 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -681,23 +657,8 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
Tcl_DStringInit(&tcl_cmd); Tcl_DStringInit(&tcl_cmd);
Tcl_DStringInit(&tcl_trigtup); Tcl_DStringInit(&tcl_trigtup);
Tcl_DStringInit(&tcl_newtup); Tcl_DStringInit(&tcl_newtup);
PG_TRY();
/************************************************************
* We call external functions below - care for elog(ERROR)
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
Tcl_DStringFree(&tcl_cmd);
Tcl_DStringFree(&tcl_trigtup);
Tcl_DStringFree(&tcl_newtup);
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
/* The procedure name */ /* The procedure name */
Tcl_DStringAppendElement(&tcl_cmd, prodesc->proname); Tcl_DStringAppendElement(&tcl_cmd, prodesc->proname);
...@@ -799,35 +760,47 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -799,35 +760,47 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
else else
elog(ERROR, "unrecognized LEVEL tg_event: %u", trigdata->tg_event); elog(ERROR, "unrecognized LEVEL tg_event: %u", trigdata->tg_event);
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); /* Finally append the arguments from CREATE TRIGGER */
Tcl_DStringFree(&tcl_trigtup);
Tcl_DStringFree(&tcl_newtup);
/************************************************************
* Finally append the arguments from CREATE TRIGGER
************************************************************/
for (i = 0; i < trigdata->tg_trigger->tgnargs; i++) for (i = 0; i < trigdata->tg_trigger->tgnargs; i++)
Tcl_DStringAppendElement(&tcl_cmd, trigdata->tg_trigger->tgargs[i]); Tcl_DStringAppendElement(&tcl_cmd, trigdata->tg_trigger->tgargs[i]);
}
PG_CATCH();
{
Tcl_DStringFree(&tcl_cmd);
Tcl_DStringFree(&tcl_trigtup);
Tcl_DStringFree(&tcl_newtup);
PG_RE_THROW();
}
PG_END_TRY();
Tcl_DStringFree(&tcl_trigtup);
Tcl_DStringFree(&tcl_newtup);
/************************************************************ /************************************************************
* Call the Tcl function * Call the Tcl function
*
* We assume no PG error can be thrown directly from this call.
************************************************************/ ************************************************************/
tcl_rc = Tcl_GlobalEval(interp, Tcl_DStringValue(&tcl_cmd)); tcl_rc = Tcl_GlobalEval(interp, Tcl_DStringValue(&tcl_cmd));
Tcl_DStringFree(&tcl_cmd); Tcl_DStringFree(&tcl_cmd);
/************************************************************ /************************************************************
* Check the return code from Tcl and handle * If there was an error in a PG callback, propagate that
* our special restart mechanism to get rid * no matter what Tcl claims about its success.
* of all nested call levels on transaction
* abort.
************************************************************/ ************************************************************/
if (tcl_rc == TCL_ERROR || pltcl_restart_in_progress) if (pltcl_error_in_progress)
{ {
if (!pltcl_restart_in_progress) ErrorData *edata = pltcl_error_in_progress;
pltcl_error_in_progress = NULL;
ReThrowError(edata);
}
/************************************************************
* Check for errors reported by Tcl itself.
************************************************************/
if (tcl_rc != TCL_OK)
{ {
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
UTF_BEGIN; UTF_BEGIN;
ereport(ERROR, ereport(ERROR,
(errmsg("pltcl: %s", interp->result), (errmsg("pltcl: %s", interp->result),
...@@ -836,19 +809,6 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -836,19 +809,6 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
TCL_GLOBAL_ONLY))))); TCL_GLOBAL_ONLY)))));
UTF_END; UTF_END;
} }
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
switch (tcl_rc)
{
case TCL_OK:
break;
default:
elog(ERROR, "unsupported TCL return code: %d", tcl_rc);
}
/************************************************************ /************************************************************
* The return value from the procedure might be one of * The return value from the procedure might be one of
...@@ -871,11 +831,12 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -871,11 +831,12 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
elog(ERROR, "could not split return value from trigger: %s", elog(ERROR, "could not split return value from trigger: %s",
interp->result); interp->result);
if (ret_numvals % 2 != 0) /* Use a TRY to ensure ret_values will get freed */
PG_TRY();
{ {
ckfree((char *) ret_values);
if (ret_numvals % 2 != 0)
elog(ERROR, "invalid return list from trigger - must have even # of elements"); elog(ERROR, "invalid return list from trigger - must have even # of elements");
}
modattrs = (int *) palloc(tupdesc->natts * sizeof(int)); modattrs = (int *) palloc(tupdesc->natts * sizeof(int));
modvalues = (Datum *) palloc(tupdesc->natts * sizeof(Datum)); modvalues = (Datum *) palloc(tupdesc->natts * sizeof(Datum));
...@@ -888,19 +849,6 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -888,19 +849,6 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
modnulls = palloc(tupdesc->natts); modnulls = palloc(tupdesc->natts);
memset(modnulls, 'n', tupdesc->natts); memset(modnulls, 'n', tupdesc->natts);
/************************************************************
* Care for possible elog(ERROR)'s below
************************************************************/
if (sigsetjmp(Warn_restart, 1) != 0)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
ckfree((char *) ret_values);
pltcl_restart_in_progress = 1;
if (--pltcl_call_level == 0)
pltcl_restart_in_progress = 0;
siglongjmp(Warn_restart, 1);
}
for (i = 0; i < ret_numvals; i += 2) for (i = 0; i < ret_numvals; i += 2)
{ {
CONST84 char *ret_name = ret_values[i]; CONST84 char *ret_name = ret_values[i];
...@@ -970,8 +918,14 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS) ...@@ -970,8 +918,14 @@ pltcl_trigger_handler(PG_FUNCTION_ARGS)
if (rettup == NULL) if (rettup == NULL)
elog(ERROR, "SPI_modifytuple() failed - RC = %d", SPI_result); elog(ERROR, "SPI_modifytuple() failed - RC = %d", SPI_result);
}
PG_CATCH();
{
ckfree((char *) ret_values);
PG_RE_THROW();
}
PG_END_TRY();
ckfree((char *) ret_values); ckfree((char *) ret_values);
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
return rettup; return rettup;
} }
...@@ -1317,13 +1271,16 @@ pltcl_elog(ClientData cdata, Tcl_Interp *interp, ...@@ -1317,13 +1271,16 @@ pltcl_elog(ClientData cdata, Tcl_Interp *interp,
int argc, CONST84 char *argv[]) int argc, CONST84 char *argv[])
{ {
volatile int level; volatile int level;
sigjmp_buf save_restart; MemoryContext oldcontext;
/************************************************************ /************************************************************
* Suppress messages during the restart process * Suppress messages if an error is already declared
************************************************************/ ************************************************************/
if (pltcl_restart_in_progress) if (pltcl_error_in_progress)
{
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
}
if (argc != 3) if (argc != 3)
{ {
...@@ -1354,26 +1311,25 @@ pltcl_elog(ClientData cdata, Tcl_Interp *interp, ...@@ -1354,26 +1311,25 @@ pltcl_elog(ClientData cdata, Tcl_Interp *interp,
} }
/************************************************************ /************************************************************
* Catch the longjmp from elog() and begin a controlled * If elog() throws an error, catch and save it, then return
* return though all interpreter levels if it happens * error indication to Tcl interpreter.
************************************************************/ ************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart)); oldcontext = CurrentMemoryContext;
if (sigsetjmp(Warn_restart, 1) != 0) PG_TRY();
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
return TCL_ERROR;
}
/************************************************************
* Call elog(), restore the original restart address
* and return to the caller (if no longjmp)
************************************************************/
UTF_BEGIN; UTF_BEGIN;
elog(level, "%s", UTF_U2E(argv[2])); elog(level, "%s", UTF_U2E(argv[2]));
UTF_END; UTF_END;
}
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
return TCL_ERROR;
}
PG_END_TRY();
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
return TCL_OK; return TCL_OK;
} }
...@@ -1535,6 +1491,7 @@ static int ...@@ -1535,6 +1491,7 @@ static int
pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
int argc, CONST84 char *argv[]) int argc, CONST84 char *argv[])
{ {
volatile int my_rc;
int spi_rc; int spi_rc;
char buf[64]; char buf[64];
int count = 0; int count = 0;
...@@ -1546,17 +1503,20 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, ...@@ -1546,17 +1503,20 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
HeapTuple *volatile tuples; HeapTuple *volatile tuples;
volatile TupleDesc tupdesc = NULL; volatile TupleDesc tupdesc = NULL;
SPITupleTable *tuptable; SPITupleTable *tuptable;
sigjmp_buf save_restart; MemoryContext oldcontext;
char *usage = "syntax error - 'SPI_exec " char *usage = "syntax error - 'SPI_exec "
"?-count n? " "?-count n? "
"?-array name? query ?loop body?"; "?-array name? query ?loop body?";
/************************************************************ /************************************************************
* Don't do anything if we are already in restart mode * Don't do anything if we are already in error mode
************************************************************/ ************************************************************/
if (pltcl_restart_in_progress) if (pltcl_error_in_progress)
{
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
}
/************************************************************ /************************************************************
* Check the call syntax and get the count option * Check the call syntax and get the count option
...@@ -1603,26 +1563,25 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, ...@@ -1603,26 +1563,25 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
return TCL_ERROR; return TCL_ERROR;
} }
/************************************************************
* Prepare to start a controlled return through all
* interpreter levels on transaction abort
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
Tcl_SetResult(interp, "Transaction abort", TCL_VOLATILE);
return TCL_ERROR;
}
/************************************************************ /************************************************************
* Execute the query and handle return codes * Execute the query and handle return codes
************************************************************/ ************************************************************/
oldcontext = CurrentMemoryContext;
PG_TRY();
{
UTF_BEGIN; UTF_BEGIN;
spi_rc = SPI_exec(UTF_U2E(argv[query_idx]), count); spi_rc = SPI_exec(UTF_U2E(argv[query_idx]), count);
UTF_END; UTF_END;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); }
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR;
}
PG_END_TRY();
switch (spi_rc) switch (spi_rc)
{ {
...@@ -1687,46 +1646,31 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, ...@@ -1687,46 +1646,31 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
} }
/************************************************************ /************************************************************
* Only SELECT queries fall through to here - remember the * Only SELECT queries fall through to here - process the tuples we got
* tuples we got
************************************************************/ ************************************************************/
ntuples = SPI_processed; ntuples = SPI_processed;
tuptable = SPI_tuptable;
if (ntuples > 0) if (ntuples > 0)
{ {
tuples = SPI_tuptable->vals; tuples = tuptable->vals;
tupdesc = SPI_tuptable->tupdesc; tupdesc = tuptable->tupdesc;
} }
/************************************************************ my_rc = TCL_OK;
* Again prepare for elog(ERROR) PG_TRY();
************************************************************/ {
if (sigsetjmp(Warn_restart, 1) != 0) if (argc == query_idx + 1)
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
Tcl_SetResult(interp, "Transaction abort", TCL_VOLATILE);
return TCL_ERROR;
}
/************************************************************ /************************************************************
* If there is no loop body given, just set the variables * If there is no loop body given, just set the variables
* from the first tuple (if any) and return the number of * from the first tuple (if any)
* tuples selected
************************************************************/ ************************************************************/
if (argc == query_idx + 1)
{
if (ntuples > 0) if (ntuples > 0)
pltcl_set_tuple_values(interp, arrayname, 0, tuples[0], tupdesc); pltcl_set_tuple_values(interp, arrayname, 0,
snprintf(buf, sizeof(buf), "%d", ntuples); tuples[0], tupdesc);
Tcl_SetResult(interp, buf, TCL_VOLATILE);
SPI_freetuptable(SPI_tuptable);
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
return TCL_OK;
} }
else
tuptable = SPI_tuptable; {
/************************************************************ /************************************************************
* There is a loop body - process all tuples and evaluate * There is a loop body - process all tuples and evaluate
* the body on each * the body on each
...@@ -1734,7 +1678,8 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, ...@@ -1734,7 +1678,8 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
query_idx++; query_idx++;
for (i = 0; i < ntuples; i++) for (i = 0; i < ntuples; i++)
{ {
pltcl_set_tuple_values(interp, arrayname, i, tuples[i], tupdesc); pltcl_set_tuple_values(interp, arrayname, i,
tuples[i], tupdesc);
loop_rc = Tcl_Eval(interp, argv[query_idx]); loop_rc = Tcl_Eval(interp, argv[query_idx]);
...@@ -1744,26 +1689,37 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp, ...@@ -1744,26 +1689,37 @@ pltcl_SPI_exec(ClientData cdata, Tcl_Interp *interp,
continue; continue;
if (loop_rc == TCL_RETURN) if (loop_rc == TCL_RETURN)
{ {
SPI_freetuptable(tuptable); my_rc = TCL_RETURN;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); break;
return TCL_RETURN;
} }
if (loop_rc == TCL_BREAK) if (loop_rc == TCL_BREAK)
break; break;
SPI_freetuptable(tuptable); my_rc = TCL_ERROR;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); break;
return TCL_ERROR; }
} }
SPI_freetuptable(tuptable); SPI_freetuptable(tuptable);
}
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR;
}
PG_END_TRY();
/************************************************************ /************************************************************
* Finally return the number of tuples * Finally return the number of tuples
************************************************************/ ************************************************************/
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); if (my_rc == TCL_OK)
{
snprintf(buf, sizeof(buf), "%d", ntuples); snprintf(buf, sizeof(buf), "%d", ntuples);
Tcl_SetResult(interp, buf, TCL_VOLATILE); Tcl_SetResult(interp, buf, TCL_VOLATILE);
return TCL_OK; }
return my_rc;
} }
...@@ -1787,14 +1743,17 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp, ...@@ -1787,14 +1743,17 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp,
HeapTuple typeTup; HeapTuple typeTup;
Tcl_HashEntry *hashent; Tcl_HashEntry *hashent;
int hashnew; int hashnew;
sigjmp_buf save_restart;
Tcl_HashTable *query_hash; Tcl_HashTable *query_hash;
MemoryContext oldcontext;
/************************************************************ /************************************************************
* Don't do anything if we are already in restart mode * Don't do anything if we are already in error mode
************************************************************/ ************************************************************/
if (pltcl_restart_in_progress) if (pltcl_error_in_progress)
{
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
}
/************************************************************ /************************************************************
* Check the call syntax * Check the call syntax
...@@ -1822,23 +1781,9 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp, ...@@ -1822,23 +1781,9 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp,
qdesc->arginfuncs = (FmgrInfo *) malloc(nargs * sizeof(FmgrInfo)); qdesc->arginfuncs = (FmgrInfo *) malloc(nargs * sizeof(FmgrInfo));
qdesc->argtypioparams = (Oid *) malloc(nargs * sizeof(Oid)); qdesc->argtypioparams = (Oid *) malloc(nargs * sizeof(Oid));
/************************************************************ oldcontext = CurrentMemoryContext;
* Prepare to start a controlled return through all PG_TRY();
* interpreter levels on transaction abort
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
free(qdesc->argtypes);
free(qdesc->arginfuncs);
free(qdesc->argtypioparams);
free(qdesc);
ckfree((char *) args);
return TCL_ERROR;
}
/************************************************************ /************************************************************
* Lookup the argument types by name in the system cache * Lookup the argument types by name in the system cache
* and remember the required information for input conversion * and remember the required information for input conversion
...@@ -1882,10 +1827,7 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp, ...@@ -1882,10 +1827,7 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp,
UTF_END; UTF_END;
if (plan == NULL) if (plan == NULL)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
elog(ERROR, "SPI_prepare() failed"); elog(ERROR, "SPI_prepare() failed");
}
/************************************************************ /************************************************************
* Save the plan into permanent memory (right now it's in the * Save the plan into permanent memory (right now it's in the
...@@ -1893,10 +1835,8 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp, ...@@ -1893,10 +1835,8 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp,
************************************************************/ ************************************************************/
qdesc->plan = SPI_saveplan(plan); qdesc->plan = SPI_saveplan(plan);
if (qdesc->plan == NULL) if (qdesc->plan == NULL)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
elog(ERROR, "SPI_saveplan() failed"); elog(ERROR, "SPI_saveplan() failed");
}
/* Release the procCxt copy to avoid within-function memory leak */ /* Release the procCxt copy to avoid within-function memory leak */
SPI_freeplan(plan); SPI_freeplan(plan);
...@@ -1909,7 +1849,21 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp, ...@@ -1909,7 +1849,21 @@ pltcl_SPI_prepare(ClientData cdata, Tcl_Interp *interp,
else else
query_hash = pltcl_safe_query_hash; query_hash = pltcl_safe_query_hash;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); }
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
free(qdesc->argtypes);
free(qdesc->arginfuncs);
free(qdesc->argtypioparams);
free(qdesc);
ckfree((char *) args);
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR;
}
PG_END_TRY();
hashent = Tcl_CreateHashEntry(query_hash, qdesc->qname, &hashnew); hashent = Tcl_CreateHashEntry(query_hash, qdesc->qname, &hashnew);
Tcl_SetHashValue(hashent, (ClientData) qdesc); Tcl_SetHashValue(hashent, (ClientData) qdesc);
...@@ -1928,6 +1882,7 @@ static int ...@@ -1928,6 +1882,7 @@ static int
pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
int argc, CONST84 char *argv[]) int argc, CONST84 char *argv[])
{ {
volatile int my_rc;
int spi_rc; int spi_rc;
char buf[64]; char buf[64];
volatile int i; volatile int i;
...@@ -1940,13 +1895,13 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -1940,13 +1895,13 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
CONST84 char *volatile arrayname = NULL; CONST84 char *volatile arrayname = NULL;
int count = 0; int count = 0;
int callnargs; int callnargs;
static CONST84 char **callargs = NULL; CONST84 char **callargs;
int loop_rc; int loop_rc;
int ntuples; int ntuples;
HeapTuple *volatile tuples = NULL; HeapTuple *volatile tuples = NULL;
volatile TupleDesc tupdesc = NULL; volatile TupleDesc tupdesc = NULL;
SPITupleTable *tuptable; SPITupleTable *tuptable;
sigjmp_buf save_restart; volatile MemoryContext oldcontext;
Tcl_HashTable *query_hash; Tcl_HashTable *query_hash;
char *usage = "syntax error - 'SPI_execp " char *usage = "syntax error - 'SPI_execp "
...@@ -1954,19 +1909,13 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -1954,19 +1909,13 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
"?-array name? query ?args? ?loop body?"; "?-array name? query ?args? ?loop body?";
/************************************************************ /************************************************************
* Tidy up from an earlier abort * Don't do anything if we are already in error mode
************************************************************/ ************************************************************/
if (callargs != NULL) if (pltcl_error_in_progress)
{ {
ckfree((char *) callargs); Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
callargs = NULL;
}
/************************************************************
* Don't do anything if we are already in restart mode
************************************************************/
if (pltcl_restart_in_progress)
return TCL_ERROR; return TCL_ERROR;
}
/************************************************************ /************************************************************
* Get the options and check syntax * Get the options and check syntax
...@@ -2074,27 +2023,7 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2074,27 +2023,7 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
Tcl_SetResult(interp, Tcl_SetResult(interp,
"argument list length doesn't match # of arguments for query", "argument list length doesn't match # of arguments for query",
TCL_VOLATILE); TCL_VOLATILE);
if (callargs != NULL)
{
ckfree((char *) callargs);
callargs = NULL;
}
return TCL_ERROR;
}
/************************************************************
* Prepare to start a controlled return through all
* interpreter levels on transaction abort during the
* parse of the arguments
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
ckfree((char *) callargs); ckfree((char *) callargs);
callargs = NULL;
pltcl_restart_in_progress = 1;
Tcl_SetResult(interp, "Transaction abort", TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
} }
...@@ -2102,6 +2031,9 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2102,6 +2031,9 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
* Setup the value array for the SPI_execp() using * Setup the value array for the SPI_execp() using
* the type specific input functions * the type specific input functions
************************************************************/ ************************************************************/
oldcontext = CurrentMemoryContext;
PG_TRY();
{
argvalues = (Datum *) palloc(callnargs * sizeof(Datum)); argvalues = (Datum *) palloc(callnargs * sizeof(Datum));
for (j = 0; j < callnargs; j++) for (j = 0; j < callnargs; j++)
...@@ -2122,13 +2054,19 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2122,13 +2054,19 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
UTF_END; UTF_END;
} }
} }
}
PG_CATCH();
{
ckfree((char *) callargs);
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR;
}
PG_END_TRY();
/************************************************************
* Free the splitted argument value list
************************************************************/
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
ckfree((char *) callargs); ckfree((char *) callargs);
callargs = NULL;
} }
else else
callnargs = 0; callnargs = 0;
...@@ -2140,23 +2078,22 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2140,23 +2078,22 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
loop_body = i; loop_body = i;
/************************************************************ /************************************************************
* Prepare to start a controlled return through all * Execute the plan
* interpreter levels on transaction abort
************************************************************/ ************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart)); oldcontext = CurrentMemoryContext;
if (sigsetjmp(Warn_restart, 1) != 0) PG_TRY();
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); spi_rc = SPI_execp(qdesc->plan, argvalues, nulls, count);
pltcl_restart_in_progress = 1; }
Tcl_SetResult(interp, "Transaction abort", TCL_VOLATILE); PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR; return TCL_ERROR;
} }
PG_END_TRY();
/************************************************************
* Execute the plan
************************************************************/
spi_rc = SPI_execp(qdesc->plan, argvalues, nulls, count);
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
/************************************************************ /************************************************************
* Check the return code from SPI_execp() * Check the return code from SPI_execp()
...@@ -2224,56 +2161,39 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2224,56 +2161,39 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
} }
/************************************************************ /************************************************************
* Only SELECT queries fall through to here - remember the * Only SELECT queries fall through to here - process the tuples we got
* tuples we got
************************************************************/ ************************************************************/
ntuples = SPI_processed; ntuples = SPI_processed;
tuptable = SPI_tuptable;
if (ntuples > 0) if (ntuples > 0)
{ {
tuples = SPI_tuptable->vals; tuples = tuptable->vals;
tupdesc = SPI_tuptable->tupdesc; tupdesc = tuptable->tupdesc;
} }
/************************************************************ my_rc = TCL_OK;
* Prepare to start a controlled return through all PG_TRY();
* interpreter levels on transaction abort during {
* the ouput conversions of the results if (loop_body >= argc)
************************************************************/
memcpy(&save_restart, &Warn_restart, sizeof(save_restart));
if (sigsetjmp(Warn_restart, 1) != 0)
{ {
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart));
pltcl_restart_in_progress = 1;
Tcl_SetResult(interp, "Transaction abort", TCL_VOLATILE);
return TCL_ERROR;
}
/************************************************************ /************************************************************
* If there is no loop body given, just set the variables * If there is no loop body given, just set the variables
* from the first tuple (if any) and return the number of * from the first tuple (if any)
* tuples selected
************************************************************/ ************************************************************/
if (loop_body >= argc)
{
if (ntuples > 0) if (ntuples > 0)
pltcl_set_tuple_values(interp, arrayname, 0, tuples[0], tupdesc); pltcl_set_tuple_values(interp, arrayname, 0,
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); tuples[0], tupdesc);
snprintf(buf, sizeof(buf), "%d", ntuples);
Tcl_SetResult(interp, buf, TCL_VOLATILE);
SPI_freetuptable(SPI_tuptable);
return TCL_OK;
} }
else
tuptable = SPI_tuptable; {
/************************************************************ /************************************************************
* There is a loop body - process all tuples and evaluate * There is a loop body - process all tuples and evaluate
* the body on each * the body on each
************************************************************/ ************************************************************/
for (i = 0; i < ntuples; i++) for (i = 0; i < ntuples; i++)
{ {
pltcl_set_tuple_values(interp, arrayname, i, tuples[i], tupdesc); pltcl_set_tuple_values(interp, arrayname, i,
tuples[i], tupdesc);
loop_rc = Tcl_Eval(interp, argv[loop_body]); loop_rc = Tcl_Eval(interp, argv[loop_body]);
...@@ -2283,26 +2203,37 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp, ...@@ -2283,26 +2203,37 @@ pltcl_SPI_execp(ClientData cdata, Tcl_Interp *interp,
continue; continue;
if (loop_rc == TCL_RETURN) if (loop_rc == TCL_RETURN)
{ {
SPI_freetuptable(tuptable); my_rc = TCL_RETURN;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); break;
return TCL_RETURN;
} }
if (loop_rc == TCL_BREAK) if (loop_rc == TCL_BREAK)
break; break;
SPI_freetuptable(tuptable); my_rc = TCL_ERROR;
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); break;
return TCL_ERROR; }
} }
SPI_freetuptable(tuptable); SPI_freetuptable(tuptable);
}
PG_CATCH();
{
MemoryContextSwitchTo(oldcontext);
pltcl_error_in_progress = CopyErrorData();
FlushErrorState();
Tcl_SetResult(interp, "Transaction aborted", TCL_VOLATILE);
return TCL_ERROR;
}
PG_END_TRY();
/************************************************************ /************************************************************
* Finally return the number of tuples * Finally return the number of tuples
************************************************************/ ************************************************************/
memcpy(&Warn_restart, &save_restart, sizeof(Warn_restart)); if (my_rc == TCL_OK)
{
snprintf(buf, sizeof(buf), "%d", ntuples); snprintf(buf, sizeof(buf), "%d", ntuples);
Tcl_SetResult(interp, buf, TCL_VOLATILE); Tcl_SetResult(interp, buf, TCL_VOLATILE);
return TCL_OK; }
return my_rc;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment