Skip to content
Snippets Groups Projects
elog.c 12.4 KiB
Newer Older
/*-------------------------------------------------------------------------
 *
Bruce Momjian's avatar
Bruce Momjian committed
 * Portions Copyright (c) 1996-2000, PostgreSQL, Inc
 * Portions Copyright (c) 1994, Regents of the University of California
Bruce Momjian's avatar
Bruce Momjian committed
 *	  $Header: /cvsroot/pgsql/src/backend/utils/error/elog.c,v 1.55 2000/02/13 18:59:50 momjian Exp $
 *
 *-------------------------------------------------------------------------
 */
#include <time.h>
#include <fcntl.h>
#ifndef O_RDONLY
#include <sys/file.h>
#endif	 /* O_RDONLY */
#include <sys/types.h>
#include <errno.h>
#include <unistd.h>
#include <signal.h>
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#ifdef USE_SYSLOG
#include <syslog.h>
#endif

#include "libpq/libpq.h"
#include "libpq/pqformat.h"
Bruce Momjian's avatar
Bruce Momjian committed
#include "miscadmin.h"
#include "storage/proc.h"
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#include "utils/trace.h"
Bruce Momjian's avatar
Bruce Momjian committed
#include "commands/copy.h"
Marc G. Fournier's avatar
 
Marc G. Fournier committed

extern int	errno;
extern int	sys_nerr;

extern CommandDest	whereToSendOutput;
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#ifdef USE_SYSLOG
/*
 * Global option to control the use of syslog(3) for logging:
 *
 *		0	stdout/stderr only
 *		1	stdout/stderr + syslog
 *		2	syslog only
Marc G. Fournier's avatar
 
Marc G. Fournier committed
 */
#define UseSyslog pg_options[OPT_SYSLOG]
#define PG_LOG_FACILITY LOG_LOCAL0
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#else
#define UseSyslog 0
#endif
static int	Debugfile = -1;
static int	Err_file = -1;
static int	ElogDebugIndentLevel = 0;
Bruce Momjian's avatar
Bruce Momjian committed
 * elog
 *		Primary error logging function.
 *
 * 'lev': error level; indicates recovery action to take, if any.
 * 'fmt': a printf-style string.
 * Additional arguments, if any, are formatted per %-escapes in 'fmt'.
 *
 * In addition to the usual %-escapes recognized by printf, "%m" in
 * fmt is replaced by the error message for the current value of errno.
 *
 * Note: no newline is needed at the end of the fmt string, since
 * elog will provide one for the output methods that need it.
 *
 * If 'lev' is ERROR or worse, control does not return to the caller.
 * See elog.h for the error level definitions.
 *--------------------
elog(int lev, const char *fmt, ...)
	/*
	 * The expanded format and final output message are dynamically
	 * allocated if necessary, but not if they fit in the "reasonable
	 * size" buffers shown here.  In extremis, we'd rather depend on
	 * having a few hundred bytes of stack space than on malloc() still
	 * working (since memory-clobber errors often take out malloc first).
	 * Don't make these buffers unreasonably large though, on pain of
	 * having to chase a bug with no error message.
	 */
	char		fmt_fixedbuf[128];
	char		msg_fixedbuf[256];
	char	   *fmt_buf = fmt_fixedbuf;
	char	   *msg_buf = msg_fixedbuf;
	/* this buffer is only used if errno has a bogus value: */
	char		errorstr_buf[32];
	const char *errorstr;
	const char *prefix;
	char	   *bp;
	int			indent = 0;
	int			space_needed;
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#ifdef USE_SYSLOG
	int			log_level;
#endif
	if (lev <= DEBUG && Debugfile < 0)
		return;					/* ignore debug msgs if noplace to send */

	if (lev == ERROR || lev == FATAL)
	{
		/* this is probably redundant... */
		if (IsInitProcessingMode())
			lev = FATAL;
	}

	/* choose message prefix and indent level */
			indent = ElogDebugIndentLevel - 1;
			if (indent < 0)
				indent = 0;
			if (indent > 30)
				indent = indent % 30;
			prefix = "DEBUG:  ";
			indent = ElogDebugIndentLevel;
			if (indent < 0)
				indent = 0;
			if (indent > 30)
				indent = indent % 30;
			prefix = "DEBUG:  ";
			/* temporarily use msg buf for prefix */
			sprintf(msg_fixedbuf, "FATAL %d:  ", lev);
			prefix = msg_fixedbuf;
			break;
	}

	/* get errno string for %m */
	if (errno < sys_nerr && errno >= 0)
		errorstr = strerror(errno);
	else
	{
		sprintf(errorstr_buf, "error %d", errno);
		errorstr = errorstr_buf;
	}

	/*
	 * Set up the expanded format, consisting of the prefix string
	 * plus input format, with any %m replaced by strerror() string
	 * (since vsnprintf won't know what to do with %m).  To keep
	 * space calculation simple, we only allow one %m.
	 */
