Commit 2594cf0e authored by Tom Lane's avatar Tom Lane

Revise the API for GUC variable assign hooks.

The previous functions of assign hooks are now split between check hooks
and assign hooks, where the former can fail but the latter shouldn't.
Aside from being conceptually clearer, this approach exposes the
"canonicalized" form of the variable value to guc.c without having to do
an actual assignment.  And that lets us fix the problem recently noted by
Bernd Helmle that the auto-tune patch for wal_buffers resulted in bogus
log messages about "parameter "wal_buffers" cannot be changed without
restarting the server".  There may be some speed advantage too, because
this design lets hook functions avoid re-parsing variable values when
restoring a previous state after a rollback (they can store a pre-parsed
representation of the value instead).  This patch also resolves a
longstanding annoyance about custom error messages from variable assign
hooks: they should modify, not appear separately from, guc.c's own message
about "invalid parameter value".
parent 5d0e4623
...@@ -53,7 +53,7 @@ auth_delay_checks(Port *port, int status) ...@@ -53,7 +53,7 @@ auth_delay_checks(Port *port, int status)
void void
_PG_init(void) _PG_init(void)
{ {
/* Define custome GUC variables */ /* Define custom GUC variables */
DefineCustomIntVariable("auth_delay.milliseconds", DefineCustomIntVariable("auth_delay.milliseconds",
"Milliseconds to delay before reporting authentication failure", "Milliseconds to delay before reporting authentication failure",
NULL, NULL,
...@@ -63,6 +63,7 @@ _PG_init(void) ...@@ -63,6 +63,7 @@ _PG_init(void)
PGC_SIGHUP, PGC_SIGHUP,
GUC_UNIT_MS, GUC_UNIT_MS,
NULL, NULL,
NULL,
NULL); NULL);
/* Install Hooks */ /* Install Hooks */
original_client_auth_hook = ClientAuthentication_hook; original_client_auth_hook = ClientAuthentication_hook;
......
...@@ -74,6 +74,7 @@ _PG_init(void) ...@@ -74,6 +74,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
GUC_UNIT_MS, GUC_UNIT_MS,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("auto_explain.log_analyze", DefineCustomBoolVariable("auto_explain.log_analyze",
...@@ -84,6 +85,7 @@ _PG_init(void) ...@@ -84,6 +85,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("auto_explain.log_verbose", DefineCustomBoolVariable("auto_explain.log_verbose",
...@@ -94,6 +96,7 @@ _PG_init(void) ...@@ -94,6 +96,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("auto_explain.log_buffers", DefineCustomBoolVariable("auto_explain.log_buffers",
...@@ -104,6 +107,7 @@ _PG_init(void) ...@@ -104,6 +107,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomEnumVariable("auto_explain.log_format", DefineCustomEnumVariable("auto_explain.log_format",
...@@ -115,6 +119,7 @@ _PG_init(void) ...@@ -115,6 +119,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("auto_explain.log_nested_statements", DefineCustomBoolVariable("auto_explain.log_nested_statements",
...@@ -125,6 +130,7 @@ _PG_init(void) ...@@ -125,6 +130,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
EmitWarningsOnPlaceholders("auto_explain"); EmitWarningsOnPlaceholders("auto_explain");
......
...@@ -219,6 +219,7 @@ _PG_init(void) ...@@ -219,6 +219,7 @@ _PG_init(void)
PGC_POSTMASTER, PGC_POSTMASTER,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomEnumVariable("pg_stat_statements.track", DefineCustomEnumVariable("pg_stat_statements.track",
...@@ -230,6 +231,7 @@ _PG_init(void) ...@@ -230,6 +231,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("pg_stat_statements.track_utility", DefineCustomBoolVariable("pg_stat_statements.track_utility",
...@@ -240,6 +242,7 @@ _PG_init(void) ...@@ -240,6 +242,7 @@ _PG_init(void)
PGC_SUSET, PGC_SUSET,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
DefineCustomBoolVariable("pg_stat_statements.save", DefineCustomBoolVariable("pg_stat_statements.save",
...@@ -250,6 +253,7 @@ _PG_init(void) ...@@ -250,6 +253,7 @@ _PG_init(void)
PGC_SIGHUP, PGC_SIGHUP,
0, 0,
NULL, NULL,
NULL,
NULL); NULL);
EmitWarningsOnPlaceholders("pg_stat_statements"); EmitWarningsOnPlaceholders("pg_stat_statements");
......
...@@ -394,6 +394,7 @@ _PG_init(void) ...@@ -394,6 +394,7 @@ _PG_init(void)
PGC_SIGHUP, PGC_SIGHUP,
GUC_NOT_IN_SAMPLE, GUC_NOT_IN_SAMPLE,
NULL, NULL,
NULL,
NULL); NULL);
/* /*
...@@ -412,6 +413,7 @@ _PG_init(void) ...@@ -412,6 +413,7 @@ _PG_init(void)
PGC_USERSET, PGC_USERSET,
GUC_NOT_IN_SAMPLE, GUC_NOT_IN_SAMPLE,
NULL, NULL,
NULL,
NULL); NULL);
/* /*
......
...@@ -4887,36 +4887,60 @@ GetSystemIdentifier(void) ...@@ -4887,36 +4887,60 @@ GetSystemIdentifier(void)
/* /*
* Auto-tune the number of XLOG buffers. * Auto-tune the number of XLOG buffers.
* *
* If the user-set value of wal_buffers is -1, we auto-tune to about 3% of * The preferred setting for wal_buffers is about 3% of shared_buffers, with
* shared_buffers, with a maximum of one XLOG segment and a minimum of 8 * a maximum of one XLOG segment (there is little reason to think that more
* blocks (8 was the default value prior to PostgreSQL 9.1, when auto-tuning * is helpful, at least so long as we force an fsync when switching log files)
* was added). We also clamp manually-set values to at least 4 blocks; prior * and a minimum of 8 blocks (which was the default value prior to PostgreSQL
* to PostgreSQL 9.1, a minimum of 4 was enforced by guc.c, but since that * 9.1, when auto-tuning was added).
* is no longer possible, we just silently treat such values as a request for *
* the minimum. * This should not be called until NBuffers has received its final value.
*/ */
static void static int
XLOGTuneNumBuffers(void) XLOGChooseNumBuffers(void)
{ {
int xbuffers = XLOGbuffers; int xbuffers;
char buf[32];
if (xbuffers == -1)
{
xbuffers = NBuffers / 32; xbuffers = NBuffers / 32;
if (xbuffers > XLOG_SEG_SIZE / XLOG_BLCKSZ) if (xbuffers > XLOG_SEG_SIZE / XLOG_BLCKSZ)
xbuffers = XLOG_SEG_SIZE / XLOG_BLCKSZ; xbuffers = XLOG_SEG_SIZE / XLOG_BLCKSZ;
if (xbuffers < 8) if (xbuffers < 8)
xbuffers = 8; xbuffers = 8;
} return xbuffers;
else if (xbuffers < 4) }
xbuffers = 4;
if (xbuffers != XLOGbuffers) /*
* GUC check_hook for wal_buffers
*/
bool
check_wal_buffers(int *newval, void **extra, GucSource source)
{
/*
* -1 indicates a request for auto-tune.
*/
if (*newval == -1)
{ {
snprintf(buf, sizeof(buf), "%d", xbuffers); /*
SetConfigOption("wal_buffers", buf, PGC_POSTMASTER, PGC_S_OVERRIDE); * If we haven't yet changed the boot_val default of -1, just let it
* be. We'll fix it when XLOGShmemSize is called.
*/
if (XLOGbuffers == -1)
return true;
/* Otherwise, substitute the auto-tune value */
*newval = XLOGChooseNumBuffers();
} }
/*
* We clamp manually-set values to at least 4 blocks. Prior to PostgreSQL
* 9.1, a minimum of 4 was enforced by guc.c, but since that is no longer
* the case, we just silently treat such values as a request for the
* minimum. (We could throw an error instead, but that doesn't seem very
* helpful.)
*/
if (*newval < 4)
*newval = 4;
return true;
} }
/* /*
...@@ -4927,8 +4951,19 @@ XLOGShmemSize(void) ...@@ -4927,8 +4951,19 @@ XLOGShmemSize(void)
{ {
Size size; Size size;
/* Figure out how many XLOG buffers we need. */ /*
XLOGTuneNumBuffers(); * If the value of wal_buffers is -1, use the preferred auto-tune value.
* This isn't an amazingly clean place to do this, but we must wait till
* NBuffers has received its final value, and must do it before using
* the value of XLOGbuffers to do anything important.
*/
if (XLOGbuffers == -1)
{
char buf[32];
snprintf(buf, sizeof(buf), "%d", XLOGChooseNumBuffers());
SetConfigOption("wal_buffers", buf, PGC_POSTMASTER, PGC_S_OVERRIDE);
}
Assert(XLOGbuffers > 0); Assert(XLOGbuffers > 0);
/* XLogCtl */ /* XLogCtl */
...@@ -8653,12 +8688,9 @@ get_sync_bit(int method) ...@@ -8653,12 +8688,9 @@ get_sync_bit(int method)
/* /*
* GUC support * GUC support
*/ */
bool void
assign_xlog_sync_method(int new_sync_method, bool doit, GucSource source) assign_xlog_sync_method(int new_sync_method, void *extra)
{ {
if (!doit)
return true;
if (sync_method != new_sync_method) if (sync_method != new_sync_method)
{ {
/* /*
...@@ -8678,8 +8710,6 @@ assign_xlog_sync_method(int new_sync_method, bool doit, GucSource source) ...@@ -8678,8 +8710,6 @@ assign_xlog_sync_method(int new_sync_method, bool doit, GucSource source)
XLogFileClose(); XLogFileClose();
} }
} }
return true;
} }
......
...@@ -3468,31 +3468,33 @@ ResetTempTableNamespace(void) ...@@ -3468,31 +3468,33 @@ ResetTempTableNamespace(void)
* Routines for handling the GUC variable 'search_path'. * Routines for handling the GUC variable 'search_path'.
*/ */
/* assign_hook: validate new search_path, do extra actions as needed */ /* check_hook: validate new search_path, if possible */
const char * bool
assign_search_path(const char *newval, bool doit, GucSource source) check_search_path(char **newval, void **extra, GucSource source)
{ {
bool result = true;
char *rawname; char *rawname;
List *namelist; List *namelist;
ListCell *l; ListCell *l;
/* Need a modifiable copy of string */ /* Need a modifiable copy of string */
rawname = pstrdup(newval); rawname = pstrdup(*newval);
/* Parse string into list of identifiers */ /* Parse string into list of identifiers */
if (!SplitIdentifierString(rawname, ',', &namelist)) if (!SplitIdentifierString(rawname, ',', &namelist))
{ {
/* syntax error in name list */ /* syntax error in name list */
GUC_check_errdetail("List syntax is invalid.");
pfree(rawname); pfree(rawname);
list_free(namelist); list_free(namelist);
return NULL; return false;
} }
/* /*
* If we aren't inside a transaction, we cannot do database access so * If we aren't inside a transaction, we cannot do database access so
* cannot verify the individual names. Must accept the list on faith. * cannot verify the individual names. Must accept the list on faith.
*/ */
if (source >= PGC_S_INTERACTIVE && IsTransactionState()) if (IsTransactionState())
{ {
/* /*
* Verify that all the names are either valid namespace names or * Verify that all the names are either valid namespace names or
...@@ -3504,7 +3506,7 @@ assign_search_path(const char *newval, bool doit, GucSource source) ...@@ -3504,7 +3506,7 @@ assign_search_path(const char *newval, bool doit, GucSource source)
* DATABASE SET or ALTER USER SET command. It could be that the * DATABASE SET or ALTER USER SET command. It could be that the
* intended use of the search path is for some other database, so we * intended use of the search path is for some other database, so we
* should not error out if it mentions schemas not present in the * should not error out if it mentions schemas not present in the
* current database. We reduce the message to NOTICE instead. * current database. We issue a NOTICE instead.
*/ */
foreach(l, namelist) foreach(l, namelist)
{ {
...@@ -3516,24 +3518,37 @@ assign_search_path(const char *newval, bool doit, GucSource source) ...@@ -3516,24 +3518,37 @@ assign_search_path(const char *newval, bool doit, GucSource source)
continue; continue;
if (!SearchSysCacheExists1(NAMESPACENAME, if (!SearchSysCacheExists1(NAMESPACENAME,
CStringGetDatum(curname))) CStringGetDatum(curname)))
ereport((source == PGC_S_TEST) ? NOTICE : ERROR, {
if (source == PGC_S_TEST)
ereport(NOTICE,
(errcode(ERRCODE_UNDEFINED_SCHEMA), (errcode(ERRCODE_UNDEFINED_SCHEMA),
errmsg("schema \"%s\" does not exist", curname))); errmsg("schema \"%s\" does not exist", curname)));
else
{
GUC_check_errdetail("schema \"%s\" does not exist", curname);
result = false;
break;
}
}
} }
} }
pfree(rawname); pfree(rawname);
list_free(namelist); list_free(namelist);
return result;
}
/* assign_hook: do extra actions as needed */
void
assign_search_path(const char *newval, void *extra)
{
/* /*
* We mark the path as needing recomputation, but don't do anything until * We mark the path as needing recomputation, but don't do anything until
* it's needed. This avoids trying to do database access during GUC * it's needed. This avoids trying to do database access during GUC
* initialization. * initialization, or outside a transaction.
*/ */
if (doit)
baseSearchPathValid = false; baseSearchPathValid = false;
return newval;
} }
/* /*
......
...@@ -1023,9 +1023,9 @@ AlterTableSpaceOptions(AlterTableSpaceOptionsStmt *stmt) ...@@ -1023,9 +1023,9 @@ AlterTableSpaceOptions(AlterTableSpaceOptionsStmt *stmt)
* Routines for handling the GUC variable 'default_tablespace'. * Routines for handling the GUC variable 'default_tablespace'.
*/ */
/* assign_hook: validate new default_tablespace, do extra actions as needed */ /* check_hook: validate new default_tablespace */
const char * bool
assign_default_tablespace(const char *newval, bool doit, GucSource source) check_default_tablespace(char **newval, void **extra, GucSource source)
{ {
/* /*
* If we aren't inside a transaction, we cannot do database access so * If we aren't inside a transaction, we cannot do database access so
...@@ -1033,18 +1033,16 @@ assign_default_tablespace(const char *newval, bool doit, GucSource source) ...@@ -1033,18 +1033,16 @@ assign_default_tablespace(const char *newval, bool doit, GucSource source)
*/ */
if (IsTransactionState()) if (IsTransactionState())
{ {
if (newval[0] != '\0' && if (**newval != '\0' &&
!OidIsValid(get_tablespace_oid(newval, true))) !OidIsValid(get_tablespace_oid(*newval, true)))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errdetail("Tablespace \"%s\" does not exist.",
(errcode(ERRCODE_UNDEFINED_OBJECT), *newval);
errmsg("tablespace \"%s\" does not exist", return false;
newval)));
return NULL;
} }
} }
return newval; return true;
} }
/* /*
...@@ -1100,23 +1098,30 @@ GetDefaultTablespace(char relpersistence) ...@@ -1100,23 +1098,30 @@ GetDefaultTablespace(char relpersistence)
* Routines for handling the GUC variable 'temp_tablespaces'. * Routines for handling the GUC variable 'temp_tablespaces'.
*/ */
/* assign_hook: validate new temp_tablespaces, do extra actions as needed */ typedef struct
const char * {
assign_temp_tablespaces(const char *newval, bool doit, GucSource source) int numSpcs;
Oid tblSpcs[1]; /* VARIABLE LENGTH ARRAY */
} temp_tablespaces_extra;
/* check_hook: validate new temp_tablespaces */
bool
check_temp_tablespaces(char **newval, void **extra, GucSource source)
{ {
char *rawname; char *rawname;
List *namelist; List *namelist;
/* Need a modifiable copy of string */ /* Need a modifiable copy of string */
rawname = pstrdup(newval); rawname = pstrdup(*newval);
/* Parse string into list of identifiers */ /* Parse string into list of identifiers */
if (!SplitIdentifierString(rawname, ',', &namelist)) if (!SplitIdentifierString(rawname, ',', &namelist))
{ {
/* syntax error in name list */ /* syntax error in name list */
GUC_check_errdetail("List syntax is invalid.");
pfree(rawname); pfree(rawname);
list_free(namelist); list_free(namelist);
return NULL; return false;
} }
/* /*
...@@ -1126,17 +1131,13 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source) ...@@ -1126,17 +1131,13 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source)
*/ */
if (IsTransactionState()) if (IsTransactionState())
{ {
/* temp_tablespaces_extra *myextra;
* If we error out below, or if we are called multiple times in one
* transaction, we'll leak a bit of TopTransactionContext memory.
* Doesn't seem worth worrying about.
*/
Oid *tblSpcs; Oid *tblSpcs;
int numSpcs; int numSpcs;
ListCell *l; ListCell *l;
tblSpcs = (Oid *) MemoryContextAlloc(TopTransactionContext, /* temporary workspace until we are done verifying the list */
list_length(namelist) * sizeof(Oid)); tblSpcs = (Oid *) palloc(list_length(namelist) * sizeof(Oid));
numSpcs = 0; numSpcs = 0;
foreach(l, namelist) foreach(l, namelist)
{ {
...@@ -1169,7 +1170,7 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source) ...@@ -1169,7 +1170,7 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source)
continue; continue;
} }
/* Check permissions similarly */ /* Check permissions, similarly complaining only if interactive */
aclresult = pg_tablespace_aclcheck(curoid, GetUserId(), aclresult = pg_tablespace_aclcheck(curoid, GetUserId(),
ACL_CREATE); ACL_CREATE);
if (aclresult != ACLCHECK_OK) if (aclresult != ACLCHECK_OK)
...@@ -1182,17 +1183,41 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source) ...@@ -1182,17 +1183,41 @@ assign_temp_tablespaces(const char *newval, bool doit, GucSource source)
tblSpcs[numSpcs++] = curoid; tblSpcs[numSpcs++] = curoid;
} }
/* If actively "doing it", give the new list to fd.c */ /* Now prepare an "extra" struct for assign_temp_tablespaces */
if (doit) myextra = malloc(offsetof(temp_tablespaces_extra, tblSpcs) +
SetTempTablespaces(tblSpcs, numSpcs); numSpcs * sizeof(Oid));
else if (!myextra)
return false;
myextra->numSpcs = numSpcs;
memcpy(myextra->tblSpcs, tblSpcs, numSpcs * sizeof(Oid));
*extra = (void *) myextra;
pfree(tblSpcs); pfree(tblSpcs);
} }
pfree(rawname); pfree(rawname);
list_free(namelist); list_free(namelist);
return newval; return true;
}
/* assign_hook: do extra actions as needed */
void
assign_temp_tablespaces(const char *newval, void *extra)
{
temp_tablespaces_extra *myextra = (temp_tablespaces_extra *) extra;
/*
* If check_temp_tablespaces was executed inside a transaction, then pass
* the list it made to fd.c. Otherwise, clear fd.c's list; we must be
* still outside a transaction, or else restoring during transaction exit,
* and in either case we can just let the next PrepareTempTablespaces call
* make things sane.
*/
if (myextra)
SetTempTablespaces(myextra->tblSpcs, myextra->numSpcs);
else
SetTempTablespaces(NULL, 0);
} }
/* /*
......
...@@ -33,10 +33,10 @@ ...@@ -33,10 +33,10 @@
*/ */
/* /*
* assign_datestyle: GUC assign_hook for datestyle * check_datestyle: GUC check_hook for datestyle
*/ */
const char * bool
assign_datestyle(const char *value, bool doit, GucSource source) check_datestyle(char **newval, void **extra, GucSource source)
{ {
int newDateStyle = DateStyle; int newDateStyle = DateStyle;
int newDateOrder = DateOrder; int newDateOrder = DateOrder;
...@@ -44,23 +44,22 @@ assign_datestyle(const char *value, bool doit, GucSource source) ...@@ -44,23 +44,22 @@ assign_datestyle(const char *value, bool doit, GucSource source)
bool have_order = false; bool have_order = false;
bool ok = true; bool ok = true;
char *rawstring; char *rawstring;
int *myextra;
char *result; char *result;
List *elemlist; List *elemlist;
ListCell *l; ListCell *l;
/* Need a modifiable copy of string */ /* Need a modifiable copy of string */
rawstring = pstrdup(value); rawstring = pstrdup(*newval);
/* Parse string into list of identifiers */ /* Parse string into list of identifiers */
if (!SplitIdentifierString(rawstring, ',', &elemlist)) if (!SplitIdentifierString(rawstring, ',', &elemlist))
{ {
/* syntax error in list */ /* syntax error in list */
GUC_check_errdetail("List syntax is invalid.");
pfree(rawstring); pfree(rawstring);
list_free(elemlist); list_free(elemlist);
ereport(GUC_complaint_elevel(source), return false;
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("invalid list syntax for parameter \"datestyle\"")));
return NULL;
} }
foreach(l, elemlist) foreach(l, elemlist)
...@@ -130,39 +129,39 @@ assign_datestyle(const char *value, bool doit, GucSource source) ...@@ -130,39 +129,39 @@ assign_datestyle(const char *value, bool doit, GucSource source)
* Easiest way to get the current DEFAULT state is to fetch the * Easiest way to get the current DEFAULT state is to fetch the
* DEFAULT string from guc.c and recursively parse it. * DEFAULT string from guc.c and recursively parse it.
* *
* We can't simply "return assign_datestyle(...)" because we need * We can't simply "return check_datestyle(...)" because we need
* to handle constructs like "DEFAULT, ISO". * to handle constructs like "DEFAULT, ISO".
*/ */
int saveDateStyle = DateStyle; char *subval;
int saveDateOrder = DateOrder; void *subextra = NULL;
const char *subval;
subval = assign_datestyle(GetConfigOptionResetString("datestyle"), subval = strdup(GetConfigOptionResetString("datestyle"));
true, source);
if (!have_style)
newDateStyle = DateStyle;
if (!have_order)
newDateOrder = DateOrder;
DateStyle = saveDateStyle;
DateOrder = saveDateOrder;
if (!subval) if (!subval)
{ {
ok = false; ok = false;
break; break;
} }
/* Here we know that our own return value is always malloc'd */ if (!check_datestyle(&subval, &subextra, source))
/* when doit is true */
free((char *) subval);
}
else
{ {
ereport(GUC_complaint_elevel(source), free(subval);
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("unrecognized \"datestyle\" key word: \"%s\"",
tok)));
ok = false; ok = false;
break; break;
} }
myextra = (int *) subextra;
if (!have_style)
newDateStyle = myextra[0];
if (!have_order)
newDateOrder = myextra[1];
free(subval);
free(subextra);
}
else
{
GUC_check_errdetail("Unrecognized key word: \"%s\".", tok);
pfree(rawstring);
list_free(elemlist);
return false;
}
} }
pfree(rawstring); pfree(rawstring);
...@@ -170,24 +169,16 @@ assign_datestyle(const char *value, bool doit, GucSource source) ...@@ -170,24 +169,16 @@ assign_datestyle(const char *value, bool doit, GucSource source)
if (!ok) if (!ok)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errdetail("Conflicting \"datestyle\" specifications.");
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), return false;
errmsg("conflicting \"datestyle\" specifications")));
return NULL;
} }
/*
* If we aren't going to do the assignment, just return OK indicator.
*/
if (!doit)
return value;
/* /*
* Prepare the canonical string to return. GUC wants it malloc'd. * Prepare the canonical string to return. GUC wants it malloc'd.
*/ */
result = (char *) malloc(32); result = (char *) malloc(32);
if (!result) if (!result)
return NULL; return false;
switch (newDateStyle) switch (newDateStyle)
{ {
...@@ -217,14 +208,32 @@ assign_datestyle(const char *value, bool doit, GucSource source) ...@@ -217,14 +208,32 @@ assign_datestyle(const char *value, bool doit, GucSource source)
break; break;
} }
free(*newval);
*newval = result;
/* /*
* Finally, it's safe to assign to the global variables; the assignment * Set up the "extra" struct actually used by assign_datestyle.
* cannot fail now. */
myextra = (int *) malloc(2 * sizeof(int));
if (!myextra)
return false;
myextra[0] = newDateStyle;
myextra[1] = newDateOrder;
*extra = (void *) myextra;
return true;
}
/*
* assign_datestyle: GUC assign_hook for datestyle
*/ */
DateStyle = newDateStyle; void
DateOrder = newDateOrder; assign_datestyle(const char *newval, void *extra)
{
int *myextra = (int *) extra;
return result; DateStyle = myextra[0];
DateOrder = myextra[1];
} }
...@@ -232,22 +241,58 @@ assign_datestyle(const char *value, bool doit, GucSource source) ...@@ -232,22 +241,58 @@ assign_datestyle(const char *value, bool doit, GucSource source)
* TIMEZONE * TIMEZONE
*/ */
typedef struct
{
pg_tz *session_timezone;
int CTimeZone;
bool HasCTZSet;
} timezone_extra;
/* /*
* assign_timezone: GUC assign_hook for timezone * check_timezone: GUC check_hook for timezone
*/ */
const char * bool
assign_timezone(const char *value, bool doit, GucSource source) check_timezone(char **newval, void **extra, GucSource source)
{ {
char *result; timezone_extra myextra;
char *endptr; char *endptr;
double hours; double hours;
if (*newval == NULL)
{
/*
* The boot_val given for TimeZone in guc.c is NULL. When we see this
* we just do nothing. If this isn't overridden from the config file
* then pg_timezone_initialize() will eventually select a default
* value from the environment. This hack has two purposes: to avoid
* wasting cycles loading values that might soon be overridden from
* the config file, and to avoid trying to read the timezone files
* during InitializeGUCOptions(). The latter doesn't work in an
* EXEC_BACKEND subprocess because my_exec_path hasn't been set yet
* and so we can't locate PGSHAREDIR.
*/
Assert(source == PGC_S_DEFAULT);
return true;
}
/* /*
* Check for INTERVAL 'foo' * Initialize the "extra" struct that will be passed to assign_timezone.
* We don't want to change any of the three global variables except as
* specified by logic below. To avoid leaking memory during failure
* returns, we set up the struct contents in a local variable, and only
* copy it to *extra at the end.
*/ */
if (pg_strncasecmp(value, "interval", 8) == 0) myextra.session_timezone = session_timezone;
myextra.CTimeZone = CTimeZone;
myextra.HasCTZSet = HasCTZSet;
if (pg_strncasecmp(*newval, "interval", 8) == 0)
{ {
const char *valueptr = value; /*
* Support INTERVAL 'foo'. This is for SQL spec compliance, not
* because it has any actual real-world usefulness.
*/
const char *valueptr = *newval;
char *val; char *val;
Interval *interval; Interval *interval;
...@@ -255,14 +300,14 @@ assign_timezone(const char *value, bool doit, GucSource source) ...@@ -255,14 +300,14 @@ assign_timezone(const char *value, bool doit, GucSource source)
while (isspace((unsigned char) *valueptr)) while (isspace((unsigned char) *valueptr))
valueptr++; valueptr++;
if (*valueptr++ != '\'') if (*valueptr++ != '\'')
return NULL; return false;
val = pstrdup(valueptr); val = pstrdup(valueptr);
/* Check and remove trailing quote */ /* Check and remove trailing quote */
endptr = strchr(val, '\''); endptr = strchr(val, '\'');
if (!endptr || endptr[1] != '\0') if (!endptr || endptr[1] != '\0')
{ {
pfree(val); pfree(val);
return NULL; return false;
} }
*endptr = '\0'; *endptr = '\0';
...@@ -280,31 +325,25 @@ assign_timezone(const char *value, bool doit, GucSource source) ...@@ -280,31 +325,25 @@ assign_timezone(const char *value, bool doit, GucSource source)
pfree(val); pfree(val);
if (interval->month != 0) if (interval->month != 0)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errdetail("Cannot specify months in time zone interval.");
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("invalid interval value for time zone: month not allowed")));
pfree(interval); pfree(interval);
return NULL; return false;
} }
if (interval->day != 0) if (interval->day != 0)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errdetail("Cannot specify days in time zone interval.");
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("invalid interval value for time zone: day not allowed")));
pfree(interval); pfree(interval);
return NULL; return false;
} }
if (doit)
{
/* Here we change from SQL to Unix sign convention */ /* Here we change from SQL to Unix sign convention */
#ifdef HAVE_INT64_TIMESTAMP #ifdef HAVE_INT64_TIMESTAMP
CTimeZone = -(interval->time / USECS_PER_SEC); myextra.CTimeZone = -(interval->time / USECS_PER_SEC);
#else #else
CTimeZone = -interval->time; myextra.CTimeZone = -interval->time;
#endif #endif
myextra.HasCTZSet = true;
HasCTZSet = true;
}
pfree(interval); pfree(interval);
} }
else else
...@@ -312,38 +351,12 @@ assign_timezone(const char *value, bool doit, GucSource source) ...@@ -312,38 +351,12 @@ assign_timezone(const char *value, bool doit, GucSource source)
/* /*
* Try it as a numeric number of hours (possibly fractional). * Try it as a numeric number of hours (possibly fractional).
*/ */
hours = strtod(value, &endptr); hours = strtod(*newval, &endptr);
if (endptr != value && *endptr == '\0') if (endptr != *newval && *endptr == '\0')
{
if (doit)
{ {
/* Here we change from SQL to Unix sign convention */ /* Here we change from SQL to Unix sign convention */
CTimeZone = -hours * SECS_PER_HOUR; myextra.CTimeZone = -hours * SECS_PER_HOUR;
HasCTZSet = true; myextra.HasCTZSet = true;
}
}
else if (pg_strcasecmp(value, "UNKNOWN") == 0)
{
/*
* UNKNOWN is the value shown as the "default" for TimeZone in
* guc.c. We interpret it as being a complete no-op; we don't
* change the timezone setting. Note that if there is a known
* timezone setting, we will return that name rather than UNKNOWN
* as the canonical spelling.
*
* During GUC initialization, since the timezone library isn't set
* up yet, pg_get_timezone_name will return NULL and we will leave
* the setting as UNKNOWN. If this isn't overridden from the
* config file then pg_timezone_initialize() will eventually
* select a default value from the environment.
*/
if (doit)
{
const char *curzone = pg_get_timezone_name(session_timezone);
if (curzone)
value = curzone;
}
} }
else else
{ {
...@@ -352,61 +365,83 @@ assign_timezone(const char *value, bool doit, GucSource source) ...@@ -352,61 +365,83 @@ assign_timezone(const char *value, bool doit, GucSource source)
*/ */
pg_tz *new_tz; pg_tz *new_tz;
new_tz = pg_tzset(value); new_tz = pg_tzset(*newval);
if (!new_tz) if (!new_tz)
{ {
ereport(GUC_complaint_elevel(source), /* Doesn't seem to be any great value in errdetail here */
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), return false;
errmsg("unrecognized time zone name: \"%s\"",
value)));
return NULL;
} }
if (!tz_acceptable(new_tz)) if (!tz_acceptable(new_tz))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errmsg("time zone \"%s\" appears to use leap seconds",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), *newval);
errmsg("time zone \"%s\" appears to use leap seconds", GUC_check_errdetail("PostgreSQL does not support leap seconds.");
value), return false;
errdetail("PostgreSQL does not support leap seconds.")));
return NULL;
} }
if (doit) myextra.session_timezone = new_tz;
{ myextra.HasCTZSet = false;
/* Save the changed TZ */
session_timezone = new_tz;
HasCTZSet = false;
}
} }
} }
/*
* If we aren't going to do the assignment, just return OK indicator.
*/
if (!doit)
return value;
/* /*
* Prepare the canonical string to return. GUC wants it malloc'd. * Prepare the canonical string to return. GUC wants it malloc'd.
*
* Note: the result string should be something that we'd accept as input.
* We use the numeric format for interval cases, because it's simpler to
* reload. In the named-timezone case, *newval is already OK and need not
* be changed; it might not have the canonical casing, but that's taken
* care of by show_timezone.
*/ */
if (HasCTZSet) if (myextra.HasCTZSet)
{ {
result = (char *) malloc(64); char *result = (char *) malloc(64);
if (!result) if (!result)
return NULL; return false;
snprintf(result, 64, "%.5f", snprintf(result, 64, "%.5f",
(double) (-CTimeZone) / (double) SECS_PER_HOUR); (double) (-myextra.CTimeZone) / (double) SECS_PER_HOUR);
free(*newval);
*newval = result;
} }
else
result = strdup(value);
return result; /*
* Pass back data for assign_timezone to use
*/
*extra = malloc(sizeof(timezone_extra));
if (!*extra)
return false;
memcpy(*extra, &myextra, sizeof(timezone_extra));
return true;
}
/*
* assign_timezone: GUC assign_hook for timezone
*/
void
assign_timezone(const char *newval, void *extra)
{
timezone_extra *myextra = (timezone_extra *) extra;
/* Do nothing for the boot_val default of NULL */
if (!myextra)
return;
session_timezone = myextra->session_timezone;
CTimeZone = myextra->CTimeZone;
HasCTZSet = myextra->HasCTZSet;
} }
/* /*
* show_timezone: GUC show_hook for timezone * show_timezone: GUC show_hook for timezone
*
* We wouldn't need this, except that historically interval values have been
* shown without an INTERVAL prefix, so the display format isn't what would
* be accepted as input. Otherwise we could have check_timezone return the
* preferred string to begin with.
*/ */
const char * const char *
show_timezone(void) show_timezone(void)
...@@ -447,83 +482,66 @@ show_timezone(void) ...@@ -447,83 +482,66 @@ show_timezone(void)
*/ */
/* /*
* assign_log_timezone: GUC assign_hook for log_timezone * check_log_timezone: GUC check_hook for log_timezone
*/ */
const char * bool
assign_log_timezone(const char *value, bool doit, GucSource source) check_log_timezone(char **newval, void **extra, GucSource source)
{ {
char *result; pg_tz *new_tz;
if (pg_strcasecmp(value, "UNKNOWN") == 0) if (*newval == NULL)
{ {
/* /*
* UNKNOWN is the value shown as the "default" for log_timezone in * The boot_val given for log_timezone in guc.c is NULL. When we see
* guc.c. We interpret it as being a complete no-op; we don't change * this we just do nothing. If this isn't overridden from the config
* the timezone setting. Note that if there is a known timezone * file then pg_timezone_initialize() will eventually select a default
* setting, we will return that name rather than UNKNOWN as the
* canonical spelling.
*
* During GUC initialization, since the timezone library isn't set up
* yet, pg_get_timezone_name will return NULL and we will leave the
* setting as UNKNOWN. If this isn't overridden from the config file
* then pg_timezone_initialize() will eventually select a default
* value from the environment. * value from the environment.
*/ */
if (doit) Assert(source == PGC_S_DEFAULT);
{ return true;
const char *curzone = pg_get_timezone_name(log_timezone);
if (curzone)
value = curzone;
}
} }
else
{
/* /*
* Otherwise assume it is a timezone name, and try to load it. * Otherwise assume it is a timezone name, and try to load it.
*/ */
pg_tz *new_tz; new_tz = pg_tzset(*newval);
new_tz = pg_tzset(value);
if (!new_tz) if (!new_tz)
{ {
ereport(GUC_complaint_elevel(source), /* Doesn't seem to be any great value in errdetail here */
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), return false;
errmsg("unrecognized time zone name: \"%s\"",
value)));
return NULL;
} }
if (!tz_acceptable(new_tz)) if (!tz_acceptable(new_tz))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errmsg("time zone \"%s\" appears to use leap seconds",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), *newval);
errmsg("time zone \"%s\" appears to use leap seconds", GUC_check_errdetail("PostgreSQL does not support leap seconds.");
value), return false;
errdetail("PostgreSQL does not support leap seconds.")));
return NULL;
}
if (doit)
{
/* Save the changed TZ */
log_timezone = new_tz;
}
} }
/* /*
* If we aren't going to do the assignment, just return OK indicator. * Pass back data for assign_log_timezone to use
*/ */
if (!doit) *extra = malloc(sizeof(pg_tz *));
return value; if (!*extra)
return false;
memcpy(*extra, &new_tz, sizeof(pg_tz *));
/* return true;
* Prepare the canonical string to return. GUC wants it malloc'd. }
/*
* assign_log_timezone: GUC assign_hook for log_timezone
*/ */
result = strdup(value); void
assign_log_timezone(const char *newval, void *extra)
{
/* Do nothing for the boot_val default of NULL */
if (!extra)
return;
return result; log_timezone = *((pg_tz **) extra);
} }
/* /*
...@@ -548,38 +566,33 @@ show_log_timezone(void) ...@@ -548,38 +566,33 @@ show_log_timezone(void)
* *
* We allow idempotent changes (r/w -> r/w and r/o -> r/o) at any time, and * We allow idempotent changes (r/w -> r/w and r/o -> r/o) at any time, and
* we also always allow changes from read-write to read-only. However, * we also always allow changes from read-write to read-only. However,
* read-only to read-write may be changed only when source == PGC_S_OVERRIDE * read-only may be changed to read-write only when in a top-level transaction
* (i.e. we're aborting a read only transaction and restoring the previous * that has not yet taken an initial snapshot. Can't do it in a hot standby
* setting) or in a top-level transaction that has not yet taken an initial * slave, either.
* snapshot.
*/ */
bool bool
assign_transaction_read_only(bool newval, bool doit, GucSource source) check_transaction_read_only(bool *newval, void **extra, GucSource source)
{ {
if (source != PGC_S_OVERRIDE && newval == false && XactReadOnly) if (*newval == false && XactReadOnly)
{ {
/* Can't go to r/w mode inside a r/o transaction */ /* Can't go to r/w mode inside a r/o transaction */
if (IsSubTransaction()) if (IsSubTransaction())
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), GUC_check_errmsg("cannot set transaction read-write mode inside a read-only transaction");
errmsg("cannot set transaction read-write mode inside a read-only transaction")));
return false; return false;
} }
/* Top level transaction can't change to r/w after first snapshot. */ /* Top level transaction can't change to r/w after first snapshot. */
if (FirstSnapshotSet) if (FirstSnapshotSet)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
(errcode(ERRCODE_ACTIVE_SQL_TRANSACTION), GUC_check_errmsg("transaction read-write mode must be set before any query");
errmsg("transaction read-write mode must be set before any query")));
return false; return false;
} }
/* Can't go to r/w mode while recovery is still active */ /* Can't go to r/w mode while recovery is still active */
if (RecoveryInProgress()) if (RecoveryInProgress())
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errmsg("cannot set transaction read-write mode during recovery");
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("cannot set transaction read-write mode during recovery")));
return false; return false;
} }
} }
...@@ -591,76 +604,78 @@ assign_transaction_read_only(bool newval, bool doit, GucSource source) ...@@ -591,76 +604,78 @@ assign_transaction_read_only(bool newval, bool doit, GucSource source)
* SET TRANSACTION ISOLATION LEVEL * SET TRANSACTION ISOLATION LEVEL
* *
* We allow idempotent changes at any time, but otherwise this can only be * We allow idempotent changes at any time, but otherwise this can only be
* changed from a toplevel transaction that has not yet taken a snapshot, or * changed in a toplevel transaction that has not yet taken a snapshot.
* when source == PGC_S_OVERRIDE (i.e. we're aborting a transaction and
* restoring the previously set value).
*/ */
const char * bool
assign_XactIsoLevel(const char *value, bool doit, GucSource source) check_XactIsoLevel(char **newval, void **extra, GucSource source)
{ {
/* source == PGC_S_OVERRIDE means do it anyway, eg at xact abort */ int newXactIsoLevel;
if (source != PGC_S_OVERRIDE && strcmp(value, XactIsoLevel_string) != 0)
if (strcmp(*newval, "serializable") == 0)
{ {
if (FirstSnapshotSet) newXactIsoLevel = XACT_SERIALIZABLE;
}
else if (strcmp(*newval, "repeatable read") == 0)
{ {
ereport(GUC_complaint_elevel(source), newXactIsoLevel = XACT_REPEATABLE_READ;
(errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
errmsg("SET TRANSACTION ISOLATION LEVEL must be called before any query")));
return NULL;
} }
/* We ignore a subtransaction setting it to the existing value. */ else if (strcmp(*newval, "read committed") == 0)
if (IsSubTransaction())
{ {
ereport(GUC_complaint_elevel(source), newXactIsoLevel = XACT_READ_COMMITTED;
(errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
errmsg("SET TRANSACTION ISOLATION LEVEL must not be called in a subtransaction")));
return NULL;
} }
/* Can't go to serializable mode while recovery is still active */ else if (strcmp(*newval, "read uncommitted") == 0)
if (RecoveryInProgress() && strcmp(value, "serializable") == 0)
{ {
ereport(GUC_complaint_elevel(source), newXactIsoLevel = XACT_READ_UNCOMMITTED;
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("cannot use serializable mode in a hot standby"),
errhint("You can use REPEATABLE READ instead.")));
return false;
} }
else if (strcmp(*newval, "default") == 0)
{
newXactIsoLevel = DefaultXactIsoLevel;
} }
else
return false;
if (strcmp(value, "serializable") == 0) if (newXactIsoLevel != XactIsoLevel)
{ {
if (doit) if (FirstSnapshotSet)
XactIsoLevel = XACT_SERIALIZABLE;
}
else if (strcmp(value, "repeatable read") == 0)
{ {
if (doit) GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
XactIsoLevel = XACT_REPEATABLE_READ; GUC_check_errmsg("SET TRANSACTION ISOLATION LEVEL must be called before any query");
return false;
} }
else if (strcmp(value, "read committed") == 0) /* We ignore a subtransaction setting it to the existing value. */
if (IsSubTransaction())
{ {
if (doit) GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
XactIsoLevel = XACT_READ_COMMITTED; GUC_check_errmsg("SET TRANSACTION ISOLATION LEVEL must not be called in a subtransaction");
return false;
} }
else if (strcmp(value, "read uncommitted") == 0) /* Can't go to serializable mode while recovery is still active */
if (newXactIsoLevel == XACT_SERIALIZABLE && RecoveryInProgress())
{ {
if (doit) GUC_check_errmsg("cannot use serializable mode in a hot standby");
XactIsoLevel = XACT_READ_UNCOMMITTED; GUC_check_errhint("You can use REPEATABLE READ instead.");
return false;
} }
else if (strcmp(value, "default") == 0)
{
if (doit)
XactIsoLevel = DefaultXactIsoLevel;
} }
else
return NULL;
return value; *extra = malloc(sizeof(int));
if (!*extra)
return false;
*((int *) *extra) = newXactIsoLevel;
return true;
}
void
assign_XactIsoLevel(const char *newval, void *extra)
{
XactIsoLevel = *((int *) extra);
} }
const char * const char *
show_XactIsoLevel(void) show_XactIsoLevel(void)
{ {
/* We need this because we don't want to show "default". */
switch (XactIsoLevel) switch (XactIsoLevel)
{ {
case XACT_READ_UNCOMMITTED: case XACT_READ_UNCOMMITTED:
...@@ -681,25 +696,18 @@ show_XactIsoLevel(void) ...@@ -681,25 +696,18 @@ show_XactIsoLevel(void)
*/ */
bool bool
assign_transaction_deferrable(bool newval, bool doit, GucSource source) check_transaction_deferrable(bool *newval, void **extra, GucSource source)
{ {
/* source == PGC_S_OVERRIDE means do it anyway, eg at xact abort */
if (source == PGC_S_OVERRIDE)
return true;
if (IsSubTransaction()) if (IsSubTransaction())
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
(errcode(ERRCODE_ACTIVE_SQL_TRANSACTION), GUC_check_errmsg("SET TRANSACTION [NOT] DEFERRABLE cannot be called within a subtransaction");
errmsg("SET TRANSACTION [NOT] DEFERRABLE cannot be called within a subtransaction")));
return false; return false;
} }
if (FirstSnapshotSet) if (FirstSnapshotSet)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errcode(ERRCODE_ACTIVE_SQL_TRANSACTION);
(errcode(ERRCODE_ACTIVE_SQL_TRANSACTION), GUC_check_errmsg("SET TRANSACTION [NOT] DEFERRABLE must be called before any query");
errmsg("SET TRANSACTION [NOT] DEFERRABLE must be called before any query")));
return false; return false;
} }
...@@ -708,17 +716,34 @@ assign_transaction_deferrable(bool newval, bool doit, GucSource source) ...@@ -708,17 +716,34 @@ assign_transaction_deferrable(bool newval, bool doit, GucSource source)
/* /*
* Random number seed * Random number seed
*
* We can't roll back the random sequence on error, and we don't want
* config file reloads to affect it, so we only want interactive SET SEED
* commands to set it. We use the "extra" storage to ensure that rollbacks
* don't try to do the operation again.
*/ */
bool bool
assign_random_seed(double value, bool doit, GucSource source) check_random_seed(double *newval, void **extra, GucSource source)
{ {
/* Can't really roll back on error, so ignore non-interactive setting */ *extra = malloc(sizeof(int));
if (doit && source >= PGC_S_INTERACTIVE) if (!*extra)
DirectFunctionCall1(setseed, Float8GetDatum(value)); return false;
/* Arm the assign only if source of value is an interactive SET */
*((int *) *extra) = (source >= PGC_S_INTERACTIVE);
return true; return true;
} }
void
assign_random_seed(double newval, void *extra)
{
/* We'll do this at most once for any setting of the GUC variable */
if (*((int *) extra))
DirectFunctionCall1(setseed, Float8GetDatum(newval));
*((int *) extra) = 0;
}
const char * const char *
show_random_seed(void) show_random_seed(void)
{ {
...@@ -727,214 +752,189 @@ show_random_seed(void) ...@@ -727,214 +752,189 @@ show_random_seed(void)
/* /*
* encoding handling functions * SET CLIENT_ENCODING
*/ */
const char * bool
assign_client_encoding(const char *value, bool doit, GucSource source) check_client_encoding(char **newval, void **extra, GucSource source)
{ {
int encoding; int encoding;
encoding = pg_valid_client_encoding(value); /* Look up the encoding by name */
encoding = pg_valid_client_encoding(*newval);
if (encoding < 0) if (encoding < 0)
return NULL; return false;
/* /*
* Note: if we are in startup phase then SetClientEncoding may not be able * If we are not within a transaction then PrepareClientEncoding will not
* to really set the encoding. In this case we will assume that the * be able to look up the necessary conversion procs. If we are still
* encoding is okay, and InitializeClientEncoding() will fix things once * starting up, it will return "OK" anyway, and InitializeClientEncoding
* initialization is complete. * will fix things once initialization is far enough along. After
* startup, we'll fail. This would only happen if someone tries to change
* client_encoding in postgresql.conf and then SIGHUP existing sessions.
* It seems like a bad idea for client_encoding to change that way anyhow,
* so we don't go out of our way to support it.
*
* Note: in the postmaster, or any other process that never calls
* InitializeClientEncoding, PrepareClientEncoding will always succeed,
* and so will SetClientEncoding; but they won't do anything, which is OK.
*/ */
if (SetClientEncoding(encoding, doit) < 0) if (PrepareClientEncoding(encoding) < 0)
{ {
ereport(GUC_complaint_elevel(source), if (IsTransactionState())
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED), {
errmsg("conversion between %s and %s is not supported", /* Must be a genuine no-such-conversion problem */
value, GetDatabaseEncodingName()))); GUC_check_errcode(ERRCODE_FEATURE_NOT_SUPPORTED);
return NULL; GUC_check_errdetail("Conversion between %s and %s is not supported.",
pg_encoding_to_char(encoding),
GetDatabaseEncodingName());
}
else
{
/* Provide a useful complaint */
GUC_check_errdetail("Cannot change \"client_encoding\" now.");
}
return false;
} }
return value;
}
/*
* Return the encoding's canonical name, and save its ID in *extra.
*/
free(*newval);
*newval = strdup(pg_encoding_to_char(encoding));
if (!*newval)
return false;
/* *extra = malloc(sizeof(int));
* SET SESSION AUTHORIZATION if (!*extra)
* return false;
* When resetting session auth after an error, we can't expect to do catalog *((int *) *extra) = encoding;
* lookups. Hence, the stored form of the value must provide a numeric oid
* that can be re-used directly. We store the string in the form of
* NAMEDATALEN 'x's, followed by T or F to indicate superuserness, followed
* by the numeric oid, followed by a comma, followed by the role name.
* This cannot be confused with a plain role name because of the NAMEDATALEN
* limit on names, so we can tell whether we're being passed an initial
* role name or a saved/restored value. (NOTE: we rely on guc.c to have
* properly truncated any incoming value, but not to truncate already-stored
* values. See GUC_IS_NAME processing.)
*/
extern char *session_authorization_string; /* in guc.c */
const char * return true;
assign_session_authorization(const char *value, bool doit, GucSource source) }
void
assign_client_encoding(const char *newval, void *extra)
{ {
Oid roleid = InvalidOid; int encoding = *((int *) extra);
bool is_superuser = false;
const char *actual_rolename = NULL;
char *result;
if (strspn(value, "x") == NAMEDATALEN && /* We do not expect an error if PrepareClientEncoding succeeded */
(value[NAMEDATALEN] == 'T' || value[NAMEDATALEN] == 'F')) if (SetClientEncoding(encoding) < 0)
{ elog(LOG, "SetClientEncoding(%d) failed", encoding);
/* might be a saved userid string */ }
Oid savedoid;
char *endptr;
savedoid = (Oid) strtoul(value + NAMEDATALEN + 1, &endptr, 10);
if (endptr != value + NAMEDATALEN + 1 && *endptr == ',') /*
{ * SET SESSION AUTHORIZATION
/* syntactically valid, so break out the data */ */
roleid = savedoid;
is_superuser = (value[NAMEDATALEN] == 'T');
actual_rolename = endptr + 1;
}
}
if (roleid == InvalidOid) typedef struct
{ {
/* not a saved ID, so look it up */ /* This is the "extra" state for both SESSION AUTHORIZATION and ROLE */
Oid roleid;
bool is_superuser;
} role_auth_extra;
bool
check_session_authorization(char **newval, void **extra, GucSource source)
{
HeapTuple roleTup; HeapTuple roleTup;
Oid roleid;
bool is_superuser;
role_auth_extra *myextra;
/* Do nothing for the boot_val default of NULL */
if (*newval == NULL)
return true;
if (!IsTransactionState()) if (!IsTransactionState())
{ {
/* /*
* Can't do catalog lookups, so fail. The upshot of this is that * Can't do catalog lookups, so fail. The result of this is that
* session_authorization cannot be set in postgresql.conf, which * session_authorization cannot be set in postgresql.conf, which
* seems like a good thing anyway. * seems like a good thing anyway, so we don't work hard to avoid it.
*/ */
return NULL; return false;
} }
roleTup = SearchSysCache1(AUTHNAME, PointerGetDatum(value)); /* Look up the username */
roleTup = SearchSysCache1(AUTHNAME, PointerGetDatum(*newval));
if (!HeapTupleIsValid(roleTup)) if (!HeapTupleIsValid(roleTup))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errmsg("role \"%s\" does not exist", *newval);
(errcode(ERRCODE_UNDEFINED_OBJECT), return false;
errmsg("role \"%s\" does not exist", value)));
return NULL;
} }
roleid = HeapTupleGetOid(roleTup); roleid = HeapTupleGetOid(roleTup);
is_superuser = ((Form_pg_authid) GETSTRUCT(roleTup))->rolsuper; is_superuser = ((Form_pg_authid) GETSTRUCT(roleTup))->rolsuper;
actual_rolename = value;
ReleaseSysCache(roleTup); ReleaseSysCache(roleTup);
}
if (doit) /* Set up "extra" struct for assign_session_authorization to use */
SetSessionAuthorization(roleid, is_superuser); myextra = (role_auth_extra *) malloc(sizeof(role_auth_extra));
if (!myextra)
result = (char *) malloc(NAMEDATALEN + 32 + strlen(actual_rolename)); return false;
if (!result) myextra->roleid = roleid;
return NULL; myextra->is_superuser = is_superuser;
*extra = (void *) myextra;
memset(result, 'x', NAMEDATALEN);
sprintf(result + NAMEDATALEN, "%c%u,%s",
is_superuser ? 'T' : 'F',
roleid,
actual_rolename);
return result; return true;
} }
const char * void
show_session_authorization(void) assign_session_authorization(const char *newval, void *extra)
{ {
/* role_auth_extra *myextra = (role_auth_extra *) extra;
* Extract the user name from the stored string; see
* assign_session_authorization
*/
const char *value = session_authorization_string;
Oid savedoid;
char *endptr;
/* If session_authorization hasn't been set in this process, return "" */
if (value == NULL || value[0] == '\0')
return "";
Assert(strspn(value, "x") == NAMEDATALEN &&
(value[NAMEDATALEN] == 'T' || value[NAMEDATALEN] == 'F'));
savedoid = (Oid) strtoul(value + NAMEDATALEN + 1, &endptr, 10); /* Do nothing for the boot_val default of NULL */
if (!myextra)
return;
Assert(endptr != value + NAMEDATALEN + 1 && *endptr == ','); SetSessionAuthorization(myextra->roleid, myextra->is_superuser);
return endptr + 1;
} }
/* /*
* SET ROLE * SET ROLE
* *
* When resetting session auth after an error, we can't expect to do catalog
* lookups. Hence, the stored form of the value must provide a numeric oid
* that can be re-used directly. We implement this exactly like SET
* SESSION AUTHORIZATION.
*
* The SQL spec requires "SET ROLE NONE" to unset the role, so we hardwire * The SQL spec requires "SET ROLE NONE" to unset the role, so we hardwire
* a translation of "none" to InvalidOid. * a translation of "none" to InvalidOid. Otherwise this is much like
* SET SESSION AUTHORIZATION.
*/ */
extern char *role_string; /* in guc.c */ extern char *role_string; /* in guc.c */
const char * bool
assign_role(const char *value, bool doit, GucSource source) check_role(char **newval, void **extra, GucSource source)
{ {
Oid roleid = InvalidOid; HeapTuple roleTup;
bool is_superuser = false; Oid roleid;
const char *actual_rolename = value; bool is_superuser;
char *result; role_auth_extra *myextra;
if (strspn(value, "x") == NAMEDATALEN &&
(value[NAMEDATALEN] == 'T' || value[NAMEDATALEN] == 'F'))
{
/* might be a saved userid string */
Oid savedoid;
char *endptr;
savedoid = (Oid) strtoul(value + NAMEDATALEN + 1, &endptr, 10);
if (endptr != value + NAMEDATALEN + 1 && *endptr == ',') if (strcmp(*newval, "none") == 0)
{ {
/* syntactically valid, so break out the data */ /* hardwired translation */
roleid = savedoid; roleid = InvalidOid;
is_superuser = (value[NAMEDATALEN] == 'T'); is_superuser = false;
actual_rolename = endptr + 1;
}
} }
else
if (roleid == InvalidOid &&
strcmp(actual_rolename, "none") != 0)
{ {
/* not a saved ID, so look it up */
HeapTuple roleTup;
if (!IsTransactionState()) if (!IsTransactionState())
{ {
/* /*
* Can't do catalog lookups, so fail. The upshot of this is that * Can't do catalog lookups, so fail. The result of this is that
* role cannot be set in postgresql.conf, which seems like a good * role cannot be set in postgresql.conf, which seems like a good
* thing anyway. * thing anyway, so we don't work hard to avoid it.
*/ */
return NULL; return false;
} }
roleTup = SearchSysCache1(AUTHNAME, PointerGetDatum(value)); /* Look up the username */
roleTup = SearchSysCache1(AUTHNAME, PointerGetDatum(*newval));
if (!HeapTupleIsValid(roleTup)) if (!HeapTupleIsValid(roleTup))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errmsg("role \"%s\" does not exist", *newval);
(errcode(ERRCODE_UNDEFINED_OBJECT), return false;
errmsg("role \"%s\" does not exist", value)));
return NULL;
} }
roleid = HeapTupleGetOid(roleTup); roleid = HeapTupleGetOid(roleTup);
...@@ -947,61 +947,45 @@ assign_role(const char *value, bool doit, GucSource source) ...@@ -947,61 +947,45 @@ assign_role(const char *value, bool doit, GucSource source)
*/ */
if (!is_member_of_role(GetSessionUserId(), roleid)) if (!is_member_of_role(GetSessionUserId(), roleid))
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errcode(ERRCODE_INSUFFICIENT_PRIVILEGE);
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), GUC_check_errmsg("permission denied to set role \"%s\"",
errmsg("permission denied to set role \"%s\"", *newval);
value))); return false;
return NULL;
} }
} }
if (doit) /* Set up "extra" struct for assign_role to use */
SetCurrentRoleId(roleid, is_superuser); myextra = (role_auth_extra *) malloc(sizeof(role_auth_extra));
if (!myextra)
result = (char *) malloc(NAMEDATALEN + 32 + strlen(actual_rolename)); return false;
if (!result) myextra->roleid = roleid;
return NULL; myextra->is_superuser = is_superuser;
*extra = (void *) myextra;
memset(result, 'x', NAMEDATALEN); return true;
}
sprintf(result + NAMEDATALEN, "%c%u,%s", void
is_superuser ? 'T' : 'F', assign_role(const char *newval, void *extra)
roleid, {
actual_rolename); role_auth_extra *myextra = (role_auth_extra *) extra;
return result; SetCurrentRoleId(myextra->roleid, myextra->is_superuser);
} }
const char * const char *
show_role(void) show_role(void)
{ {
/* /*
* Extract the role name from the stored string; see assign_role * Check whether SET ROLE is active; if not return "none". This is a
*/ * kluge to deal with the fact that SET SESSION AUTHORIZATION logically
const char *value = role_string; * resets SET ROLE to NONE, but we cannot set the GUC role variable from
Oid savedoid; * assign_session_authorization (because we haven't got enough info to
char *endptr; * call set_config_option).
/* This special case only applies if no SET ROLE has been done */
if (value == NULL || strcmp(value, "none") == 0)
return "none";
Assert(strspn(value, "x") == NAMEDATALEN &&
(value[NAMEDATALEN] == 'T' || value[NAMEDATALEN] == 'F'));
savedoid = (Oid) strtoul(value + NAMEDATALEN + 1, &endptr, 10);
Assert(endptr != value + NAMEDATALEN + 1 && *endptr == ',');
/*
* Check that the stored string still matches the effective setting, else
* return "none". This is a kluge to deal with the fact that SET SESSION
* AUTHORIZATION logically resets SET ROLE to NONE, but we cannot set the
* GUC role variable from assign_session_authorization (because we haven't
* got enough info to call set_config_option).
*/ */
if (savedoid != GetCurrentRoleId()) if (!OidIsValid(GetCurrentRoleId()))
return "none"; return "none";
return endptr + 1; /* Otherwise we can just use the GUC string */
return role_string ? role_string : "none";
} }
...@@ -2,7 +2,10 @@ ...@@ -2,7 +2,10 @@
CATALOG_NAME := postgres CATALOG_NAME := postgres
AVAIL_LANGUAGES := de es fr ja pt_BR tr AVAIL_LANGUAGES := de es fr ja pt_BR tr
GETTEXT_FILES := + gettext-files GETTEXT_FILES := + gettext-files
GETTEXT_TRIGGERS:= _ errmsg errmsg_plural:1,2 errdetail errdetail_log errdetail_plural:1,2 errhint errcontext write_stderr yyerror parser_yyerror GETTEXT_TRIGGERS:= _ errmsg errmsg_plural:1,2 errdetail errdetail_log \
errdetail_plural:1,2 errhint errcontext \
GUC_check_errmsg GUC_check_errdetail GUC_check_errhint \
write_stderr yyerror parser_yyerror
gettext-files: distprep gettext-files: distprep
find $(srcdir)/ $(srcdir)/../port/ -name '*.c' -print >$@ find $(srcdir)/ $(srcdir)/../port/ -name '*.c' -print >$@
......
...@@ -639,25 +639,23 @@ SyncRepQueueIsOrderedByLSN(void) ...@@ -639,25 +639,23 @@ SyncRepQueueIsOrderedByLSN(void)
* =========================================================== * ===========================================================
*/ */
const char * bool
assign_synchronous_standby_names(const char *newval, bool doit, GucSource source) check_synchronous_standby_names(char **newval, void **extra, GucSource source)
{ {
char *rawstring; char *rawstring;
List *elemlist; List *elemlist;
/* Need a modifiable copy of string */ /* Need a modifiable copy of string */
rawstring = pstrdup(newval); rawstring = pstrdup(*newval);
/* Parse string into list of identifiers */ /* Parse string into list of identifiers */
if (!SplitIdentifierString(rawstring, ',', &elemlist)) if (!SplitIdentifierString(rawstring, ',', &elemlist))
{ {
/* syntax error in list */ /* syntax error in list */
GUC_check_errdetail("List syntax is invalid.");
pfree(rawstring); pfree(rawstring);
list_free(elemlist); list_free(elemlist);
ereport(GUC_complaint_elevel(source), return false;
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("invalid list syntax for parameter \"synchronous_standby_names\"")));
return NULL;
} }
/* /*
...@@ -671,5 +669,5 @@ assign_synchronous_standby_names(const char *newval, bool doit, GucSource source ...@@ -671,5 +669,5 @@ assign_synchronous_standby_names(const char *newval, bool doit, GucSource source
pfree(rawstring); pfree(rawstring);
list_free(elemlist); list_free(elemlist);
return newval; return true;
} }
...@@ -2804,7 +2804,8 @@ RecoveryConflictInterrupt(ProcSignalReason reason) ...@@ -2804,7 +2804,8 @@ RecoveryConflictInterrupt(ProcSignalReason reason)
break; break;
default: default:
elog(FATAL, "Unknown conflict mode"); elog(FATAL, "unrecognized conflict mode: %d",
(int) reason);
} }
Assert(RecoveryConflictPending && (QueryCancelPending || ProcDiePending)); Assert(RecoveryConflictPending && (QueryCancelPending || ProcDiePending));
...@@ -3062,27 +3063,32 @@ check_stack_depth(void) ...@@ -3062,27 +3063,32 @@ check_stack_depth(void)
#endif /* IA64 */ #endif /* IA64 */
} }
/* GUC assign hook for max_stack_depth */ /* GUC check hook for max_stack_depth */
bool bool
assign_max_stack_depth(int newval, bool doit, GucSource source) check_max_stack_depth(int *newval, void **extra, GucSource source)
{ {
long newval_bytes = newval * 1024L; long newval_bytes = *newval * 1024L;
long stack_rlimit = get_stack_depth_rlimit(); long stack_rlimit = get_stack_depth_rlimit();
if (stack_rlimit > 0 && newval_bytes > stack_rlimit - STACK_DEPTH_SLOP) if (stack_rlimit > 0 && newval_bytes > stack_rlimit - STACK_DEPTH_SLOP)
{ {
ereport(GUC_complaint_elevel(source), GUC_check_errdetail("\"max_stack_depth\" must not exceed %ldkB.",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (stack_rlimit - STACK_DEPTH_SLOP) / 1024L);
errmsg("\"max_stack_depth\" must not exceed %ldkB", GUC_check_errhint("Increase the platform's stack depth limit via \"ulimit -s\" or local equivalent.");
(stack_rlimit - STACK_DEPTH_SLOP) / 1024L),
errhint("Increase the platform's stack depth limit via \"ulimit -s\" or local equivalent.")));
return false; return false;
} }
if (doit)
max_stack_depth_bytes = newval_bytes;
return true; return true;
} }
/* GUC assign hook for max_stack_depth */
void
assign_max_stack_depth(int newval, void *extra)
{
long newval_bytes = newval * 1024L;
max_stack_depth_bytes = newval_bytes;
}
/* /*
* set_debug_options --- apply "-d N" command line option * set_debug_options --- apply "-d N" command line option
......
...@@ -4140,20 +4140,17 @@ CheckDateTokenTables(void) ...@@ -4140,20 +4140,17 @@ CheckDateTokenTables(void)
/* /*
* This function gets called during timezone config file load or reload * This function gets called during timezone config file load or reload
* to create the final array of timezone tokens. The argument array * to create the final array of timezone tokens. The argument array
* is already sorted in name order. This data is in a temporary memory * is already sorted in name order. The data is converted to datetkn
* context and must be copied to somewhere permanent. * format and installed in *tbl, which must be allocated by the caller.
*/ */
void void
InstallTimeZoneAbbrevs(tzEntry *abbrevs, int n) ConvertTimeZoneAbbrevs(TimeZoneAbbrevTable *tbl,
struct tzEntry *abbrevs, int n)
{ {
datetkn *newtbl; datetkn *newtbl = tbl->abbrevs;
int i; int i;
/* tbl->numabbrevs = n;
* Copy the data into TopMemoryContext and convert to datetkn format.
*/
newtbl = (datetkn *) MemoryContextAlloc(TopMemoryContext,
n * sizeof(datetkn));
for (i = 0; i < n; i++) for (i = 0; i < n; i++)
{ {
strncpy(newtbl[i].token, abbrevs[i].abbrev, TOKMAXLEN); strncpy(newtbl[i].token, abbrevs[i].abbrev, TOKMAXLEN);
...@@ -4163,12 +4160,20 @@ InstallTimeZoneAbbrevs(tzEntry *abbrevs, int n) ...@@ -4163,12 +4160,20 @@ InstallTimeZoneAbbrevs(tzEntry *abbrevs, int n)
/* Check the ordering, if testing */ /* Check the ordering, if testing */
Assert(CheckDateTokenTable("timezone offset", newtbl, n)); Assert(CheckDateTokenTable("timezone offset", newtbl, n));
}
/*
* Install a TimeZoneAbbrevTable as the active table.
*
* Caller is responsible that the passed table doesn't go away while in use.
*/
void
InstallTimeZoneAbbrevs(TimeZoneAbbrevTable *tbl)
{
int i;
/* Now safe to replace existing table (if any) */ timezonetktbl = tbl->abbrevs;
if (timezonetktbl) sztimezonetktbl = tbl->numabbrevs;
pfree(timezonetktbl);
timezonetktbl = newtbl;
sztimezonetktbl = n;
/* clear date cache in case it contains any stale timezone names */ /* clear date cache in case it contains any stale timezone names */
for (i = 0; i < MAXDATEFIELDS; i++) for (i = 0; i < MAXDATEFIELDS; i++)
......
...@@ -236,52 +236,53 @@ check_locale(int category, const char *value) ...@@ -236,52 +236,53 @@ check_locale(int category, const char *value)
return ret; return ret;
} }
/* GUC assign hooks */
/* /*
* This is common code for several locale categories. This doesn't * GUC check/assign hooks
* actually set the locale permanently, it only tests if the locale is *
* valid. (See explanation at the top of this file.) * For most locale categories, the assign hook doesn't actually set the locale
* permanently, just reset flags so that the next use will cache the
* appropriate values. (See explanation at the top of this file.)
* *
* Note: we accept value = "" as selecting the postmaster's environment * Note: we accept value = "" as selecting the postmaster's environment
* value, whatever it was (so long as the environment setting is legal). * value, whatever it was (so long as the environment setting is legal).
* This will have been locked down by an earlier call to pg_perm_setlocale. * This will have been locked down by an earlier call to pg_perm_setlocale.
*/ */
static const char * bool
locale_xxx_assign(int category, const char *value, bool doit, GucSource source) check_locale_monetary(char **newval, void **extra, GucSource source)
{ {
if (!check_locale(category, value)) return check_locale(LC_MONETARY, *newval);
value = NULL; /* set failure return marker */ }
/* need to reload cache next time? */ void
if (doit && value != NULL) assign_locale_monetary(const char *newval, void *extra)
{ {
CurrentLocaleConvValid = false; CurrentLocaleConvValid = false;
CurrentLCTimeValid = false;
}
return value;
} }
bool
const char * check_locale_numeric(char **newval, void **extra, GucSource source)
locale_monetary_assign(const char *value, bool doit, GucSource source)
{ {
return locale_xxx_assign(LC_MONETARY, value, doit, source); return check_locale(LC_NUMERIC, *newval);
} }
const char * void
locale_numeric_assign(const char *value, bool doit, GucSource source) assign_locale_numeric(const char *newval, void *extra)
{ {
return locale_xxx_assign(LC_NUMERIC, value, doit, source); CurrentLocaleConvValid = false;
} }
const char * bool
locale_time_assign(const char *value, bool doit, GucSource source) check_locale_time(char **newval, void **extra, GucSource source)
{ {
return locale_xxx_assign(LC_TIME, value, doit, source); return check_locale(LC_TIME, *newval);
} }
void
assign_locale_time(const char *newval, void *extra)
{
CurrentLCTimeValid = false;
}
/* /*
* We allow LC_MESSAGES to actually be set globally. * We allow LC_MESSAGES to actually be set globally.
...@@ -293,31 +294,39 @@ locale_time_assign(const char *value, bool doit, GucSource source) ...@@ -293,31 +294,39 @@ locale_time_assign(const char *value, bool doit, GucSource source)
* The idea there is just to accept the environment setting *if possible* * The idea there is just to accept the environment setting *if possible*
* during startup, until we can read the proper value from postgresql.conf. * during startup, until we can read the proper value from postgresql.conf.
*/ */
const char * bool
locale_messages_assign(const char *value, bool doit, GucSource source) check_locale_messages(char **newval, void **extra, GucSource source)
{ {
if (*value == '\0' && source != PGC_S_DEFAULT) if (**newval == '\0')
return NULL; {
if (source == PGC_S_DEFAULT)
return true;
else
return false;
}
/* /*
* LC_MESSAGES category does not exist everywhere, but accept it anyway * LC_MESSAGES category does not exist everywhere, but accept it anyway
* *
* On Windows, we can't even check the value, so the non-doit case is a * On Windows, we can't even check the value, so accept blindly
* no-op */
#if defined(LC_MESSAGES) && !defined(WIN32)
return check_locale(LC_MESSAGES, *newval);
#else
return true;
#endif
}
void
assign_locale_messages(const char *newval, void *extra)
{
/*
* LC_MESSAGES category does not exist everywhere, but accept it anyway.
* We ignore failure, as per comment above.
*/ */
#ifdef LC_MESSAGES #ifdef LC_MESSAGES
if (doit) (void) pg_perm_setlocale(LC_MESSAGES, newval);
{ #endif
if (!pg_perm_setlocale(LC_MESSAGES, value))
if (source != PGC_S_DEFAULT)
return NULL;
}
#ifndef WIN32
else
value = locale_xxx_assign(LC_MESSAGES, value, false, source);
#endif /* WIN32 */
#endif /* LC_MESSAGES */
return value;
} }
......
...@@ -587,8 +587,9 @@ getTSCurrentConfig(bool emitError) ...@@ -587,8 +587,9 @@ getTSCurrentConfig(bool emitError)
return TSCurrentConfigCache; return TSCurrentConfigCache;
} }
const char * /* GUC check_hook for default_text_search_config */
assignTSCurrentConfig(const char *newval, bool doit, GucSource source) bool
check_TSCurrentConfig(char **newval, void **extra, GucSource source)
{ {
/* /*
* If we aren't inside a transaction, we cannot do database access so * If we aren't inside a transaction, we cannot do database access so
...@@ -601,10 +602,10 @@ assignTSCurrentConfig(const char *newval, bool doit, GucSource source) ...@@ -601,10 +602,10 @@ assignTSCurrentConfig(const char *newval, bool doit, GucSource source)
Form_pg_ts_config cfg; Form_pg_ts_config cfg;
char *buf; char *buf;
cfgId = get_ts_config_oid(stringToQualifiedNameList(newval), true); cfgId = get_ts_config_oid(stringToQualifiedNameList(*newval), true);
if (!OidIsValid(cfgId)) if (!OidIsValid(cfgId))
return NULL; return false;
/* /*
* Modify the actually stored value to be fully qualified, to ensure * Modify the actually stored value to be fully qualified, to ensure
...@@ -622,17 +623,20 @@ assignTSCurrentConfig(const char *newval, bool doit, GucSource source) ...@@ -622,17 +623,20 @@ assignTSCurrentConfig(const char *newval, bool doit, GucSource source)
ReleaseSysCache(tuple); ReleaseSysCache(tuple);
/* GUC wants it malloc'd not palloc'd */ /* GUC wants it malloc'd not palloc'd */
newval = strdup(buf); free(*newval);
*newval = strdup(buf);
pfree(buf); pfree(buf);
if (!newval)
if (doit && newval) return false;
TSCurrentConfigCache = cfgId;
}
else
{
if (doit)
TSCurrentConfigCache = InvalidOid;
} }
return newval; return true;
}
/* GUC assign_hook for default_text_search_config */
void
assign_TSCurrentConfig(const char *newval, void *extra)
{
/* Just reset the cache to force a lookup on first use */
TSCurrentConfigCache = InvalidOid;
} }
...@@ -1156,6 +1156,62 @@ elog_finish(int elevel, const char *fmt,...) ...@@ -1156,6 +1156,62 @@ elog_finish(int elevel, const char *fmt,...)
errfinish(0); errfinish(0);
} }
/*
* Functions to allow construction of error message strings separately from
* the ereport() call itself.
*
* The expected calling convention is
*
* pre_format_elog_string(errno, domain), var = format_elog_string(format,...)
*
* which can be hidden behind a macro such as GUC_check_errdetail(). We
* assume that any functions called in the arguments of format_elog_string()
* cannot result in re-entrant use of these functions --- otherwise the wrong
* text domain might be used, or the wrong errno substituted for %m. This is
* okay for the current usage with GUC check hooks, but might need further
* effort someday.
*
* The result of format_elog_string() is stored in ErrorContext, and will
* therefore survive until FlushErrorState() is called.
*/
static int save_format_errnumber;
static const char *save_format_domain;
void
pre_format_elog_string(int errnumber, const char *domain)
{
/* Save errno before evaluation of argument functions can change it */
save_format_errnumber = errnumber;
/* Save caller's text domain */
save_format_domain = domain;
}
char *
format_elog_string(const char *fmt, ...)
{
ErrorData errdata;
ErrorData *edata;
MemoryContext oldcontext;
/* Initialize a mostly-dummy error frame */
edata = &errdata;
MemSet(edata, 0, sizeof(ErrorData));
/* the default text domain is the backend's */
edata->domain = save_format_domain ? save_format_domain : PG_TEXTDOMAIN("postgres");
/* set the errno to be used to interpret %m */
edata->saved_errno = save_format_errnumber;
oldcontext = MemoryContextSwitchTo(ErrorContext);
EVALUATE_MESSAGE(message, false, true);
MemoryContextSwitchTo(oldcontext);
return edata->message;
}
/* /*
* Actual output of the top-of-stack error message * Actual output of the top-of-stack error message
* *
......
...@@ -77,12 +77,16 @@ static int cliplen(const char *str, int len, int limit); ...@@ -77,12 +77,16 @@ static int cliplen(const char *str, int len, int limit);
/* /*
* Set the client encoding and save fmgrinfo for the conversion * Prepare for a future call to SetClientEncoding. Success should mean
* function if necessary. Returns 0 if okay, -1 if not (bad encoding * that SetClientEncoding is guaranteed to succeed for this encoding request.
* or can't support conversion) *
* (But note that success before backend_startup_complete does not guarantee
* success after ...)
*
* Returns 0 if okay, -1 if not (bad encoding or can't support conversion)
*/ */
int int
SetClientEncoding(int encoding, bool doit) PrepareClientEncoding(int encoding)
{ {
int current_server_encoding; int current_server_encoding;
ListCell *lc; ListCell *lc;
...@@ -92,11 +96,7 @@ SetClientEncoding(int encoding, bool doit) ...@@ -92,11 +96,7 @@ SetClientEncoding(int encoding, bool doit)
/* Can't do anything during startup, per notes above */ /* Can't do anything during startup, per notes above */
if (!backend_startup_complete) if (!backend_startup_complete)
{
if (doit)
pending_client_encoding = encoding;
return 0; return 0;
}
current_server_encoding = GetDatabaseEncoding(); current_server_encoding = GetDatabaseEncoding();
...@@ -106,15 +106,7 @@ SetClientEncoding(int encoding, bool doit) ...@@ -106,15 +106,7 @@ SetClientEncoding(int encoding, bool doit)
if (current_server_encoding == encoding || if (current_server_encoding == encoding ||
current_server_encoding == PG_SQL_ASCII || current_server_encoding == PG_SQL_ASCII ||
encoding == PG_SQL_ASCII) encoding == PG_SQL_ASCII)
{
if (doit)
{
ClientEncoding = &pg_enc2name_tbl[encoding];
ToServerConvProc = NULL;
ToClientConvProc = NULL;
}
return 0; return 0;
}
if (IsTransactionState()) if (IsTransactionState())
{ {
...@@ -138,12 +130,6 @@ SetClientEncoding(int encoding, bool doit) ...@@ -138,12 +130,6 @@ SetClientEncoding(int encoding, bool doit)
if (!OidIsValid(to_client_proc)) if (!OidIsValid(to_client_proc))
return -1; return -1;
/*
* Done if not wanting to actually apply setting.
*/
if (!doit)
return 0;
/* /*
* Load the fmgr info into TopMemoryContext (could still fail here) * Load the fmgr info into TopMemoryContext (could still fail here)
*/ */
...@@ -162,30 +148,9 @@ SetClientEncoding(int encoding, bool doit) ...@@ -162,30 +148,9 @@ SetClientEncoding(int encoding, bool doit)
MemoryContextSwitchTo(oldcontext); MemoryContextSwitchTo(oldcontext);
/* /*
* Everything is okay, so apply the setting. * We cannot yet remove any older entry for the same encoding pair,
*/ * since it could still be in use. SetClientEncoding will clean up.
ClientEncoding = &pg_enc2name_tbl[encoding];
ToServerConvProc = &convinfo->to_server_info;
ToClientConvProc = &convinfo->to_client_info;
/*
* Remove any older entry for the same encoding pair (this is just to
* avoid memory leakage).
*/ */
foreach(lc, ConvProcList)
{
ConvProcInfo *oldinfo = (ConvProcInfo *) lfirst(lc);
if (oldinfo == convinfo)
continue;
if (oldinfo->s_encoding == convinfo->s_encoding &&
oldinfo->c_encoding == convinfo->c_encoding)
{
ConvProcList = list_delete_ptr(ConvProcList, oldinfo);
pfree(oldinfo);
break; /* need not look further */
}
}
return 0; /* success */ return 0; /* success */
} }
...@@ -205,24 +170,91 @@ SetClientEncoding(int encoding, bool doit) ...@@ -205,24 +170,91 @@ SetClientEncoding(int encoding, bool doit)
if (oldinfo->s_encoding == current_server_encoding && if (oldinfo->s_encoding == current_server_encoding &&
oldinfo->c_encoding == encoding) oldinfo->c_encoding == encoding)
return 0;
}
return -1; /* it's not cached, so fail */
}
}
/*
* Set the active client encoding and set up the conversion-function pointers.
* PrepareClientEncoding should have been called previously for this encoding.
*
* Returns 0 if okay, -1 if not (bad encoding or can't support conversion)
*/
int
SetClientEncoding(int encoding)
{
int current_server_encoding;
bool found;
ListCell *lc;
if (!PG_VALID_FE_ENCODING(encoding))
return -1;
/* Can't do anything during startup, per notes above */
if (!backend_startup_complete)
{ {
if (doit) pending_client_encoding = encoding;
return 0;
}
current_server_encoding = GetDatabaseEncoding();
/*
* Check for cases that require no conversion function.
*/
if (current_server_encoding == encoding ||
current_server_encoding == PG_SQL_ASCII ||
encoding == PG_SQL_ASCII)
{ {
ClientEncoding = &pg_enc2name_tbl[encoding]; ClientEncoding = &pg_enc2name_tbl[encoding];
ToServerConvProc = &oldinfo->to_server_info; ToServerConvProc = NULL;
ToClientConvProc = &oldinfo->to_client_info; ToClientConvProc = NULL;
}
return 0; return 0;
} }
/*
* Search the cache for the entry previously prepared by
* PrepareClientEncoding; if there isn't one, we lose. While at it,
* release any duplicate entries so that repeated Prepare/Set cycles
* don't leak memory.
*/
found = false;
foreach(lc, ConvProcList)
{
ConvProcInfo *convinfo = (ConvProcInfo *) lfirst(lc);
if (convinfo->s_encoding == current_server_encoding &&
convinfo->c_encoding == encoding)
{
if (!found)
{
/* Found newest entry, so set up */
ClientEncoding = &pg_enc2name_tbl[encoding];
ToServerConvProc = &convinfo->to_server_info;
ToClientConvProc = &convinfo->to_client_info;
found = true;
}
else
{
/* Duplicate entry, release it */
ConvProcList = list_delete_ptr(ConvProcList, convinfo);
pfree(convinfo);
}
}
} }
if (found)
return 0; /* success */
else
return -1; /* it's not cached, so fail */ return -1; /* it's not cached, so fail */
}
} }
/* /*
* Initialize client encoding if necessary. * Initialize client encoding conversions.
* called from InitPostgres() once during backend startup. * Called from InitPostgres() once during backend startup.
*/ */
void void
InitializeClientEncoding(void) InitializeClientEncoding(void)
...@@ -230,7 +262,8 @@ InitializeClientEncoding(void) ...@@ -230,7 +262,8 @@ InitializeClientEncoding(void)
Assert(!backend_startup_complete); Assert(!backend_startup_complete);
backend_startup_complete = true; backend_startup_complete = true;
if (SetClientEncoding(pending_client_encoding, true) < 0) if (PrepareClientEncoding(pending_client_encoding) < 0 ||
SetClientEncoding(pending_client_encoding) < 0)
{ {
/* /*
* Oops, the requested conversion is not available. We couldn't fail * Oops, the requested conversion is not available. We couldn't fail
......
src/backend/utils/misc/README src/backend/utils/misc/README
Guc Implementation Notes GUC Implementation Notes
======================== ========================
The GUC (Grand Unified Configuration) module implements configuration The GUC (Grand Unified Configuration) module implements configuration
variables of multiple types (currently boolean, enum, int, float, and string). variables of multiple types (currently boolean, enum, int, real, and string).
Variable settings can come from various places, with a priority ordering Variable settings can come from various places, with a priority ordering
determining which setting is used. determining which setting is used.
...@@ -12,65 +12,112 @@ determining which setting is used. ...@@ -12,65 +12,112 @@ determining which setting is used.
Per-Variable Hooks Per-Variable Hooks
------------------ ------------------
Each variable known to GUC can optionally have an assign_hook and/or Each variable known to GUC can optionally have a check_hook, an
a show_hook to provide customized behavior. Assign hooks are used to assign_hook, and/or a show_hook to provide customized behavior.
perform validity checking on variable values (above and beyond what Check hooks are used to perform validity checking on variable values
GUC can do). They are also used to update any derived state that needs (above and beyond what GUC can do), to compute derived settings when
to change when a GUC variable is set. Show hooks are used to modify nontrivial work is needed to do that, and optionally to "canonicalize"
the default SHOW display for a variable. user-supplied values. Assign hooks are used to update any derived state
that needs to change when a GUC variable is set. Show hooks are used to
modify the default SHOW display for a variable.
If a check_hook is provided, it points to a function of the signature
bool check_hook(datatype *newvalue, void **extra, GucSource source)
The "newvalue" argument is of type bool *, int *, double *, or char **
for bool, int/enum, real, or string variables respectively. The check
function should validate the proposed new value, and return true if it is
OK or false if not. The function can optionally do a few other things:
* When rejecting a bad proposed value, it may be useful to append some
additional information to the generic "invalid value for parameter FOO"
complaint that guc.c will emit. To do that, call
void GUC_check_errdetail(const char *format, ...)
where the format string and additional arguments follow the rules for
errdetail() arguments. The resulting string will be emitted as the
DETAIL line of guc.c's error report, so it should follow the message style
guidelines for DETAIL messages. There is also
void GUC_check_errhint(const char *format, ...)
which can be used in the same way to append a HINT message.
Occasionally it may even be appropriate to override guc.c's generic primary
message or error code, which can be done with
void GUC_check_errcode(int sqlerrcode)
void GUC_check_errmsg(const char *format, ...)
In general, check_hooks should avoid throwing errors directly if possible,
though this may be impractical to avoid for some corner cases such as
out-of-memory.
* Since the newvalue is pass-by-reference, the function can modify it.
This might be used for example to canonicalize the spelling of a string
value, round off a buffer size to the nearest supported value, or replace
a special value such as "-1" with a computed default value. If the
function wishes to replace a string value, it must malloc (not palloc)
the replacement value, and be sure to free() the previous value.
* Derived information, such as the role OID represented by a user name,
can be stored for use by the assign hook. To do this, malloc (not palloc)
storage space for the information, and return its address at *extra.
guc.c will automatically free() this space when the associated GUC setting
is no longer of interest. *extra is initialized to NULL before call, so
it can be ignored if not needed.
The "source" argument indicates the source of the proposed new value,
If it is >= PGC_S_INTERACTIVE, then we are performing an interactive
assignment (e.g., a SET command). But when source < PGC_S_INTERACTIVE,
we are reading a non-interactive option source, such as postgresql.conf.
This is sometimes needed to determine whether a setting should be
allowed. The check_hook might also look at the current actual value of
the variable to determine what is allowed.
Note that check hooks are sometimes called just to validate a value,
without any intention of actually changing the setting. Therefore the
check hook must *not* take any action based on the assumption that an
assignment will occur.
If an assign_hook is provided, it points to a function of the signature If an assign_hook is provided, it points to a function of the signature
bool assign_hook(newvalue, bool doit, GucSource source) void assign_hook(datatype newvalue, void *extra)
where the type of "newvalue" matches the kind of variable. This function where the type of "newvalue" matches the kind of variable, and "extra"
is called immediately before actually setting the variable's value (so it is the derived-information pointer returned by the check_hook (always
can look at the actual variable to determine the old value). If the NULL if there is no check_hook). This function is called immediately
function returns "true" then the assignment is completed; if it returns before actually setting the variable's value (so it can look at the actual
"false" then newvalue is considered invalid and the assignment is not variable to determine the old value, for example to avoid doing work when
performed. If "doit" is false then the function should simply check the value isn't really changing).
validity of newvalue and not change any derived state. The "source" parameter
indicates where the new value came from. If it is >= PGC_S_INTERACTIVE, Note that there is no provision for a failure result code. assign_hooks
then we are performing an interactive assignment (e.g., a SET command), and should never fail except under the most dire circumstances, since a failure
ereport(ERROR) is safe to do. But when source < PGC_S_INTERACTIVE, we are may for example result in GUC settings not being rolled back properly during
reading a non-interactive option source, such as postgresql.conf. In this transaction abort. In general, try to do anything that could conceivably
case the assign_hook should *not* ereport but should just return false if it fail in a check_hook instead, and pass along the results in an "extra"
doesn't like the newvalue. struct, so that the assign hook has little to do beyond copying the data to
someplace. This applies particularly to catalog lookups: any required
If an assign_hook returns false then guc.c will report a generic "invalid lookups must be done in the check_hook, since the assign_hook may be
value for option FOO" error message. If you feel the need to provide a more executed during transaction rollback when lookups will be unsafe.
specific error message, ereport() it using "GUC_complaint_elevel(source)"
as the error level. Note that this might return either ERROR or a lower level Note that check_hooks are sometimes called outside any transaction, too.
such as LOG, so the ereport call might or might not return. If it does This happens when processing the wired-in "bootstrap" value, values coming
return, return false out of the assign_hook. from the postmaster command line or environment, or values coming from
postgresql.conf. Therefore, any catalog lookups done in a check_hook
For string variables, the signature for assign hooks is a bit different: should be guarded with an IsTransactionState() test, and there must be a
const char *assign_hook(const char *newvalue, fallback path to allow derived values to be computed during the first
bool doit, subsequent use of the GUC setting within a transaction. A typical
GucSource source) arrangement is for the catalog values computed by the check_hook and
The meanings of the parameters are the same as for the other types of GUC installed by the assign_hook to be used only for the remainder of the
variables, but the return value is handled differently: transaction in which the new setting is made. Each subsequent transaction
NULL --- assignment fails (like returning false for other datatypes) looks up the values afresh on first use. This arrangement is useful to
newvalue --- assignment succeeds, assign the newvalue as-is prevent use of stale catalog values, independently of the problem of
malloc'd (not palloc'd!!!) string --- assign that value instead needing to check GUC values outside a transaction.
The third choice is allowed in case the assign_hook wants to return a
"canonical" version of the new value. For example, the assign_hook for
datestyle always returns a string that includes both output and input
datestyle options, although the input might have specified only one.
Note that a string variable's assign_hook will NEVER be called with a NULL
value for newvalue, since there would be no way to distinguish success
and failure returns. If the boot_val or reset_val for a string variable
is NULL, it will just be assigned without calling the assign_hook.
Therefore, a NULL boot_val should never be used in combination with an
assign_hook that has side-effects, as the side-effects wouldn't happen
during a RESET that re-institutes the boot-time setting.
If a show_hook is provided, it points to a function of the signature If a show_hook is provided, it points to a function of the signature
const char *show_hook(void) const char *show_hook(void)
This hook allows variable-specific computation of the value displayed This hook allows variable-specific computation of the value displayed
by SHOW. by SHOW (and other SQL features for showing GUC variable values).
The return value can point to a static buffer, since show functions are
not used re-entrantly.
Saving/Restoring Guc Variable Values Saving/Restoring GUC Variable Values
------------------------------------ ------------------------------------
Prior values of configuration variables must be remembered in order to deal Prior values of configuration variables must be remembered in order to deal
...@@ -200,27 +247,49 @@ these has a current source priority <= PGC_S_FILE. (It is thus possible ...@@ -200,27 +247,49 @@ these has a current source priority <= PGC_S_FILE. (It is thus possible
for reset_val to track the config-file setting even if there is for reset_val to track the config-file setting even if there is
currently a different interactive value of the actual variable.) currently a different interactive value of the actual variable.)
The assign_hook and show_hook routines work only with the actual variable, The check_hook, assign_hook and show_hook routines work only with the
and are not directly aware of the additional values maintained by GUC. actual variable, and are not directly aware of the additional values
This is not a problem for normal usage, since we can assign first to the maintained by GUC.
actual variable and then (if that succeeds) to the additional values as
needed. However, for SIGHUP rereads we may not want to assign to the
actual variable. Our procedure in that case is to call the assign_hook
with doit = false so that the value is validated, but no derived state is
changed.
String Memory Handling GUC Memory Handling
---------------------- -------------------
String option values are allocated with strdup, not with the String variable values are allocated with malloc/strdup, not with the
pstrdup/palloc mechanisms. We would need to keep them in a permanent palloc/pstrdup mechanisms. We would need to keep them in a permanent
context anyway, and strdup gives us more control over handling context anyway, and malloc gives us more control over handling
out-of-memory failures. out-of-memory failures.
We allow a string variable's actual value, reset_val, boot_val, and stacked We allow a string variable's actual value, reset_val, boot_val, and stacked
values to point at the same storage. This makes it slightly harder to free values to point at the same storage. This makes it slightly harder to free
space (we must test whether a value to be freed isn't equal to any of the space (we must test whether a value to be freed isn't equal to any of the
other pointers in the GUC entry or associated stack items). The main other pointers in the GUC entry or associated stack items). The main
advantage is that we never need to strdup during transaction commit/abort, advantage is that we never need to malloc during transaction commit/abort,
so cannot cause an out-of-memory failure there. so cannot cause an out-of-memory failure there.
"Extra" structs returned by check_hook routines are managed in the same
way as string values. Note that we support "extra" structs for all types
of GUC variables, although they are mainly useful with strings.
GUC and Null String Variables
-----------------------------
A GUC string variable can have a boot_val of NULL. guc.c handles this
unsurprisingly, assigning the NULL to the underlying C variable. Any code
using such a variable, as well as any hook functions for it, must then be
prepared to deal with a NULL value.
However, it is not possible to assign a NULL value to a GUC string
variable in any other way: values coming from SET, postgresql.conf, etc,
might be empty strings, but they'll never be NULL. And SHOW displays
a NULL the same as an empty string. It is therefore not appropriate to
treat a NULL value as a distinct user-visible setting. A typical use
for a NULL boot_val is to denote that a value hasn't yet been set for
a variable that will receive a real value later in startup.
If it's undesirable for code using the underlying C variable to have to
worry about NULL values ever, the variable can be given a non-null static
initializer as well as a non-null boot_val. guc.c will overwrite the
static initializer pointer with a copy of the boot_val during
InitializeGUCOptions, but the variable will never contain a NULL.
...@@ -141,7 +141,8 @@ ProcessConfigFile(GucContext context) ...@@ -141,7 +141,8 @@ ProcessConfigFile(GucContext context)
*/ */
cvc_struct = (struct config_string *) cvc_struct = (struct config_string *)
find_option("custom_variable_classes", false, elevel); find_option("custom_variable_classes", false, elevel);
if (cvc_struct && cvc_struct->gen.reset_source > PGC_S_FILE) Assert(cvc_struct);
if (cvc_struct->gen.reset_source > PGC_S_FILE)
{ {
cvc = guc_strdup(elevel, cvc_struct->reset_val); cvc = guc_strdup(elevel, cvc_struct->reset_val);
if (cvc == NULL) if (cvc == NULL)
...@@ -151,19 +152,18 @@ ProcessConfigFile(GucContext context) ...@@ -151,19 +152,18 @@ ProcessConfigFile(GucContext context)
guc_name_compare(head->name, "custom_variable_classes") == 0) guc_name_compare(head->name, "custom_variable_classes") == 0)
{ {
/* /*
* Need to canonicalize the value via the assign hook. Casting away * Need to canonicalize the value by calling the check hook.
* const is a bit ugly, but we know the result is malloc'd.
*/ */
cvc = (char *) assign_custom_variable_classes(head->value, void *extra = NULL;
false, PGC_S_FILE);
cvc = guc_strdup(elevel, head->value);
if (cvc == NULL) if (cvc == NULL)
{
ereport(elevel,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("invalid value for parameter \"%s\": \"%s\"",
head->name, head->value)));
goto cleanup_list; goto cleanup_list;
} if (!call_string_check_hook(cvc_struct, &cvc, &extra,
PGC_S_FILE, elevel))
goto cleanup_list;
if (extra)
free(extra);
} }
/* /*
......
This source diff could not be displayed because it is too large. You can view the blob instead.
...@@ -3,10 +3,12 @@ ...@@ -3,10 +3,12 @@
* tzparser.c * tzparser.c
* Functions for parsing timezone offset files * Functions for parsing timezone offset files
* *
* Note: we generally should not throw any errors in this file, but instead * Note: this code is invoked from the check_hook for the GUC variable
* try to return an error code. This is not completely bulletproof at * timezone_abbreviations. Therefore, it should report problems using
* present --- in particular out-of-memory will throw an error. Could * GUC_check_errmsg() and related functions, and try to avoid throwing
* probably fix with PG_TRY if necessary. * elog(ERROR). This is not completely bulletproof at present --- in
* particular out-of-memory will throw an error. Could probably fix with
* PG_TRY if necessary.
* *
* *
* Portions Copyright (c) 1996-2011, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2011, PostgreSQL Global Development Group
...@@ -24,15 +26,13 @@ ...@@ -24,15 +26,13 @@
#include "miscadmin.h" #include "miscadmin.h"
#include "storage/fd.h" #include "storage/fd.h"
#include "utils/datetime.h" #include "utils/guc.h"
#include "utils/memutils.h" #include "utils/memutils.h"
#include "utils/tzparser.h" #include "utils/tzparser.h"
#define WHITESPACE " \t\n\r" #define WHITESPACE " \t\n\r"
static int tz_elevel; /* to avoid passing this around a lot */
static bool validateTzEntry(tzEntry *tzentry); static bool validateTzEntry(tzEntry *tzentry);
static bool splitTzLine(const char *filename, int lineno, static bool splitTzLine(const char *filename, int lineno,
char *line, tzEntry *tzentry); char *line, tzEntry *tzentry);
...@@ -58,20 +58,16 @@ validateTzEntry(tzEntry *tzentry) ...@@ -58,20 +58,16 @@ validateTzEntry(tzEntry *tzentry)
*/ */
if (strlen(tzentry->abbrev) > TOKMAXLEN) if (strlen(tzentry->abbrev) > TOKMAXLEN)
{ {
ereport(tz_elevel, GUC_check_errmsg("time zone abbreviation \"%s\" is too long (maximum %d characters) in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("time zone abbreviation \"%s\" is too long (maximum %d characters) in time zone file \"%s\", line %d",
tzentry->abbrev, TOKMAXLEN, tzentry->abbrev, TOKMAXLEN,
tzentry->filename, tzentry->lineno))); tzentry->filename, tzentry->lineno);
return false; return false;
} }
if (tzentry->offset % 900 != 0) if (tzentry->offset % 900 != 0)
{ {
ereport(tz_elevel, GUC_check_errmsg("time zone offset %d is not a multiple of 900 sec (15 min) in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("time zone offset %d is not a multiple of 900 sec (15 min) in time zone file \"%s\", line %d",
tzentry->offset, tzentry->offset,
tzentry->filename, tzentry->lineno))); tzentry->filename, tzentry->lineno);
return false; return false;
} }
...@@ -81,11 +77,9 @@ validateTzEntry(tzEntry *tzentry) ...@@ -81,11 +77,9 @@ validateTzEntry(tzEntry *tzentry)
if (tzentry->offset > 14 * 60 * 60 || if (tzentry->offset > 14 * 60 * 60 ||
tzentry->offset < -14 * 60 * 60) tzentry->offset < -14 * 60 * 60)
{ {
ereport(tz_elevel, GUC_check_errmsg("time zone offset %d is out of range in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("time zone offset %d is out of range in time zone file \"%s\", line %d",
tzentry->offset, tzentry->offset,
tzentry->filename, tzentry->lineno))); tzentry->filename, tzentry->lineno);
return false; return false;
} }
...@@ -118,10 +112,8 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry) ...@@ -118,10 +112,8 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry)
abbrev = strtok(line, WHITESPACE); abbrev = strtok(line, WHITESPACE);
if (!abbrev) if (!abbrev)
{ {
ereport(tz_elevel, GUC_check_errmsg("missing time zone abbreviation in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename, lineno);
errmsg("missing time zone abbreviation in time zone file \"%s\", line %d",
filename, lineno)));
return false; return false;
} }
tzentry->abbrev = abbrev; tzentry->abbrev = abbrev;
...@@ -129,19 +121,15 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry) ...@@ -129,19 +121,15 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry)
offset = strtok(NULL, WHITESPACE); offset = strtok(NULL, WHITESPACE);
if (!offset) if (!offset)
{ {
ereport(tz_elevel, GUC_check_errmsg("missing time zone offset in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename, lineno);
errmsg("missing time zone offset in time zone file \"%s\", line %d",
filename, lineno)));
return false; return false;
} }
tzentry->offset = strtol(offset, &offset_endptr, 10); tzentry->offset = strtol(offset, &offset_endptr, 10);
if (offset_endptr == offset || *offset_endptr != '\0') if (offset_endptr == offset || *offset_endptr != '\0')
{ {
ereport(tz_elevel, GUC_check_errmsg("invalid number for time zone offset in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename, lineno);
errmsg("invalid number for time zone offset in time zone file \"%s\", line %d",
filename, lineno)));
return false; return false;
} }
...@@ -163,10 +151,8 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry) ...@@ -163,10 +151,8 @@ splitTzLine(const char *filename, int lineno, char *line, tzEntry *tzentry)
if (remain[0] != '#') /* must be a comment */ if (remain[0] != '#') /* must be a comment */
{ {
ereport(tz_elevel, GUC_check_errmsg("invalid syntax in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename, lineno);
errmsg("invalid syntax in time zone file \"%s\", line %d",
filename, lineno)));
return false; return false;
} }
return true; return true;
...@@ -229,13 +215,11 @@ addToArray(tzEntry **base, int *arraysize, int n, ...@@ -229,13 +215,11 @@ addToArray(tzEntry **base, int *arraysize, int n,
return n; return n;
} }
/* same abbrev but something is different, complain */ /* same abbrev but something is different, complain */
ereport(tz_elevel, GUC_check_errmsg("time zone abbreviation \"%s\" is multiply defined",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), entry->abbrev);
errmsg("time zone abbreviation \"%s\" is multiply defined", GUC_check_errdetail("Entry in time zone file \"%s\", line %d, conflicts with entry in file \"%s\", line %d.",
entry->abbrev),
errdetail("Entry in time zone file \"%s\", line %d, conflicts with entry in file \"%s\", line %d.",
midptr->filename, midptr->lineno, midptr->filename, midptr->lineno,
entry->filename, entry->lineno))); entry->filename, entry->lineno);
return -1; return -1;
} }
} }
...@@ -296,12 +280,10 @@ ParseTzFile(const char *filename, int depth, ...@@ -296,12 +280,10 @@ ParseTzFile(const char *filename, int depth,
{ {
if (!isalpha((unsigned char) *p)) if (!isalpha((unsigned char) *p))
{ {
/* at level 0, we need no ereport since guc.c will say enough */ /* at level 0, just use guc.c's regular "invalid value" message */
if (depth > 0) if (depth > 0)
ereport(tz_elevel, GUC_check_errmsg("invalid time zone file name \"%s\"",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename);
errmsg("invalid time zone file name \"%s\"",
filename)));
return -1; return -1;
} }
} }
...@@ -313,10 +295,8 @@ ParseTzFile(const char *filename, int depth, ...@@ -313,10 +295,8 @@ ParseTzFile(const char *filename, int depth,
*/ */
if (depth > 3) if (depth > 3)
{ {
ereport(tz_elevel, GUC_check_errmsg("time zone file recursion limit exceeded in file \"%s\"",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename);
errmsg("time zone file recursion limit exceeded in file \"%s\"",
filename)));
return -1; return -1;
} }
...@@ -340,12 +320,10 @@ ParseTzFile(const char *filename, int depth, ...@@ -340,12 +320,10 @@ ParseTzFile(const char *filename, int depth,
tzdir = AllocateDir(file_path); tzdir = AllocateDir(file_path);
if (tzdir == NULL) if (tzdir == NULL)
{ {
ereport(tz_elevel, GUC_check_errmsg("could not open directory \"%s\": %m",
(errcode_for_file_access(), file_path);
errmsg("could not open directory \"%s\": %m", GUC_check_errhint("This may indicate an incomplete PostgreSQL installation, or that the file \"%s\" has been moved away from its proper location.",
file_path), my_exec_path);
errhint("This may indicate an incomplete PostgreSQL installation, or that the file \"%s\" has been moved away from its proper location.",
my_exec_path)));
return -1; return -1;
} }
FreeDir(tzdir); FreeDir(tzdir);
...@@ -356,10 +334,8 @@ ParseTzFile(const char *filename, int depth, ...@@ -356,10 +334,8 @@ ParseTzFile(const char *filename, int depth,
* complaint is enough * complaint is enough
*/ */
if (errno != ENOENT || depth > 0) if (errno != ENOENT || depth > 0)
ereport(tz_elevel, GUC_check_errmsg("could not read time zone file \"%s\": %m",
(errcode_for_file_access(), filename);
errmsg("could not read time zone file \"%s\": %m",
filename)));
return -1; return -1;
} }
...@@ -371,10 +347,8 @@ ParseTzFile(const char *filename, int depth, ...@@ -371,10 +347,8 @@ ParseTzFile(const char *filename, int depth,
{ {
if (ferror(tzFile)) if (ferror(tzFile))
{ {
ereport(tz_elevel, GUC_check_errmsg("could not read time zone file \"%s\": %m",
(errcode_for_file_access(), filename);
errmsg("could not read time zone file \"%s\": %m",
filename)));
return -1; return -1;
} }
/* else we're at EOF after all */ /* else we're at EOF after all */
...@@ -383,10 +357,8 @@ ParseTzFile(const char *filename, int depth, ...@@ -383,10 +357,8 @@ ParseTzFile(const char *filename, int depth,
if (strlen(tzbuf) == sizeof(tzbuf) - 1) if (strlen(tzbuf) == sizeof(tzbuf) - 1)
{ {
/* the line is too long for tzbuf */ /* the line is too long for tzbuf */
ereport(tz_elevel, GUC_check_errmsg("line is too long in time zone file \"%s\", line %d",
(errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED), filename, lineno);
errmsg("line is too long in time zone file \"%s\", line %d",
filename, lineno)));
return -1; return -1;
} }
...@@ -408,10 +380,8 @@ ParseTzFile(const char *filename, int depth, ...@@ -408,10 +380,8 @@ ParseTzFile(const char *filename, int depth,
includeFile = strtok(includeFile, WHITESPACE); includeFile = strtok(includeFile, WHITESPACE);
if (!includeFile || !*includeFile) if (!includeFile || !*includeFile)
{ {
ereport(tz_elevel, GUC_check_errmsg("@INCLUDE without file name in time zone file \"%s\", line %d",
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), filename, lineno);
errmsg("@INCLUDE without file name in time zone file \"%s\", line %d",
filename, lineno)));
return -1; return -1;
} }
n = ParseTzFile(includeFile, depth + 1, n = ParseTzFile(includeFile, depth + 1,
...@@ -444,23 +414,20 @@ ParseTzFile(const char *filename, int depth, ...@@ -444,23 +414,20 @@ ParseTzFile(const char *filename, int depth,
/* /*
* load_tzoffsets --- read and parse the specified timezone offset file * load_tzoffsets --- read and parse the specified timezone offset file
* *
* filename: name specified by user * On success, return a filled-in TimeZoneAbbrevTable, which must have been
* doit: whether to actually apply the new values, or just check * malloc'd not palloc'd. On failure, return NULL, using GUC_check_errmsg
* elevel: elog reporting level (will be less than ERROR) * and friends to give details of the problem.
*
* Returns TRUE if OK, FALSE if not; should avoid erroring out
*/ */
bool TimeZoneAbbrevTable *
load_tzoffsets(const char *filename, bool doit, int elevel) load_tzoffsets(const char *filename)
{ {
TimeZoneAbbrevTable *result = NULL;
MemoryContext tmpContext; MemoryContext tmpContext;
MemoryContext oldContext; MemoryContext oldContext;
tzEntry *array; tzEntry *array;
int arraysize; int arraysize;
int n; int n;
tz_elevel = elevel;
/* /*
* Create a temp memory context to work in. This makes it easy to clean * Create a temp memory context to work in. This makes it easy to clean
* up afterwards. * up afterwards.
...@@ -479,13 +446,20 @@ load_tzoffsets(const char *filename, bool doit, int elevel) ...@@ -479,13 +446,20 @@ load_tzoffsets(const char *filename, bool doit, int elevel)
/* Parse the file(s) */ /* Parse the file(s) */
n = ParseTzFile(filename, 0, &array, &arraysize, 0); n = ParseTzFile(filename, 0, &array, &arraysize, 0);
/* If no errors and we should apply the result, pass it to datetime.c */ /* If no errors so far, allocate result and let datetime.c convert data */
if (n >= 0 && doit) if (n >= 0)
InstallTimeZoneAbbrevs(array, n); {
result = malloc(offsetof(TimeZoneAbbrevTable, abbrevs) +
n * sizeof(datetkn));
if (!result)
GUC_check_errmsg("out of memory");
else
ConvertTimeZoneAbbrevs(result, array, n);
}
/* Clean up */ /* Clean up */
MemoryContextSwitchTo(oldContext); MemoryContextSwitchTo(oldContext);
MemoryContextDelete(tmpContext); MemoryContextDelete(tmpContext);
return (n >= 0); return result;
} }
...@@ -13,31 +13,28 @@ ...@@ -13,31 +13,28 @@
#include "utils/guc.h" #include "utils/guc.h"
extern const char *assign_datestyle(const char *value, extern bool check_datestyle(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_datestyle(const char *newval, void *extra);
extern const char *assign_timezone(const char *value, extern bool check_timezone(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_timezone(const char *newval, void *extra);
extern const char *show_timezone(void); extern const char *show_timezone(void);
extern const char *assign_log_timezone(const char *value, extern bool check_log_timezone(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_log_timezone(const char *newval, void *extra);
extern const char *show_log_timezone(void); extern const char *show_log_timezone(void);
extern bool assign_transaction_read_only(bool value, extern bool check_transaction_read_only(bool *newval, void **extra, GucSource source);
bool doit, GucSource source); extern bool check_XactIsoLevel(char **newval, void **extra, GucSource source);
extern const char *assign_XactIsoLevel(const char *value, extern void assign_XactIsoLevel(const char *newval, void *extra);
bool doit, GucSource source);
extern const char *show_XactIsoLevel(void); extern const char *show_XactIsoLevel(void);
extern bool assign_transaction_deferrable(bool newval, bool doit, extern bool check_transaction_deferrable(bool *newval, void **extra, GucSource source);
GucSource source); extern bool check_random_seed(double *newval, void **extra, GucSource source);
extern bool assign_random_seed(double value, extern void assign_random_seed(double newval, void *extra);
bool doit, GucSource source);
extern const char *show_random_seed(void); extern const char *show_random_seed(void);
extern const char *assign_client_encoding(const char *value, extern bool check_client_encoding(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_client_encoding(const char *newval, void *extra);
extern const char *assign_role(const char *value, extern bool check_session_authorization(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_session_authorization(const char *newval, void *extra);
extern bool check_role(char **newval, void **extra, GucSource source);
extern void assign_role(const char *newval, void *extra);
extern const char *show_role(void); extern const char *show_role(void);
extern const char *assign_session_authorization(const char *value,
bool doit, GucSource source);
extern const char *show_session_authorization(void);
#endif /* VARIABLE_H */ #endif /* VARIABLE_H */
...@@ -397,7 +397,8 @@ extern size_t wchar2char(char *to, const wchar_t *from, size_t tolen, Oid collat ...@@ -397,7 +397,8 @@ extern size_t wchar2char(char *to, const wchar_t *from, size_t tolen, Oid collat
extern size_t char2wchar(wchar_t *to, size_t tolen, const char *from, size_t fromlen, Oid collation); extern size_t char2wchar(wchar_t *to, size_t tolen, const char *from, size_t fromlen, Oid collation);
#endif #endif
extern int SetClientEncoding(int encoding, bool doit); extern int PrepareClientEncoding(int encoding);
extern int SetClientEncoding(int encoding);
extern void InitializeClientEncoding(void); extern void InitializeClientEncoding(void);
extern int pg_get_client_encoding(void); extern int pg_get_client_encoding(void);
extern const char *pg_get_client_encoding_name(void); extern const char *pg_get_client_encoding_name(void);
......
...@@ -45,6 +45,6 @@ extern void SyncRepUpdateSyncStandbysDefined(void); ...@@ -45,6 +45,6 @@ extern void SyncRepUpdateSyncStandbysDefined(void);
/* called by various procs */ /* called by various procs */
extern int SyncRepWakeQueue(bool all); extern int SyncRepWakeQueue(bool all);
extern const char *assign_synchronous_standby_names(const char *newval, bool doit, GucSource source); extern bool check_synchronous_standby_names(char **newval, void **extra, GucSource source);
#endif /* _SYNCREP_H */ #endif /* _SYNCREP_H */
...@@ -57,7 +57,8 @@ extern PlannedStmt *pg_plan_query(Query *querytree, int cursorOptions, ...@@ -57,7 +57,8 @@ extern PlannedStmt *pg_plan_query(Query *querytree, int cursorOptions,
extern List *pg_plan_queries(List *querytrees, int cursorOptions, extern List *pg_plan_queries(List *querytrees, int cursorOptions,
ParamListInfo boundParams); ParamListInfo boundParams);
extern bool assign_max_stack_depth(int newval, bool doit, GucSource source); extern bool check_max_stack_depth(int *newval, void **extra, GucSource source);
extern void assign_max_stack_depth(int newval, void *extra);
extern void die(SIGNAL_ARGS); extern void die(SIGNAL_ARGS);
extern void quickdie(SIGNAL_ARGS); extern void quickdie(SIGNAL_ARGS);
......
...@@ -93,6 +93,7 @@ extern TSDictionaryCacheEntry *lookup_ts_dictionary_cache(Oid dictId); ...@@ -93,6 +93,7 @@ extern TSDictionaryCacheEntry *lookup_ts_dictionary_cache(Oid dictId);
extern TSConfigCacheEntry *lookup_ts_config_cache(Oid cfgId); extern TSConfigCacheEntry *lookup_ts_config_cache(Oid cfgId);
extern Oid getTSCurrentConfig(bool emitError); extern Oid getTSCurrentConfig(bool emitError);
extern const char *assignTSCurrentConfig(const char *newval, bool doit, GucSource source); extern bool check_TSCurrentConfig(char **newval, void **extra, GucSource source);
extern void assign_TSCurrentConfig(const char *newval, void *extra);
#endif /* TS_CACHE_H */ #endif /* TS_CACHE_H */
...@@ -20,7 +20,9 @@ ...@@ -20,7 +20,9 @@
#include <math.h> #include <math.h>
#include "utils/timestamp.h" #include "utils/timestamp.h"
#include "utils/tzparser.h"
/* this struct is declared in utils/tzparser.h: */
struct tzEntry;
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
...@@ -203,6 +205,13 @@ typedef struct ...@@ -203,6 +205,13 @@ typedef struct
char value; /* this may be unsigned, alas */ char value; /* this may be unsigned, alas */
} datetkn; } datetkn;
/* one of its uses is in tables of time zone abbreviations */
typedef struct TimeZoneAbbrevTable
{
int numabbrevs;
datetkn abbrevs[1]; /* VARIABLE LENGTH ARRAY */
} TimeZoneAbbrevTable;
/* FMODULO() /* FMODULO()
* Macro to replace modf(), which is broken on some platforms. * Macro to replace modf(), which is broken on some platforms.
...@@ -317,7 +326,10 @@ extern int DecodeUnits(int field, char *lowtoken, int *val); ...@@ -317,7 +326,10 @@ extern int DecodeUnits(int field, char *lowtoken, int *val);
extern int j2day(int jd); extern int j2day(int jd);
extern bool CheckDateTokenTables(void); extern bool CheckDateTokenTables(void);
extern void InstallTimeZoneAbbrevs(tzEntry *abbrevs, int n);
extern void ConvertTimeZoneAbbrevs(TimeZoneAbbrevTable *tbl,
struct tzEntry *abbrevs, int n);
extern void InstallTimeZoneAbbrevs(TimeZoneAbbrevTable *tbl);
extern Datum pg_timezone_abbrevs(PG_FUNCTION_ARGS); extern Datum pg_timezone_abbrevs(PG_FUNCTION_ARGS);
extern Datum pg_timezone_names(PG_FUNCTION_ARGS); extern Datum pg_timezone_names(PG_FUNCTION_ARGS);
......
...@@ -200,6 +200,15 @@ elog_finish(int elevel, const char *fmt,...) ...@@ -200,6 +200,15 @@ elog_finish(int elevel, const char *fmt,...)
__attribute__((format(printf, 2, 3))); __attribute__((format(printf, 2, 3)));
/* Support for constructing error strings separately from ereport() calls */
extern void pre_format_elog_string(int errnumber, const char *domain);
extern char *format_elog_string(const char *fmt,...)
/* This extension allows gcc to check the format string for consistency with
the supplied arguments. */
__attribute__((format(printf, 1, 2)));
/* Support for attaching context information to error reports */ /* Support for attaching context information to error reports */
typedef struct ErrorContextCallback typedef struct ErrorContextCallback
......
...@@ -97,7 +97,8 @@ typedef enum ...@@ -97,7 +97,8 @@ typedef enum
} GucSource; } GucSource;
/* /*
* Parsing the configuation file will return a list of name-value pairs * Parsing the configuration file will return a list of name-value pairs
* with source location info.
*/ */
typedef struct ConfigVariable typedef struct ConfigVariable
{ {
...@@ -117,7 +118,9 @@ extern bool ParseConfigFp(FILE *fp, const char *config_file, ...@@ -117,7 +118,9 @@ extern bool ParseConfigFp(FILE *fp, const char *config_file,
extern void FreeConfigVariables(ConfigVariable *list); extern void FreeConfigVariables(ConfigVariable *list);
/* /*
* Enum values are made up of an array of name-value pairs * The possible values of an enum variable are specified by an array of
* name-value pairs. The "hidden" flag means the value is accepted but
* won't be displayed when guc.c is asked for a list of acceptable values.
*/ */
struct config_enum_entry struct config_enum_entry
{ {
...@@ -126,15 +129,26 @@ struct config_enum_entry ...@@ -126,15 +129,26 @@ struct config_enum_entry
bool hidden; bool hidden;
}; };
/*
typedef const char *(*GucStringAssignHook) (const char *newval, bool doit, GucSource source); * Signatures for per-variable check/assign/show hook functions
typedef bool (*GucBoolAssignHook) (bool newval, bool doit, GucSource source); */
typedef bool (*GucIntAssignHook) (int newval, bool doit, GucSource source); typedef bool (*GucBoolCheckHook) (bool *newval, void **extra, GucSource source);
typedef bool (*GucRealAssignHook) (double newval, bool doit, GucSource source); typedef bool (*GucIntCheckHook) (int *newval, void **extra, GucSource source);
typedef bool (*GucEnumAssignHook) (int newval, bool doit, GucSource source); typedef bool (*GucRealCheckHook) (double *newval, void **extra, GucSource source);
typedef bool (*GucStringCheckHook) (char **newval, void **extra, GucSource source);
typedef bool (*GucEnumCheckHook) (int *newval, void **extra, GucSource source);
typedef void (*GucBoolAssignHook) (bool newval, void *extra);
typedef void (*GucIntAssignHook) (int newval, void *extra);
typedef void (*GucRealAssignHook) (double newval, void *extra);
typedef void (*GucStringAssignHook) (const char *newval, void *extra);
typedef void (*GucEnumAssignHook) (int newval, void *extra);
typedef const char *(*GucShowHook) (void); typedef const char *(*GucShowHook) (void);
/*
* Miscellaneous
*/
typedef enum typedef enum
{ {
/* Types of set_config_option actions */ /* Types of set_config_option actions */
...@@ -201,7 +215,6 @@ extern char *ConfigFileName; ...@@ -201,7 +215,6 @@ extern char *ConfigFileName;
extern char *HbaFileName; extern char *HbaFileName;
extern char *IdentFileName; extern char *IdentFileName;
extern char *external_pid_file; extern char *external_pid_file;
extern char *XactIsoLevel_string;
extern char *application_name; extern char *application_name;
...@@ -209,6 +222,9 @@ extern int tcp_keepalives_idle; ...@@ -209,6 +222,9 @@ extern int tcp_keepalives_idle;
extern int tcp_keepalives_interval; extern int tcp_keepalives_interval;
extern int tcp_keepalives_count; extern int tcp_keepalives_count;
/*
* Functions exported by guc.c
*/
extern void SetConfigOption(const char *name, const char *value, extern void SetConfigOption(const char *name, const char *value,
GucContext context, GucSource source); GucContext context, GucSource source);
...@@ -220,6 +236,7 @@ extern void DefineCustomBoolVariable( ...@@ -220,6 +236,7 @@ extern void DefineCustomBoolVariable(
bool bootValue, bool bootValue,
GucContext context, GucContext context,
int flags, int flags,
GucBoolCheckHook check_hook,
GucBoolAssignHook assign_hook, GucBoolAssignHook assign_hook,
GucShowHook show_hook); GucShowHook show_hook);
...@@ -233,6 +250,7 @@ extern void DefineCustomIntVariable( ...@@ -233,6 +250,7 @@ extern void DefineCustomIntVariable(
int maxValue, int maxValue,
GucContext context, GucContext context,
int flags, int flags,
GucIntCheckHook check_hook,
GucIntAssignHook assign_hook, GucIntAssignHook assign_hook,
GucShowHook show_hook); GucShowHook show_hook);
...@@ -246,6 +264,7 @@ extern void DefineCustomRealVariable( ...@@ -246,6 +264,7 @@ extern void DefineCustomRealVariable(
double maxValue, double maxValue,
GucContext context, GucContext context,
int flags, int flags,
GucRealCheckHook check_hook,
GucRealAssignHook assign_hook, GucRealAssignHook assign_hook,
GucShowHook show_hook); GucShowHook show_hook);
...@@ -257,6 +276,7 @@ extern void DefineCustomStringVariable( ...@@ -257,6 +276,7 @@ extern void DefineCustomStringVariable(
const char *bootValue, const char *bootValue,
GucContext context, GucContext context,
int flags, int flags,
GucStringCheckHook check_hook,
GucStringAssignHook assign_hook, GucStringAssignHook assign_hook,
GucShowHook show_hook); GucShowHook show_hook);
...@@ -269,6 +289,7 @@ extern void DefineCustomEnumVariable( ...@@ -269,6 +289,7 @@ extern void DefineCustomEnumVariable(
const struct config_enum_entry * options, const struct config_enum_entry * options,
GucContext context, GucContext context,
int flags, int flags,
GucEnumCheckHook check_hook,
GucEnumAssignHook assign_hook, GucEnumAssignHook assign_hook,
GucShowHook show_hook); GucShowHook show_hook);
...@@ -308,8 +329,6 @@ extern ArrayType *GUCArrayAdd(ArrayType *array, const char *name, const char *va ...@@ -308,8 +329,6 @@ extern ArrayType *GUCArrayAdd(ArrayType *array, const char *name, const char *va
extern ArrayType *GUCArrayDelete(ArrayType *array, const char *name); extern ArrayType *GUCArrayDelete(ArrayType *array, const char *name);
extern ArrayType *GUCArrayReset(ArrayType *array); extern ArrayType *GUCArrayReset(ArrayType *array);
extern int GUC_complaint_elevel(GucSource source);
extern void pg_timezone_abbrev_initialize(void); extern void pg_timezone_abbrev_initialize(void);
#ifdef EXEC_BACKEND #ifdef EXEC_BACKEND
...@@ -317,6 +336,27 @@ extern void write_nondefault_variables(GucContext context); ...@@ -317,6 +336,27 @@ extern void write_nondefault_variables(GucContext context);
extern void read_nondefault_variables(void); extern void read_nondefault_variables(void);
#endif #endif
/* Support for messages reported from GUC check hooks */
extern PGDLLIMPORT char *GUC_check_errmsg_string;
extern PGDLLIMPORT char *GUC_check_errdetail_string;
extern PGDLLIMPORT char *GUC_check_errhint_string;
extern void GUC_check_errcode(int sqlerrcode);
#define GUC_check_errmsg \
pre_format_elog_string(errno, TEXTDOMAIN), \
GUC_check_errmsg_string = format_elog_string
#define GUC_check_errdetail \
pre_format_elog_string(errno, TEXTDOMAIN), \
GUC_check_errdetail_string = format_elog_string
#define GUC_check_errhint \
pre_format_elog_string(errno, TEXTDOMAIN), \
GUC_check_errhint_string = format_elog_string
/* /*
* The following functions are not in guc.c, but are declared here to avoid * The following functions are not in guc.c, but are declared here to avoid
* having to include guc.h in some widely used headers that it really doesn't * having to include guc.h in some widely used headers that it really doesn't
...@@ -324,17 +364,16 @@ extern void read_nondefault_variables(void); ...@@ -324,17 +364,16 @@ extern void read_nondefault_variables(void);
*/ */
/* in commands/tablespace.c */ /* in commands/tablespace.c */
extern const char *assign_default_tablespace(const char *newval, extern bool check_default_tablespace(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern bool check_temp_tablespaces(char **newval, void **extra, GucSource source);
extern const char *assign_temp_tablespaces(const char *newval, extern void assign_temp_tablespaces(const char *newval, void *extra);
bool doit, GucSource source);
/* in catalog/namespace.c */ /* in catalog/namespace.c */
extern const char *assign_search_path(const char *newval, extern bool check_search_path(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_search_path(const char *newval, void *extra);
/* in access/transam/xlog.c */ /* in access/transam/xlog.c */
extern bool assign_xlog_sync_method(int newval, extern bool check_wal_buffers(int *newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_xlog_sync_method(int new_sync_method, void *extra);
#endif /* GUC_H */ #endif /* GUC_H */
...@@ -28,7 +28,7 @@ enum config_type ...@@ -28,7 +28,7 @@ enum config_type
PGC_ENUM PGC_ENUM
}; };
union config_var_value union config_var_val
{ {
bool boolval; bool boolval;
int intval; int intval;
...@@ -37,6 +37,16 @@ union config_var_value ...@@ -37,6 +37,16 @@ union config_var_value
int enumval; int enumval;
}; };
/*
* The actual value of a GUC variable can include a malloc'd opaque struct
* "extra", which is created by its check_hook and used by its assign_hook.
*/
typedef struct config_var_value
{
union config_var_val val;
void *extra;
} config_var_value;
/* /*
* Groupings to help organize all the run-time options for display * Groupings to help organize all the run-time options for display
*/ */
...@@ -105,8 +115,8 @@ typedef struct guc_stack ...@@ -105,8 +115,8 @@ typedef struct guc_stack
int nest_level; /* nesting depth at which we made entry */ int nest_level; /* nesting depth at which we made entry */
GucStackState state; /* see enum above */ GucStackState state; /* see enum above */
GucSource source; /* source of the prior value */ GucSource source; /* source of the prior value */
union config_var_value prior; /* previous value of variable */ config_var_value prior; /* previous value of variable */
union config_var_value masked; /* SET value in a GUC_SET_LOCAL entry */ config_var_value masked; /* SET value in a GUC_SET_LOCAL entry */
/* masked value's source must be PGC_S_SESSION, so no need to store it */ /* masked value's source must be PGC_S_SESSION, so no need to store it */
} GucStack; } GucStack;
...@@ -116,6 +126,11 @@ typedef struct guc_stack ...@@ -116,6 +126,11 @@ typedef struct guc_stack
* The short description should be less than 80 chars in length. Some * The short description should be less than 80 chars in length. Some
* applications may use the long description as well, and will append * applications may use the long description as well, and will append
* it to the short description. (separated by a newline or '. ') * it to the short description. (separated by a newline or '. ')
*
* Note that sourcefile/sourceline are kept here, and not pushed into stacked
* values, although in principle they belong with some stacked value if the
* active value is session- or transaction-local. This is to avoid bloating
* stack entries. We know they are only relevant when source == PGC_S_FILE.
*/ */
struct config_generic struct config_generic
{ {
...@@ -132,7 +147,8 @@ struct config_generic ...@@ -132,7 +147,8 @@ struct config_generic
GucSource reset_source; /* source of the reset_value */ GucSource reset_source; /* source of the reset_value */
GucSource source; /* source of the current actual value */ GucSource source; /* source of the current actual value */
GucStack *stack; /* stacked prior values */ GucStack *stack; /* stacked prior values */
char *sourcefile; /* file this settings is from (NULL if not void *extra; /* "extra" pointer for current actual value */
char *sourcefile; /* file current setting is from (NULL if not
* file) */ * file) */
int sourceline; /* line in source file */ int sourceline; /* line in source file */
}; };
...@@ -155,10 +171,12 @@ struct config_bool ...@@ -155,10 +171,12 @@ struct config_bool
/* constant fields, must be set correctly in initial value: */ /* constant fields, must be set correctly in initial value: */
bool *variable; bool *variable;
bool boot_val; bool boot_val;
GucBoolCheckHook check_hook;
GucBoolAssignHook assign_hook; GucBoolAssignHook assign_hook;
GucShowHook show_hook; GucShowHook show_hook;
/* variable fields, initialized at runtime: */ /* variable fields, initialized at runtime: */
bool reset_val; bool reset_val;
void *reset_extra;
}; };
struct config_int struct config_int
...@@ -169,10 +187,12 @@ struct config_int ...@@ -169,10 +187,12 @@ struct config_int
int boot_val; int boot_val;
int min; int min;
int max; int max;
GucIntCheckHook check_hook;
GucIntAssignHook assign_hook; GucIntAssignHook assign_hook;
GucShowHook show_hook; GucShowHook show_hook;
/* variable fields, initialized at runtime: */ /* variable fields, initialized at runtime: */
int reset_val; int reset_val;
void *reset_extra;
}; };
struct config_real struct config_real
...@@ -183,10 +203,12 @@ struct config_real ...@@ -183,10 +203,12 @@ struct config_real
double boot_val; double boot_val;
double min; double min;
double max; double max;
GucRealCheckHook check_hook;
GucRealAssignHook assign_hook; GucRealAssignHook assign_hook;
GucShowHook show_hook; GucShowHook show_hook;
/* variable fields, initialized at runtime: */ /* variable fields, initialized at runtime: */
double reset_val; double reset_val;
void *reset_extra;
}; };
struct config_string struct config_string
...@@ -195,10 +217,12 @@ struct config_string ...@@ -195,10 +217,12 @@ struct config_string
/* constant fields, must be set correctly in initial value: */ /* constant fields, must be set correctly in initial value: */
char **variable; char **variable;
const char *boot_val; const char *boot_val;
GucStringCheckHook check_hook;
GucStringAssignHook assign_hook; GucStringAssignHook assign_hook;
GucShowHook show_hook; GucShowHook show_hook;
/* variable fields, initialized at runtime: */ /* variable fields, initialized at runtime: */
char *reset_val; char *reset_val;
void *reset_extra;
}; };
struct config_enum struct config_enum
...@@ -208,10 +232,12 @@ struct config_enum ...@@ -208,10 +232,12 @@ struct config_enum
int *variable; int *variable;
int boot_val; int boot_val;
const struct config_enum_entry *options; const struct config_enum_entry *options;
GucEnumCheckHook check_hook;
GucEnumAssignHook assign_hook; GucEnumAssignHook assign_hook;
GucShowHook show_hook; GucShowHook show_hook;
/* variable fields, initialized at runtime: */ /* variable fields, initialized at runtime: */
int reset_val; int reset_val;
void *reset_extra;
}; };
/* constant tables corresponding to enums above and in guc.h */ /* constant tables corresponding to enums above and in guc.h */
......
...@@ -33,14 +33,14 @@ extern char *localized_abbrev_months[]; ...@@ -33,14 +33,14 @@ extern char *localized_abbrev_months[];
extern char *localized_full_months[]; extern char *localized_full_months[];
extern const char *locale_messages_assign(const char *value, extern bool check_locale_messages(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_locale_messages(const char *newval, void *extra);
extern const char *locale_monetary_assign(const char *value, extern bool check_locale_monetary(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_locale_monetary(const char *newval, void *extra);
extern const char *locale_numeric_assign(const char *value, extern bool check_locale_numeric(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_locale_numeric(const char *newval, void *extra);
extern const char *locale_time_assign(const char *value, extern bool check_locale_time(char **newval, void **extra, GucSource source);
bool doit, GucSource source); extern void assign_locale_time(const char *newval, void *extra);
extern bool check_locale(int category, const char *locale); extern bool check_locale(int category, const char *locale);
extern char *pg_perm_setlocale(int category, const char *locale); extern char *pg_perm_setlocale(int category, const char *locale);
......
...@@ -13,6 +13,8 @@ ...@@ -13,6 +13,8 @@
#ifndef TZPARSER_H #ifndef TZPARSER_H
#define TZPARSER_H #define TZPARSER_H
#include "utils/datetime.h"
/* /*
* The result of parsing a timezone configuration file is an array of * The result of parsing a timezone configuration file is an array of
* these structs, in order by abbrev. We export this because datetime.c * these structs, in order by abbrev. We export this because datetime.c
...@@ -30,6 +32,6 @@ typedef struct tzEntry ...@@ -30,6 +32,6 @@ typedef struct tzEntry
} tzEntry; } tzEntry;
extern bool load_tzoffsets(const char *filename, bool doit, int elevel); extern TimeZoneAbbrevTable *load_tzoffsets(const char *filename);
#endif /* TZPARSER_H */ #endif /* TZPARSER_H */
...@@ -363,7 +363,7 @@ _PG_init(void) ...@@ -363,7 +363,7 @@ _PG_init(void)
&plperl_use_strict, &plperl_use_strict,
false, false,
PGC_USERSET, 0, PGC_USERSET, 0,
NULL, NULL); NULL, NULL, NULL);
/* /*
* plperl.on_init is marked PGC_SIGHUP to support the idea that it might * plperl.on_init is marked PGC_SIGHUP to support the idea that it might
...@@ -377,7 +377,7 @@ _PG_init(void) ...@@ -377,7 +377,7 @@ _PG_init(void)
&plperl_on_init, &plperl_on_init,
NULL, NULL,
PGC_SIGHUP, 0, PGC_SIGHUP, 0,
NULL, NULL); NULL, NULL, NULL);
/* /*
* plperl.on_plperl_init is marked PGC_SUSET to avoid issues whereby a * plperl.on_plperl_init is marked PGC_SUSET to avoid issues whereby a
...@@ -399,7 +399,7 @@ _PG_init(void) ...@@ -399,7 +399,7 @@ _PG_init(void)
&plperl_on_plperl_init, &plperl_on_plperl_init,
NULL, NULL,
PGC_SUSET, 0, PGC_SUSET, 0,
NULL, NULL); NULL, NULL, NULL);
DefineCustomStringVariable("plperl.on_plperlu_init", DefineCustomStringVariable("plperl.on_plperlu_init",
gettext_noop("Perl initialization code to execute once when plperlu is first used."), gettext_noop("Perl initialization code to execute once when plperlu is first used."),
...@@ -407,7 +407,7 @@ _PG_init(void) ...@@ -407,7 +407,7 @@ _PG_init(void)
&plperl_on_plperlu_init, &plperl_on_plperlu_init,
NULL, NULL,
PGC_SUSET, 0, PGC_SUSET, 0,
NULL, NULL); NULL, NULL, NULL);
EmitWarningsOnPlaceholders("plperl"); EmitWarningsOnPlaceholders("plperl");
......
...@@ -63,7 +63,7 @@ _PG_init(void) ...@@ -63,7 +63,7 @@ _PG_init(void)
PLPGSQL_RESOLVE_ERROR, PLPGSQL_RESOLVE_ERROR,
variable_conflict_options, variable_conflict_options,
PGC_SUSET, 0, PGC_SUSET, 0,
NULL, NULL); NULL, NULL, NULL);
EmitWarningsOnPlaceholders("plpgsql"); EmitWarningsOnPlaceholders("plpgsql");
......
...@@ -622,7 +622,8 @@ alter function report_guc(text) set search_path = no_such_schema; ...@@ -622,7 +622,8 @@ alter function report_guc(text) set search_path = no_such_schema;
NOTICE: schema "no_such_schema" does not exist NOTICE: schema "no_such_schema" does not exist
-- with error occurring here -- with error occurring here
select report_guc('work_mem'), current_setting('work_mem'); select report_guc('work_mem'), current_setting('work_mem');
ERROR: schema "no_such_schema" does not exist ERROR: invalid value for parameter "search_path": "no_such_schema"
DETAIL: schema "no_such_schema" does not exist
alter function report_guc(text) reset search_path set work_mem = '2MB'; alter function report_guc(text) reset search_path set work_mem = '2MB';
select report_guc('work_mem'), current_setting('work_mem'); select report_guc('work_mem'), current_setting('work_mem');
report_guc | current_setting report_guc | current_setting
......
...@@ -1444,27 +1444,39 @@ pg_timezone_initialize(void) ...@@ -1444,27 +1444,39 @@ pg_timezone_initialize(void)
{ {
pg_tz *def_tz = NULL; pg_tz *def_tz = NULL;
/* Do we need to try to figure the session timezone? */ /*
if (pg_strcasecmp(GetConfigOption("timezone", false), "UNKNOWN") == 0) * Make sure that session_timezone and log_timezone are set.
* (session_timezone could still be NULL even if a timezone value was set
* in postgresql.conf, if that setting was interval-based rather than
* timezone-based.)
*/
if (!session_timezone)
{ {
/* Select setting */
def_tz = select_default_timezone(); def_tz = select_default_timezone();
session_timezone = def_tz; session_timezone = def_tz;
/* Tell GUC about the value. Will redundantly call pg_tzset() */
SetConfigOption("timezone", pg_get_timezone_name(def_tz),
PGC_POSTMASTER, PGC_S_ARGV);
} }
if (!log_timezone)
/* What about the log timezone? */
if (pg_strcasecmp(GetConfigOption("log_timezone", false), "UNKNOWN") == 0)
{ {
/* Select setting, but don't duplicate work */ /* Don't duplicate work */
if (!def_tz) if (!def_tz)
def_tz = select_default_timezone(); def_tz = select_default_timezone();
log_timezone = def_tz; log_timezone = def_tz;
}
/* Now, set the timezone GUC if it's not already set */
if (GetConfigOption("timezone", false) == NULL)
{
/* Tell GUC about the value. Will redundantly call pg_tzset() */
SetConfigOption("timezone", pg_get_timezone_name(session_timezone),
PGC_POSTMASTER, PGC_S_ENV_VAR);
}
/* Likewise for log timezone */
if (GetConfigOption("log_timezone", false) == NULL)
{
/* Tell GUC about the value. Will redundantly call pg_tzset() */ /* Tell GUC about the value. Will redundantly call pg_tzset() */
SetConfigOption("log_timezone", pg_get_timezone_name(def_tz), SetConfigOption("log_timezone", pg_get_timezone_name(log_timezone),
PGC_POSTMASTER, PGC_S_ARGV); PGC_POSTMASTER, PGC_S_ENV_VAR);
} }
} }
......
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