</itemizedlist>
</entry>
</row>
+ <row>
+ <entry><structfield>backend_xid</structfield></entry>
+ <entry><type>xid</type></entry>
+ <entry>Toplevel transaction identifier of this backend, if any.</entry>
+ </row>
+ <row>
+ <entry><structfield>backend_xmin</structfield></entry>
+ <entry><type>xid</type></entry>
+ <entry>The current backend's <literal>xmin</> horizon.</entry>
+ </row>
<row>
<entry><structfield>query</></entry>
<entry><type>text</></entry>
client connected to this WAL sender
</entry>
</row>
+ <row>
+ <entry><structfield>backend_xid</structfield></entry>
+ <entry><type>xid</type></entry>
+ <entry>This standby's <literal>xmin</> horizon reported
+ by <xref linkend="guc-hot-standby-feedback">.</entry>
+ </row>
<row>
<entry><structfield>state</></entry>
<entry><type>text</></entry>
S.state_change,
S.waiting,
S.state,
+ S.backend_xid,
+ s.backend_xmin,
S.query
FROM pg_database D, pg_stat_get_activity(NULL) AS S, pg_authid U
WHERE S.datid = D.oid AND
S.client_hostname,
S.client_port,
S.backend_start,
+ S.backend_xmin,
W.state,
W.sent_location,
W.write_location,
#include "postmaster/autovacuum.h"
#include "postmaster/fork_process.h"
#include "postmaster/postmaster.h"
+#include "storage/proc.h"
#include "storage/backendid.h"
#include "storage/fd.h"
#include "storage/ipc.h"
#include "storage/latch.h"
#include "storage/pg_shmem.h"
#include "storage/procsignal.h"
+#include "storage/sinvaladt.h"
#include "utils/ascii.h"
#include "utils/guc.h"
#include "utils/memutils.h"
*/
static MemoryContext pgStatLocalContext = NULL;
static HTAB *pgStatDBHash = NULL;
-static PgBackendStatus *localBackendStatusTable = NULL;
+static LocalPgBackendStatus *localBackendStatusTable = NULL;
static int localNumBackends = 0;
/*
{
pgstat_read_current_status();
+ if (beid < 1 || beid > localNumBackends)
+ return NULL;
+
+ return &localBackendStatusTable[beid - 1].backendStatus;
+}
+
+
+/* ----------
+ * pgstat_fetch_stat_local_beentry() -
+ *
+ * Like pgstat_fetch_stat_beentry() but with locally computed addtions (like
+ * xid and xmin values of the backend)
+ *
+ * NB: caller is responsible for a check if the user is permitted to see
+ * this info (especially the querystring).
+ * ----------
+ */
+LocalPgBackendStatus *
+pgstat_fetch_stat_local_beentry(int beid)
+{
+ pgstat_read_current_status();
+
if (beid < 1 || beid > localNumBackends)
return NULL;
pgstat_read_current_status(void)
{
volatile PgBackendStatus *beentry;
- PgBackendStatus *localtable;
- PgBackendStatus *localentry;
+ LocalPgBackendStatus *localtable;
+ LocalPgBackendStatus *localentry;
char *localappname,
*localactivity;
int i;
pgstat_setup_memcxt();
- localtable = (PgBackendStatus *)
+ localtable = (LocalPgBackendStatus *)
MemoryContextAlloc(pgStatLocalContext,
- sizeof(PgBackendStatus) * MaxBackends);
+ sizeof(LocalPgBackendStatus) * MaxBackends);
localappname = (char *)
MemoryContextAlloc(pgStatLocalContext,
NAMEDATALEN * MaxBackends);
{
int save_changecount = beentry->st_changecount;
- localentry->st_procpid = beentry->st_procpid;
- if (localentry->st_procpid > 0)
+ localentry->backendStatus.st_procpid = beentry->st_procpid;
+ if (localentry->backendStatus.st_procpid > 0)
{
- memcpy(localentry, (char *) beentry, sizeof(PgBackendStatus));
+ memcpy(&localentry->backendStatus, (char *) beentry, sizeof(PgBackendStatus));
/*
* strcpy is safe even if the string is modified concurrently,
* because there's always a \0 at the end of the buffer.
*/
strcpy(localappname, (char *) beentry->st_appname);
- localentry->st_appname = localappname;
+ localentry->backendStatus.st_appname = localappname;
strcpy(localactivity, (char *) beentry->st_activity);
- localentry->st_activity = localactivity;
+ localentry->backendStatus.st_activity = localactivity;
}
if (save_changecount == beentry->st_changecount &&
beentry++;
/* Only valid entries get included into the local array */
- if (localentry->st_procpid > 0)
+ if (localentry->backendStatus.st_procpid > 0)
{
+ BackendIdGetTransactionIds(i,
+ &localentry->backend_xid,
+ &localentry->backend_xmin);
+
localentry++;
localappname += NAMEDATALEN;
localactivity += pgstat_track_activity_query_size;
#include "storage/shmem.h"
#include "storage/sinvaladt.h"
#include "storage/spin.h"
+#include "access/transam.h"
/*
return result;
}
+/*
+ * BackendIdGetTransactionIds
+ * Get the xid and xmin of the backend. The result may be out of date
+ * arbitrarily quickly, so the caller must be careful about how this
+ * information is used.
+ */
+void
+BackendIdGetTransactionIds(int backendID, TransactionId *xid, TransactionId *xmin)
+{
+ ProcState *stateP;
+ SISeg *segP = shmInvalBuffer;
+ PGXACT *xact;
+
+ *xid = InvalidTransactionId;
+ *xmin = InvalidTransactionId;
+
+ /* Need to lock out additions/removals of backends */
+ LWLockAcquire(SInvalWriteLock, LW_SHARED);
+
+ if (backendID > 0 && backendID <= segP->lastBackend)
+ {
+ stateP = &segP->procState[backendID - 1];
+ xact = &ProcGlobal->allPgXact[stateP->proc->pgprocno];
+
+ *xid = xact->xid;
+ *xmin = xact->xmin;
+ }
+
+ LWLockRelease(SInvalWriteLock);
+}
+
/*
* SIInsertDataEntries
* Add new invalidation message(s) to the buffer.
oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);
- tupdesc = CreateTemplateTupleDesc(14, false);
+ tupdesc = CreateTemplateTupleDesc(16, false);
TupleDescInitEntry(tupdesc, (AttrNumber) 1, "datid",
OIDOID, -1, 0);
TupleDescInitEntry(tupdesc, (AttrNumber) 2, "pid",
TEXTOID, -1, 0);
TupleDescInitEntry(tupdesc, (AttrNumber) 14, "client_port",
INT4OID, -1, 0);
+ TupleDescInitEntry(tupdesc, (AttrNumber) 15, "backend_xid",
+ XIDOID, -1, 0);
+ TupleDescInitEntry(tupdesc, (AttrNumber) 16, "backend_xmin",
+ XIDOID, -1, 0);
funcctx->tuple_desc = BlessTupleDesc(tupdesc);
if (funcctx->call_cntr < funcctx->max_calls)
{
/* for each row */
- Datum values[14];
- bool nulls[14];
+ Datum values[16];
+ bool nulls[16];
HeapTuple tuple;
+ LocalPgBackendStatus *local_beentry;
PgBackendStatus *beentry;
MemSet(values, 0, sizeof(values));
if (*(int *) (funcctx->user_fctx) > 0)
{
/* Get specific pid slot */
- beentry = pgstat_fetch_stat_beentry(*(int *) (funcctx->user_fctx));
+ local_beentry = pgstat_fetch_stat_local_beentry(*(int *) (funcctx->user_fctx));
+ beentry = &local_beentry->backendStatus;
}
else
{
/* Get the next one in the list */
- beentry = pgstat_fetch_stat_beentry(funcctx->call_cntr + 1); /* 1-based index */
+ local_beentry = pgstat_fetch_stat_local_beentry(funcctx->call_cntr + 1); /* 1-based index */
+ beentry = &local_beentry->backendStatus;
}
if (!beentry)
{
else
nulls[3] = true;
+ if (TransactionIdIsValid(local_beentry->backend_xid))
+ values[14] = TransactionIdGetDatum(local_beentry->backend_xid);
+ else
+ nulls[14] = true;
+
+ if (TransactionIdIsValid(local_beentry->backend_xmin))
+ values[15] = TransactionIdGetDatum(local_beentry->backend_xmin);
+ else
+ nulls[15] = true;
+
/* Values only available to same user or superuser */
if (superuser() || beentry->st_userid == GetUserId())
{
*/
/* yyyymmddN */
-#define CATALOG_VERSION_NO 201402241
+#define CATALOG_VERSION_NO 201402251
#endif
DESCR("statistics: number of auto analyzes for a table");
DATA(insert OID = 1936 ( pg_stat_get_backend_idset PGNSP PGUID 12 1 100 0 0 f f f f t t s 0 0 23 "" _null_ _null_ _null_ _null_ pg_stat_get_backend_idset _null_ _null_ _null_ ));
DESCR("statistics: currently active backend IDs");
-DATA(insert OID = 2022 ( pg_stat_get_activity PGNSP PGUID 12 1 100 0 0 f f f f f t s 1 0 2249 "23" "{23,26,23,26,25,25,25,16,1184,1184,1184,1184,869,25,23}" "{i,o,o,o,o,o,o,o,o,o,o,o,o,o,o}" "{pid,datid,pid,usesysid,application_name,state,query,waiting,xact_start,query_start,backend_start,state_change,client_addr,client_hostname,client_port}" _null_ pg_stat_get_activity _null_ _null_ _null_ ));
+DATA(insert OID = 2022 ( pg_stat_get_activity PGNSP PGUID 12 1 100 0 0 f f f f f t s 1 0 2249 "23" "{23,26,23,26,25,25,25,16,1184,1184,1184,1184,869,25,23,28,28}" "{i,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o}" "{pid,datid,pid,usesysid,application_name,state,query,waiting,xact_start,query_start,backend_start,state_change,client_addr,client_hostname,client_port,backend_xid,backend_xmin}" _null_ pg_stat_get_activity _null_ _null_ _null_ ));
DESCR("statistics: information about currently active backends");
DATA(insert OID = 3099 ( pg_stat_get_wal_senders PGNSP PGUID 12 1 10 0 0 f f f f f t s 0 0 2249 "" "{23,25,3220,3220,3220,3220,23,25}" "{o,o,o,o,o,o,o,o}" "{pid,state,sent_location,write_location,flush_location,replay_location,sync_priority,sync_state}" _null_ pg_stat_get_wal_senders _null_ _null_ _null_ ));
DESCR("statistics: information about currently active replication");
char *st_activity;
} PgBackendStatus;
+/* ----------
+ * LocalPgBackendStatus
+ *
+ * When we build the backend status array, we use LocalPgBackendStatus to be
+ * able to add new values to the struct when needed without adding new fields
+ * to the shared memory. It contains the backend status as a first member.
+ * ----------
+ */
+typedef struct LocalPgBackendStatus
+{
+ /*
+ * Local version of the backend status entry.
+ */
+ PgBackendStatus backendStatus;
+
+ /*
+ * The xid of the current transaction if available, InvalidTransactionId
+ * if not.
+ */
+ TransactionId backend_xid;
+
+ /*
+ * The xmin of the current session if available, InvalidTransactionId
+ * if not.
+ */
+ TransactionId backend_xmin;
+} LocalPgBackendStatus;
+
/*
* Working state needed to accumulate per-function-call timing statistics.
*/
extern PgStat_StatDBEntry *pgstat_fetch_stat_dbentry(Oid dbid);
extern PgStat_StatTabEntry *pgstat_fetch_stat_tabentry(Oid relid);
extern PgBackendStatus *pgstat_fetch_stat_beentry(int beid);
+extern LocalPgBackendStatus *pgstat_fetch_stat_local_beentry(int beid);
extern PgStat_StatFuncEntry *pgstat_fetch_stat_funcentry(Oid funcid);
extern int pgstat_fetch_stat_numbackends(void);
extern PgStat_ArchiverStats *pgstat_fetch_stat_archiver(void);
extern void CreateSharedInvalidationState(void);
extern void SharedInvalBackendInit(bool sendOnly);
extern PGPROC *BackendIdGetProc(int backendID);
+extern void BackendIdGetTransactionIds(int backendID, TransactionId *xid, TransactionId *xmin);
extern void SIInsertDataEntries(const SharedInvalidationMessage *data, int n);
extern int SIGetDataEntries(SharedInvalidationMessage *data, int datasize);
s.state_change,
s.waiting,
s.state,
+ s.backend_xid,
+ s.backend_xmin,
s.query
FROM pg_database d,
- pg_stat_get_activity(NULL::integer) s(datid, pid, usesysid, application_name, state, query, waiting, xact_start, query_start, backend_start, state_change, client_addr, client_hostname, client_port),
+ pg_stat_get_activity(NULL::integer) s(datid, pid, usesysid, application_name, state, query, waiting, xact_start, query_start, backend_start, state_change, client_addr, client_hostname, client_port, backend_xid, backend_xmin),
pg_authid u
WHERE ((s.datid = d.oid) AND (s.usesysid = u.oid));
pg_stat_all_indexes| SELECT c.oid AS relid,
s.client_hostname,
s.client_port,
s.backend_start,
+ s.backend_xmin,
w.state,
w.sent_location,
w.write_location,
w.replay_location,
w.sync_priority,
w.sync_state
- FROM pg_stat_get_activity(NULL::integer) s(datid, pid, usesysid, application_name, state, query, waiting, xact_start, query_start, backend_start, state_change, client_addr, client_hostname, client_port),
+ FROM pg_stat_get_activity(NULL::integer) s(datid, pid, usesysid, application_name, state, query, waiting, xact_start, query_start, backend_start, state_change, client_addr, client_hostname, client_port, backend_xid, backend_xmin),
pg_authid u,
pg_stat_get_wal_senders() w(pid, state, sent_location, write_location, flush_location, replay_location, sync_priority, sync_state)
WHERE ((s.usesysid = u.oid) AND (s.pid = w.pid));