diff options
author | Robert Haas <rhaas@postgresql.org> | 2016-04-21 14:02:15 -0400 |
---|---|---|
committer | Robert Haas <rhaas@postgresql.org> | 2016-04-21 14:02:15 -0400 |
commit | c4a586c4860477ddae6d4f9cef88486f0e37c37e (patch) | |
tree | 7e62ae702d8f468c29c748a27e0d57fd8bee0e00 /src/backend/utils/adt/pgstatfuncs.c | |
parent | 36f69faeff540cd93de0b6aa7c2d2a7781d637a6 (diff) | |
download | postgresql-c4a586c4860477ddae6d4f9cef88486f0e37c37e.tar.gz postgresql-c4a586c4860477ddae6d4f9cef88486f0e37c37e.zip |
Prevent possible crash reading pg_stat_activity.
Also, avoid reading PGPROC's wait_event field twice, once for the wait
event and again for the wait_event_type, because the value might change
in the middle.
Petr Jelinek and Robert Haas
Diffstat (limited to 'src/backend/utils/adt/pgstatfuncs.c')
-rw-r--r-- | src/backend/utils/adt/pgstatfuncs.c | 33 |
1 files changed, 21 insertions, 12 deletions
diff --git a/src/backend/utils/adt/pgstatfuncs.c b/src/backend/utils/adt/pgstatfuncs.c index 64c4cc4538d..17c5cb0fb64 100644 --- a/src/backend/utils/adt/pgstatfuncs.c +++ b/src/backend/utils/adt/pgstatfuncs.c @@ -27,6 +27,8 @@ #include "utils/inet.h" #include "utils/timestamp.h" +#define UINT32_ACCESS_ONCE(var) ((uint32)(*((volatile uint32 *)&(var)))) + /* bogus ... these externs should be in a header file */ extern Datum pg_stat_get_numscans(PG_FUNCTION_ARGS); extern Datum pg_stat_get_tuples_returned(PG_FUNCTION_ARGS); @@ -783,13 +785,26 @@ pg_stat_get_activity(PG_FUNCTION_ARGS) values[5] = CStringGetTextDatum(beentry->st_activity); proc = BackendPidGetProc(beentry->st_procpid); - wait_event_type = pgstat_get_wait_event_type(proc->wait_event_info); + if (proc != NULL) + { + uint32 raw_wait_event; + + raw_wait_event = UINT32_ACCESS_ONCE(proc->wait_event_info); + wait_event_type = pgstat_get_wait_event_type(raw_wait_event); + wait_event = pgstat_get_wait_event(raw_wait_event); + + } + else + { + wait_event_type = NULL; + wait_event = NULL; + } + if (wait_event_type) values[6] = CStringGetTextDatum(wait_event_type); else nulls[6] = true; - wait_event = pgstat_get_wait_event(proc->wait_event_info); if (wait_event) values[7] = CStringGetTextDatum(wait_event); else @@ -984,17 +999,14 @@ pg_stat_get_backend_wait_event_type(PG_FUNCTION_ARGS) int32 beid = PG_GETARG_INT32(0); PgBackendStatus *beentry; PGPROC *proc; - const char *wait_event_type; + const char *wait_event_type = NULL; if ((beentry = pgstat_fetch_stat_beentry(beid)) == NULL) wait_event_type = "<backend information not available>"; else if (!has_privs_of_role(GetUserId(), beentry->st_userid)) wait_event_type = "<insufficient privilege>"; - else - { - proc = BackendPidGetProc(beentry->st_procpid); + else if ((proc = BackendPidGetProc(beentry->st_procpid)) != NULL) wait_event_type = pgstat_get_wait_event_type(proc->wait_event_info); - } if (!wait_event_type) PG_RETURN_NULL(); @@ -1008,17 +1020,14 @@ pg_stat_get_backend_wait_event(PG_FUNCTION_ARGS) int32 beid = PG_GETARG_INT32(0); PgBackendStatus *beentry; PGPROC *proc; - const char *wait_event; + const char *wait_event = NULL; if ((beentry = pgstat_fetch_stat_beentry(beid)) == NULL) wait_event = "<backend information not available>"; else if (!has_privs_of_role(GetUserId(), beentry->st_userid)) wait_event = "<insufficient privilege>"; - else - { - proc = BackendPidGetProc(beentry->st_procpid); + else if ((proc = BackendPidGetProc(beentry->st_procpid)) != NULL) wait_event = pgstat_get_wait_event(proc->wait_event_info); - } if (!wait_event) PG_RETURN_NULL(); |