Bruce Momjian's avatar
Bruce Momjian committed
	space_needed = TIMESTAMP_SIZE + strlen(prefix) + indent + (lineno ? 24 : 0)
		+ strlen(fmt) + strlen(errorstr) + 1;
	if (space_needed > (int) sizeof(fmt_fixedbuf))
	{
		fmt_buf = (char *) malloc(space_needed);
		if (fmt_buf == NULL)
		{
			/* We're up against it, convert to fatal out-of-memory error */
			fmt_buf = fmt_fixedbuf;
			lev = REALLYFATAL;
			fmt = "elog: out of memory"; /* this must fit in fmt_fixedbuf! */
		}
#ifdef ELOG_TIMESTAMPS
	strcpy(fmt_buf, tprintf_timestamp());
	strcat(fmt_buf, prefix);
	bp = fmt_buf + strlen(fmt_buf);
	while (indent-- > 0)
Bruce Momjian's avatar
Bruce Momjian committed

	/* If error was in CopyFrom() print the offending line number -- dz */
	if (lineno) {
	    sprintf(bp, "copy: line %d, ", lineno);
	    bp = fmt_buf + strlen(fmt_buf);
	    lineno = 0;
	}

	for (cp = fmt; *cp; cp++)
	{
		if (cp[0] == '%' && cp[1] != '\0')
			if (cp[1] == 'm')
			{
				/* XXX If there are any %'s in errorstr then vsnprintf
				 * will do the Wrong Thing; do we need to cope?
				 * Seems unlikely that % would appear in system errors.
				 */
				strcpy(bp, errorstr);
				/* copy the rest of fmt literally, since we can't
				 * afford to insert another %m.
				 */
				strcat(bp, cp+2);
				bp += strlen(bp);
				break;
			}
			{
				/* copy % and next char --- this avoids trouble with %%m */
				*bp++ = *cp++;
				*bp++ = *cp;
			}

	/*
	 * Now generate the actual output text using vsnprintf().
	 * Be sure to leave space for \n added later as well as trailing null.
	 */
	space_needed = sizeof(msg_fixedbuf);
	for (;;)
	{
		int nprinted;

		va_start(ap, fmt);
		nprinted = vsnprintf(msg_buf, space_needed - 2, fmt_buf, ap);
		va_end(ap);
		/*
		 * Note: some versions of vsnprintf return the number of chars
		 * actually stored, but at least one returns -1 on failure.
		 * Be conservative about believing whether the print worked.
		 */
		if (nprinted >= 0 && nprinted < space_needed - 3)
			break;
		/* It didn't work, try to get a bigger buffer */
		if (msg_buf != msg_fixedbuf)
			free(msg_buf);
		space_needed *= 2;
		msg_buf = (char *) malloc(space_needed);
		if (msg_buf == NULL)
		{
			/* We're up against it, convert to fatal out-of-memory error */
            msg_buf = msg_fixedbuf;
            lev = REALLYFATAL;
#ifdef ELOG_TIMESTAMPS
			strcpy(msg_buf, tprintf_timestamp());
			strcat(msg_buf, "FATAL:  elog: out of memory");
#else
			strcpy(msg_buf, "FATAL:  elog: out of memory");
#endif
			break;
		}
	}

	/*
	 * Message prepared; send it where it should go
	 */
Marc G. Fournier's avatar
 
Marc G. Fournier committed

#ifdef USE_SYSLOG
	switch (lev)
	{
		case NOIND:
			log_level = LOG_DEBUG;
			break;
		case DEBUG:
			log_level = LOG_DEBUG;
			break;
		case NOTICE:
			log_level = LOG_NOTICE;
			break;
		case ERROR:
			log_level = LOG_WARNING;
			break;
		case FATAL:
		default:
			log_level = LOG_ERR;
			break;
Marc G. Fournier's avatar
 
Marc G. Fournier committed
	}
	write_syslog(log_level, msg_buf + TIMESTAMP_SIZE);
Marc G. Fournier's avatar
 
Marc G. Fournier committed
#endif

	/* syslog doesn't want a trailing newline, but other destinations do */
	strcat(msg_buf, "\n");

	len = strlen(msg_buf);

	if (Debugfile >= 0 && UseSyslog <= 1)
		write(Debugfile, msg_buf, len);
	 * If there's an error log file other than our channel to the
	 * front-end program, write to it first.  This is important because
	 * there's a bug in the socket code on ultrix.  If the front end has
	 * gone away (so the channel to it has been closed at the other end),
Bruce Momjian's avatar
Bruce Momjian committed
	 * then writing here can cause this backend to exit without warning
	 * that is, write() does an exit(). In this case, our only hope of
	 * finding out what's going on is if Err_file was set to some disk
	 * log.  This is a major pain.  (It's probably also long-dead code...
	 * does anyone still use ultrix?)
	if (lev > DEBUG && Err_file >= 0 &&
		Debugfile != Err_file && UseSyslog <= 1)
		if (write(Err_file, msg_buf, len) < 0)
			write(open("/dev/console", O_WRONLY, 0666), msg_buf, len);
			lev = REALLYFATAL;
		}
		fsync(Err_file);
	}

