mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Eliminate elog()'s hardwired limit on length of an error message.
This change seems necessary in conjunction with long queries, and it cleans up some bogosity in connection with long EXPLAIN texts anyway. Note that current libpq will accept any length error message (at least until it runs out of memory); prior versions have a limit of 8K, but will cleanly discard excess error text, so there shouldn't be any big compatibility problems with old clients.
This commit is contained in:
		| @@ -4,7 +4,7 @@ | |||||||
|  * |  * | ||||||
|  * Copyright (c) 1994-5, Regents of the University of California |  * Copyright (c) 1994-5, Regents of the University of California | ||||||
|  * |  * | ||||||
|  *	  $Id: explain.c,v 1.46 1999/08/31 01:28:28 tgl Exp $ |  *	  $Id: explain.c,v 1.47 1999/09/11 19:06:36 tgl Exp $ | ||||||
|  * |  * | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| @@ -28,7 +28,6 @@ typedef struct ExplainState | |||||||
| } ExplainState; | } ExplainState; | ||||||
|  |  | ||||||
| static char *Explain_PlanToString(Plan *plan, ExplainState *es); | static char *Explain_PlanToString(Plan *plan, ExplainState *es); | ||||||
| static void printLongNotice(const char *header, const char *message); |  | ||||||
| static void ExplainOneQuery(Query *query, bool verbose, CommandDest dest); | static void ExplainOneQuery(Query *query, bool verbose, CommandDest dest); | ||||||
|  |  | ||||||
| /* Convert a null string pointer into "<>" */ | /* Convert a null string pointer into "<>" */ | ||||||
| @@ -110,7 +109,7 @@ ExplainOneQuery(Query *query, bool verbose, CommandDest dest) | |||||||
| 		s = nodeToString(plan); | 		s = nodeToString(plan); | ||||||
| 		if (s) | 		if (s) | ||||||
| 		{ | 		{ | ||||||
| 			printLongNotice("QUERY DUMP:\n\n", s); | 			elog(NOTICE, "QUERY DUMP:\n\n%s", s); | ||||||
| 			pfree(s); | 			pfree(s); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -120,7 +119,7 @@ ExplainOneQuery(Query *query, bool verbose, CommandDest dest) | |||||||
| 		s = Explain_PlanToString(plan, es); | 		s = Explain_PlanToString(plan, es); | ||||||
| 		if (s) | 		if (s) | ||||||
| 		{ | 		{ | ||||||
| 			printLongNotice("QUERY PLAN:\n\n", s); | 			elog(NOTICE, "QUERY PLAN:\n\n%s", s); | ||||||
| 			pfree(s); | 			pfree(s); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -332,7 +331,6 @@ explain_outNode(StringInfo str, Plan *plan, int indent, ExplainState *es) | |||||||
| 		} | 		} | ||||||
| 		es->rtable = saved_rtable; | 		es->rtable = saved_rtable; | ||||||
| 	} | 	} | ||||||
| 	return; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| static char * | static char * | ||||||
| @@ -346,22 +344,3 @@ Explain_PlanToString(Plan *plan, ExplainState *es) | |||||||
| 		explain_outNode(&str, plan, 0, es); | 		explain_outNode(&str, plan, 0, es); | ||||||
| 	return str.data; | 	return str.data; | ||||||
| } | } | ||||||
|  |  | ||||||
| /* |  | ||||||
|  * Print a message that might exceed the size of the elog message buffer. |  | ||||||
|  * This is a crock ... there shouldn't be an upper limit to what you can elog(). |  | ||||||
|  */ |  | ||||||
| static void |  | ||||||
| printLongNotice(const char *header, const char *message) |  | ||||||
| { |  | ||||||
| 	int			len = strlen(message); |  | ||||||
|  |  | ||||||
| 	elog(NOTICE, "%.20s%.*s", header, ELOG_MAXLEN - 64, message); |  | ||||||
| 	len -= ELOG_MAXLEN - 64; |  | ||||||
| 	while (len > 0) |  | ||||||
| 	{ |  | ||||||
| 		message += ELOG_MAXLEN - 64; |  | ||||||
| 		elog(NOTICE, "%.*s", ELOG_MAXLEN - 64, message); |  | ||||||
| 		len -= ELOG_MAXLEN - 64; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -7,10 +7,13 @@ | |||||||
|  * |  * | ||||||
|  * |  * | ||||||
|  * IDENTIFICATION |  * IDENTIFICATION | ||||||
|  *	  $Header: /cvsroot/pgsql/src/backend/utils/error/elog.c,v 1.47 1999/07/17 20:18:03 momjian Exp $ |  *	  $Header: /cvsroot/pgsql/src/backend/utils/error/elog.c,v 1.48 1999/09/11 19:06:31 tgl Exp $ | ||||||
|  * |  * | ||||||
|  *------------------------------------------------------------------------- |  *------------------------------------------------------------------------- | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | #include "postgres.h" | ||||||
|  |  | ||||||
| #include <time.h> | #include <time.h> | ||||||
| #include <fcntl.h> | #include <fcntl.h> | ||||||
| #ifndef O_RDONLY | #ifndef O_RDONLY | ||||||
| @@ -20,8 +23,6 @@ | |||||||
| #include <errno.h> | #include <errno.h> | ||||||
| #include <unistd.h> | #include <unistd.h> | ||||||
| #include <signal.h> | #include <signal.h> | ||||||
|  |  | ||||||
| #include "postgres.h" |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| #include <syslog.h> | #include <syslog.h> | ||||||
| #endif | #endif | ||||||
| @@ -33,6 +34,10 @@ | |||||||
| #include "tcop/tcopprot.h" | #include "tcop/tcopprot.h" | ||||||
| #include "utils/trace.h" | #include "utils/trace.h" | ||||||
|  |  | ||||||
|  | extern int	errno; | ||||||
|  | extern int	sys_nerr; | ||||||
|  |  | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| /* | /* | ||||||
|  * Global option to control the use of syslog(3) for logging: |  * Global option to control the use of syslog(3) for logging: | ||||||
| @@ -51,84 +56,198 @@ static int	Debugfile = -1; | |||||||
| static int	Err_file = -1; | static int	Err_file = -1; | ||||||
| static int	ElogDebugIndentLevel = 0; | static int	ElogDebugIndentLevel = 0; | ||||||
|  |  | ||||||
| /* | /*-------------------- | ||||||
|  * elog |  * elog | ||||||
|  *		Old error logging function. |  *		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. | ||||||
|  |  *-------------------- | ||||||
|  */ |  */ | ||||||
| void | void | ||||||
| elog(int lev, const char *fmt,...) | elog(int lev, const char *fmt, ...) | ||||||
| { | { | ||||||
| 	va_list		ap; | 	va_list		ap; | ||||||
| 	char		buf[ELOG_MAXLEN], | 	/* | ||||||
| 				line[ELOG_MAXLEN]; | 	 * The expanded format and final output message are dynamically | ||||||
| 	char	   *bp; | 	 * 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; | ||||||
| 	const char *cp; | 	const char *cp; | ||||||
| 	extern int	errno, | 	char	   *bp; | ||||||
| 				sys_nerr; | 	int			indent = 0; | ||||||
|  | 	int			space_needed; | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| 	int			log_level; | 	int			log_level; | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| 	int			len; | 	int			len; | ||||||
| 	int			i = 0; |  | ||||||
|  |  | ||||||
| 	va_start(ap, fmt); | 	if (lev <= DEBUG && Debugfile < 0) | ||||||
| 	if (lev == DEBUG && Debugfile < 0) | 		return;					/* ignore debug msgs if noplace to send */ | ||||||
| 		return; |  | ||||||
|  | 	/* choose message prefix and indent level */ | ||||||
| 	switch (lev) | 	switch (lev) | ||||||
| 	{ | 	{ | ||||||
| 		case NOIND: | 		case NOIND: | ||||||
| 			i = ElogDebugIndentLevel - 1; | 			indent = ElogDebugIndentLevel - 1; | ||||||
| 			if (i < 0) | 			if (indent < 0) | ||||||
| 				i = 0; | 				indent = 0; | ||||||
| 			if (i > 30) | 			if (indent > 30) | ||||||
| 				i = i % 30; | 				indent = indent % 30; | ||||||
| 			cp = "DEBUG:  "; | 			prefix = "DEBUG:  "; | ||||||
| 			break; | 			break; | ||||||
| 		case DEBUG: | 		case DEBUG: | ||||||
| 			i = ElogDebugIndentLevel; | 			indent = ElogDebugIndentLevel; | ||||||
| 			if (i < 0) | 			if (indent < 0) | ||||||
| 				i = 0; | 				indent = 0; | ||||||
| 			if (i > 30) | 			if (indent > 30) | ||||||
| 				i = i % 30; | 				indent = indent % 30; | ||||||
| 			cp = "DEBUG:  "; | 			prefix = "DEBUG:  "; | ||||||
| 			break; | 			break; | ||||||
| 		case NOTICE: | 		case NOTICE: | ||||||
| 			cp = "NOTICE:  "; | 			prefix = "NOTICE:  "; | ||||||
| 			break; | 			break; | ||||||
| 		case ERROR: | 		case ERROR: | ||||||
| 			cp = "ERROR:  "; | 			prefix = "ERROR:  "; | ||||||
| 			break; | 			break; | ||||||
| 		default: | 		default: | ||||||
| 			sprintf(line, "FATAL %d:  ", lev); | 			/* temporarily use msg buf for prefix */ | ||||||
| 			cp = line; | 			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. | ||||||
|  | 	 */ | ||||||
|  | 	space_needed = TIMESTAMP_SIZE + strlen(prefix) + indent  | ||||||
|  | 		+ 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 | #ifdef ELOG_TIMESTAMPS | ||||||
| 	strcpy(buf, tprintf_timestamp()); | 	strcpy(fmt_buf, tprintf_timestamp()); | ||||||
| 	strcat(buf, cp); | 	strcat(fmt_buf, prefix); | ||||||
| #else | #else | ||||||
| 	strcpy(buf, cp); | 	strcpy(fmt_buf, prefix); | ||||||
| #endif | #endif | ||||||
| 	bp = buf + strlen(buf); | 	bp = fmt_buf + strlen(fmt_buf); | ||||||
| 	while (i-- > 0) | 	while (indent-- > 0) | ||||||
| 		*bp++ = ' '; | 		*bp++ = ' '; | ||||||
| 	for (cp = fmt; *cp; cp++) | 	for (cp = fmt; *cp; cp++) | ||||||
| 		if (*cp == '%' && *(cp + 1) == 'm') | 	{ | ||||||
|  | 		if (cp[0] == '%' && cp[1] != '\0') | ||||||
| 		{ | 		{ | ||||||
| 			if (errno < sys_nerr && errno >= 0) | 			if (cp[1] == 'm') | ||||||
| 				strcpy(bp, strerror(errno)); | 			{ | ||||||
|  | 				/* 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; | ||||||
|  | 			} | ||||||
| 			else | 			else | ||||||
| 				sprintf(bp, "error %d", errno); | 			{ | ||||||
| 			bp += strlen(bp); | 				/* copy % and next char --- this avoids trouble with %%m */ | ||||||
| 			cp++; | 				*bp++ = *cp++; | ||||||
|  | 				*bp++ = *cp; | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		else | 		else | ||||||
| 			*bp++ = *cp; | 			*bp++ = *cp; | ||||||
|  | 	} | ||||||
| 	*bp = '\0'; | 	*bp = '\0'; | ||||||
| 	vsnprintf(line, ELOG_MAXLEN - 1, buf, ap); |  | ||||||
| 	va_end(ap); | 	/* | ||||||
|  | 	 * 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 | ||||||
|  | 	 */ | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| 	switch (lev) | 	switch (lev) | ||||||
| @@ -150,14 +269,16 @@ elog(int lev, const char *fmt,...) | |||||||
| 			log_level = LOG_ERR; | 			log_level = LOG_ERR; | ||||||
| 			break; | 			break; | ||||||
| 	} | 	} | ||||||
| 	write_syslog(log_level, line + TIMESTAMP_SIZE); | 	write_syslog(log_level, msg_buf + TIMESTAMP_SIZE); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| 	len = strlen(strcat(line, "\n")); | 	/* syslog doesn't want a trailing newline, but other destinations do */ | ||||||
| 	if ((Debugfile > -1) && (UseSyslog <= 1)) | 	strcat(msg_buf, "\n"); | ||||||
| 		write(Debugfile, line, len); |  | ||||||
| 	if (lev == DEBUG || lev == NOIND) | 	len = strlen(msg_buf); | ||||||
| 		return; |  | ||||||
|  | 	if (Debugfile >= 0 && UseSyslog <= 1) | ||||||
|  | 		write(Debugfile, msg_buf, len); | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * If there's an error log file other than our channel to the | 	 * If there's an error log file other than our channel to the | ||||||
| @@ -167,33 +288,31 @@ elog(int lev, const char *fmt,...) | |||||||
| 	 * then writing here can cause this backend to exit without warning | 	 * then writing here can cause this backend to exit without warning | ||||||
| 	 * that is, write() does an exit(). In this case, our only hope of | 	 * 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 | 	 * finding out what's going on is if Err_file was set to some disk | ||||||
| 	 * log.  This is a major pain. | 	 * log.  This is a major pain.  (It's probably also long-dead code... | ||||||
|  | 	 * does anyone still use ultrix?) | ||||||
| 	 */ | 	 */ | ||||||
|  | 	if (lev > DEBUG && Err_file >= 0 && | ||||||
| 	if (Err_file > -1 && Debugfile != Err_file && (UseSyslog <= 1)) | 		Debugfile != Err_file && UseSyslog <= 1) | ||||||
| 	{ | 	{ | ||||||
| 		if (write(Err_file, line, len) < 0) | 		if (write(Err_file, msg_buf, len) < 0) | ||||||
| 		{ | 		{ | ||||||
| 			write(open("/dev/console", O_WRONLY, 0666), line, len); | 			write(open("/dev/console", O_WRONLY, 0666), msg_buf, len); | ||||||
| 			fflush(stdout); | 			lev = REALLYFATAL; | ||||||
| 			fflush(stderr); |  | ||||||
| 			proc_exit(lev); |  | ||||||
| 		} | 		} | ||||||
| 		fsync(Err_file); | 		fsync(Err_file); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| #ifndef PG_STANDALONE | #ifndef PG_STANDALONE | ||||||
| 	/* Send IPC message to the front-end program */ |  | ||||||
| 	if (IsUnderPostmaster && lev > DEBUG) | 	if (lev > DEBUG && IsUnderPostmaster) | ||||||
| 	{ | 	{ | ||||||
| 		/* notices are not errors, handle 'em differently */ | 		/* Send IPC message to the front-end program */ | ||||||
| 		char		msgtype; | 		char		msgtype; | ||||||
|  |  | ||||||
| 		if (lev == NOTICE) | 		if (lev == NOTICE) | ||||||
| 			msgtype = 'N'; | 			msgtype = 'N'; | ||||||
| 		else | 		else | ||||||
| 		{ | 		{ | ||||||
|  |  | ||||||
| 			/* | 			/* | ||||||
| 			 * Abort any COPY OUT in progress when an error is detected. | 			 * Abort any COPY OUT in progress when an error is detected. | ||||||
| 			 * This hack is necessary because of poor design of copy | 			 * This hack is necessary because of poor design of copy | ||||||
| @@ -203,7 +322,7 @@ elog(int lev, const char *fmt,...) | |||||||
| 			msgtype = 'E'; | 			msgtype = 'E'; | ||||||
| 		} | 		} | ||||||
| 		/* exclude the timestamp from msg sent to frontend */ | 		/* exclude the timestamp from msg sent to frontend */ | ||||||
| 		pq_puttextmessage(msgtype, line + TIMESTAMP_SIZE); | 		pq_puttextmessage(msgtype, msg_buf + TIMESTAMP_SIZE); | ||||||
|  |  | ||||||
| 		/* | 		/* | ||||||
| 		 * This flush is normally not necessary, since postgres.c will | 		 * This flush is normally not necessary, since postgres.c will | ||||||
| @@ -211,35 +330,45 @@ elog(int lev, const char *fmt,...) | |||||||
| 		 * But it seems best to leave it here, so that the client has some | 		 * But it seems best to leave it here, so that the client has some | ||||||
| 		 * clue what happened if the backend dies before getting back to | 		 * clue what happened if the backend dies before getting back to | ||||||
| 		 * the main loop ... error/notice messages should not be a | 		 * the main loop ... error/notice messages should not be a | ||||||
| 		 * performance- critical path anyway, so an extra flush won't hurt | 		 * performance-critical path anyway, so an extra flush won't hurt | ||||||
| 		 * much ... | 		 * much ... | ||||||
| 		 */ | 		 */ | ||||||
| 		pq_flush(); | 		pq_flush(); | ||||||
| 	} | 	} | ||||||
| 	if (!IsUnderPostmaster) |  | ||||||
| 	{ |  | ||||||
|  |  | ||||||
| 		/* | 	if (lev > DEBUG && ! IsUnderPostmaster) | ||||||
| 		 * There is no socket.	One explanation for this is we are running | 	{ | ||||||
| 		 * as the Postmaster.  So we'll write the message to stderr. | 		/* We are running as an interactive backend, so just send | ||||||
|  | 		 * the message to stderr. | ||||||
| 		 */ | 		 */ | ||||||
| 		fputs(line, stderr); | 		fputs(msg_buf, stderr); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| #endif	 /* !PG_STANDALONE */ | #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) | 	if (lev == ERROR) | ||||||
| 	{ | 	{ | ||||||
| 		ProcReleaseSpins(NULL); /* get rid of spinlocks we hold */ | 		if (InError) | ||||||
| 		if (!InError) |  | ||||||
| 		{ | 		{ | ||||||
| 			/* exit to main loop */ | 			/* error reported during error recovery; don't loop forever */ | ||||||
| 			siglongjmp(Warn_restart, 1); | 			elog(REALLYFATAL, "elog: error during error recovery, giving up!"); | ||||||
| 		} | 		} | ||||||
|  | 		/* exit to main loop */ | ||||||
|  | 		ProcReleaseSpins(NULL); /* get rid of spinlocks we hold */ | ||||||
|  | 		siglongjmp(Warn_restart, 1); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (lev == FATAL) | 	if (lev == FATAL) | ||||||
| 	{ | 	{ | ||||||
|  |  | ||||||
| 		/* | 		/* | ||||||
| 		 * Assume that if we have detected the failure we can exit with a | 		 * Assume that if we have detected the failure we can exit with a | ||||||
| 		 * normal exit status.	This will prevent the postmaster from | 		 * normal exit status.	This will prevent the postmaster from | ||||||
| @@ -254,13 +383,20 @@ elog(int lev, const char *fmt,...) | |||||||
|  |  | ||||||
| 	if (lev > FATAL) | 	if (lev > FATAL) | ||||||
| 	{ | 	{ | ||||||
|  | 		/* | ||||||
|  | 		 * Serious crash time.  Postmaster will observe nonzero | ||||||
|  | 		 * process exit status and kill the other backends too. | ||||||
|  | 		 */ | ||||||
| 		fflush(stdout); | 		fflush(stdout); | ||||||
| 		fflush(stderr); | 		fflush(stderr); | ||||||
| 		proc_exit(lev); | 		proc_exit(lev); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	/* We reach here if lev <= NOTICE.  OK to return to caller. */ | ||||||
| } | } | ||||||
|  |  | ||||||
| #ifndef PG_STANDALONE | #ifndef PG_STANDALONE | ||||||
|  |  | ||||||
| int | int | ||||||
| DebugFileOpen(void) | DebugFileOpen(void) | ||||||
| { | { | ||||||
|   | |||||||
| @@ -2,22 +2,21 @@ | |||||||
|  * |  * | ||||||
|  * trace.c |  * trace.c | ||||||
|  * |  * | ||||||
|  *	  Conditional trace ans logging functions. |  *	  Conditional trace and logging functions. | ||||||
|  * |  * | ||||||
|  *	  Massimo Dal Zotto <dz@cs.unitn.it> |  *	  Massimo Dal Zotto <dz@cs.unitn.it> | ||||||
|  * |  * | ||||||
|  *------------------------------------------------------------------------- |  *------------------------------------------------------------------------- | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | #include "postgres.h" | ||||||
|  |  | ||||||
| #include <unistd.h> | #include <unistd.h> | ||||||
| #include <signal.h> | #include <signal.h> | ||||||
| #include <sys/time.h> | #include <sys/time.h> | ||||||
| #include <sys/types.h> | #include <sys/types.h> | ||||||
| #include <sys/stat.h> | #include <sys/stat.h> | ||||||
| #include <fcntl.h> | #include <fcntl.h> | ||||||
|  |  | ||||||
| #include "postgres.h" |  | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| #include <syslog.h> | #include <syslog.h> | ||||||
| #endif | #endif | ||||||
| @@ -25,6 +24,13 @@ | |||||||
| #include "miscadmin.h" | #include "miscadmin.h" | ||||||
| #include "utils/trace.h" | #include "utils/trace.h" | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * We could support trace messages of indefinite length, as elog() does, | ||||||
|  |  * but it's probably not worth the trouble.  Instead limit trace message | ||||||
|  |  * length to this. | ||||||
|  |  */ | ||||||
|  | #define TRACEMSG_MAXLEN		1024 | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| /* | /* | ||||||
|  * Global option to control the use of syslog(3) for logging: |  * Global option to control the use of syslog(3) for logging: | ||||||
| @@ -87,15 +93,14 @@ int | |||||||
| tprintf(int flag, const char *fmt,...) | tprintf(int flag, const char *fmt,...) | ||||||
| { | { | ||||||
| 	va_list		ap; | 	va_list		ap; | ||||||
| 	char		line[ELOG_MAXLEN + TIMESTAMP_SIZE + 1]; | 	char		line[TRACEMSG_MAXLEN + TIMESTAMP_SIZE + 1]; | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| 	int			log_level; | 	int			log_level; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| 	if ((flag == TRACE_ALL) || (pg_options[TRACE_ALL] > 0)) | 	if ((flag == TRACE_ALL) || (pg_options[TRACE_ALL] > 0)) | ||||||
| 	{ | 	{ | ||||||
| 		/* uconditional trace or trace all option set */ | 		/* unconditional trace or trace all option set */ | ||||||
| 	} | 	} | ||||||
| 	else if (pg_options[TRACE_ALL] == 0) | 	else if (pg_options[TRACE_ALL] == 0) | ||||||
| 	{ | 	{ | ||||||
| @@ -105,11 +110,11 @@ tprintf(int flag, const char *fmt,...) | |||||||
| 	else if (pg_options[TRACE_ALL] < 0) | 	else if (pg_options[TRACE_ALL] < 0) | ||||||
| 		return 0; | 		return 0; | ||||||
|  |  | ||||||
| 	va_start(ap, fmt); |  | ||||||
| #ifdef ELOG_TIMESTAMPS | #ifdef ELOG_TIMESTAMPS | ||||||
| 	strcpy(line, tprintf_timestamp()); | 	strcpy(line, tprintf_timestamp()); | ||||||
| #endif | #endif | ||||||
| 	vsnprintf(line + TIMESTAMP_SIZE, ELOG_MAXLEN, fmt, ap); | 	va_start(ap, fmt); | ||||||
|  | 	vsnprintf(line + TIMESTAMP_SIZE, TRACEMSG_MAXLEN, fmt, ap); | ||||||
| 	va_end(ap); | 	va_end(ap); | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| @@ -134,13 +139,13 @@ int | |||||||
| tprintf1(const char *fmt,...) | tprintf1(const char *fmt,...) | ||||||
| { | { | ||||||
| 	va_list		ap; | 	va_list		ap; | ||||||
| 	char		line[ELOG_MAXLEN + TIMESTAMP_SIZE + 1]; | 	char		line[TRACEMSG_MAXLEN + TIMESTAMP_SIZE + 1]; | ||||||
|  |  | ||||||
| 	va_start(ap, fmt); |  | ||||||
| #ifdef ELOG_TIMESTAMPS | #ifdef ELOG_TIMESTAMPS | ||||||
| 	strcpy(line, tprintf_timestamp()); | 	strcpy(line, tprintf_timestamp()); | ||||||
| #endif | #endif | ||||||
| 	vsnprintf(line + TIMESTAMP_SIZE, ELOG_MAXLEN, fmt, ap); | 	va_start(ap, fmt); | ||||||
|  | 	vsnprintf(line + TIMESTAMP_SIZE, TRACEMSG_MAXLEN, fmt, ap); | ||||||
| 	va_end(ap); | 	va_end(ap); | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
| @@ -164,13 +169,13 @@ int | |||||||
| eprintf(const char *fmt,...) | eprintf(const char *fmt,...) | ||||||
| { | { | ||||||
| 	va_list		ap; | 	va_list		ap; | ||||||
| 	char		line[ELOG_MAXLEN + TIMESTAMP_SIZE + 1]; | 	char		line[TRACEMSG_MAXLEN + TIMESTAMP_SIZE + 1]; | ||||||
|  |  | ||||||
| 	va_start(ap, fmt); |  | ||||||
| #ifdef ELOG_TIMESTAMPS | #ifdef ELOG_TIMESTAMPS | ||||||
| 	strcpy(line, tprintf_timestamp()); | 	strcpy(line, tprintf_timestamp()); | ||||||
| #endif | #endif | ||||||
| 	vsnprintf(line + TIMESTAMP_SIZE, ELOG_MAXLEN, fmt, ap); | 	va_start(ap, fmt); | ||||||
|  | 	vsnprintf(line + TIMESTAMP_SIZE, TRACEMSG_MAXLEN, fmt, ap); | ||||||
| 	va_end(ap); | 	va_end(ap); | ||||||
|  |  | ||||||
| #ifdef USE_SYSLOG | #ifdef USE_SYSLOG | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  * |  * | ||||||
|  * Copyright (c) 1994, Regents of the University of California |  * Copyright (c) 1994, Regents of the University of California | ||||||
|  * |  * | ||||||
|  * $Id: elog.h,v 1.11 1999/07/13 21:17:42 momjian Exp $ |  * $Id: elog.h,v 1.12 1999/09/11 19:06:25 tgl Exp $ | ||||||
|  * |  * | ||||||
|  *------------------------------------------------------------------------- |  *------------------------------------------------------------------------- | ||||||
|  */ |  */ | ||||||
| @@ -14,36 +14,16 @@ | |||||||
| #define ELOG_H | #define ELOG_H | ||||||
|  |  | ||||||
| #define NOTICE	0				/* random info - no special action */ | #define NOTICE	0				/* random info - no special action */ | ||||||
| #define ERROR	-1				/* user error - return to known state */ | #define ERROR	(-1)			/* user error - return to known state */ | ||||||
| #define FATAL	1				/* Fatal error - abort process */ | #define FATAL	1				/* fatal error - abort process */ | ||||||
| #define DEBUG	-2				/* debug message */ | #define REALLYFATAL	2			/* take down the other backends with me */ | ||||||
| #define NOIND	-3				/* debug message, don't indent as far */ | #define DEBUG	(-2)			/* debug message */ | ||||||
|  | #define NOIND	(-3)			/* debug message, don't indent as far */ | ||||||
|  |  | ||||||
| #ifdef NOT_USED | extern void elog(int lev, const char *fmt, ...); | ||||||
| #define PTIME	0x100			/* prepend time to message */ |  | ||||||
| #define POS		0x200			/* prepend source position to message */ |  | ||||||
| #define USERMSG 0x400			/* send message to user */ |  | ||||||
| #define TERM	0x800			/* send message to terminal */ |  | ||||||
| #define DBLOG	0x1000			/* put message in per db log */ |  | ||||||
| #define SLOG	0x2000			/* put message in system log */ |  | ||||||
| #define ABORTX	0x4000			/* abort process after logging */ |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| /* Increase this to be able to use postmaster -d 3 with complex |  | ||||||
|  * view definitions (which are transformed to very, very large INSERT statements |  | ||||||
|  * and if -d 3 is used the query string of these statements is printed using |  | ||||||
|  * vsprintf which expects enough memory reserved! */ |  | ||||||
| #define ELOG_MAXLEN 12288 |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* uncomment the following if you want your elog's to be timestamped */ |  | ||||||
| /* #define ELOG_TIMESTAMPS */ |  | ||||||
|  |  | ||||||
| extern void elog(int lev, const char *fmt,...); |  | ||||||
|  |  | ||||||
| #ifndef PG_STANDALONE | #ifndef PG_STANDALONE | ||||||
| int			DebugFileOpen(void); | extern int	DebugFileOpen(void); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #endif	 /* ELOG_H */ | #endif	 /* ELOG_H */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user