#ifndef PG_STANDALONE
	if (lev > DEBUG && whereToSendOutput == Remote)
		/* Send IPC message to the front-end program */
Bruce Momjian's avatar
Bruce Momjian committed
		char		msgtype;

Bruce Momjian's avatar
Bruce Momjian committed
			/*
			 * Abort any COPY OUT in progress when an error is detected.
			 * This hack is necessary because of poor design of copy
			 * protocol.
			 */
			pq_endcopyout(true);
			msgtype = 'E';
		}
		/* exclude the timestamp from msg sent to frontend */
		pq_puttextmessage(msgtype, msg_buf + TIMESTAMP_SIZE);
		/*
		 * This flush is normally not necessary, since postgres.c will
		 * flush out waiting data when control returns to the main loop.
		 * But it seems best to leave it here, so that the client has some
Bruce Momjian's avatar
Bruce Momjian committed
		 * clue what happened if the backend dies before getting back to
		 * the main loop ... error/notice messages should not be a
		 * performance-critical path anyway, so an extra flush won't hurt
Bruce Momjian's avatar
Bruce Momjian committed
		 * much ...
	if (lev > DEBUG && whereToSendOutput != Remote)
	{
		/* We are running as an interactive backend, so just send
		 * the message to stderr.
#endif	 /* !PG_STANDALONE */
	/* done with the message, release space */
	if (fmt_buf != fmt_fixedbuf)
		free(fmt_buf);
	if (msg_buf != msg_fixedbuf)
		free(msg_buf);

	/*
	 * Perform error recovery action as specified by lev.
	 */
	if (lev == ERROR || lev == FATAL)
		/*
		 * If we have not yet entered the main backend loop (ie, we are in
		 * the postmaster or in backend startup), then go directly to
		 * proc_exit.  The same is true if anyone tries to report an error
		 * after proc_exit has begun to run.  (It's proc_exit's responsibility
		 * to see that this doesn't turn into infinite recursion!)  But in
		 * the latter case, we exit with nonzero exit code to indicate that
		 * something's pretty wrong.
		 */
		if (proc_exit_inprogress || ! Warn_restart_ready)
			fflush(stdout);
			fflush(stderr);
			ProcReleaseSpins(NULL); /* get rid of spinlocks we hold */
			ProcReleaseLocks();		/* get rid of real locks we hold */
			/* XXX shouldn't proc_exit be doing the above?? */
			proc_exit((int) proc_exit_inprogress);
		/*
		 * Guard against infinite loop from elog() during error recovery.
		 */
		if (InError)
			elog(REALLYFATAL, "elog: error during error recovery, giving up!");
		/*
		 * Otherwise we can return to the main loop in postgres.c.
		 * In the FATAL case, postgres.c will call proc_exit, but not
		 * till after completing a standard transaction-abort sequence.
		 */
		ProcReleaseSpins(NULL); /* get rid of spinlocks we hold */
		if (lev == FATAL)
			ExitAfterAbort = true;
		siglongjmp(Warn_restart, 1);
		 * Serious crash time. Postmaster will observe nonzero
		 * process exit status and kill the other backends too.
		 *
		 * XXX: what if we are *in* the postmaster?  proc_exit()
		 * won't kill our children...
		fflush(stdout);
		fflush(stderr);

	/* We reach here if lev <= NOTICE.  OK to return to caller. */

	Err_file = Debugfile = -1;
	ElogDebugIndentLevel = 0;

	if (OutputFileName[0])
	{
		if ((fd = open(OutputFileName, O_CREAT | O_APPEND | O_WRONLY,
					   0666)) < 0)
			elog(FATAL, "DebugFileOpen: open of %s: %m",
				 OutputFileName);
		istty = isatty(fd);
		close(fd);

		/*
		 * If the file is a tty and we're running under the postmaster,
		 * try to send stdout there as well (if it isn't a tty then stderr
		 * will block out stdout, so we may as well let stdout go wherever
		 * it was going before).
		 */
		if (istty &&
			IsUnderPostmaster &&
			!freopen(OutputFileName, "a", stdout))
			elog(FATAL, "DebugFileOpen: %s reopen as stdout: %m",
				 OutputFileName);
		if (!freopen(OutputFileName, "a", stderr))
			elog(FATAL, "DebugFileOpen: %s reopen as stderr: %m",
				 OutputFileName);
		Err_file = Debugfile = fileno(stderr);
		return Debugfile;
	}

	/*
	 * If no filename was specified, send debugging output to stderr. If
	 * stderr has been hosed, try to open a file.
	fd = fileno(stderr);
	if (fcntl(fd, F_GETFD, 0) < 0)
	{
		snprintf(OutputFileName, MAXPGPATH, "%s%cpg.errors.%d",
				 DataDir, SEP_CHAR, (int) MyProcPid);
		fd = open(OutputFileName, O_CREAT | O_APPEND | O_WRONLY, 0666);
	}
	if (fd < 0)
		elog(FATAL, "DebugFileOpen: could not open debugging file");

	Err_file = Debugfile = fd;
	return Debugfile;