mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-03-01 19:45:33 +08:00
pgstattuple: Use a BufferAccessStrategy object to avoid cache-trashing.
Jaime Casanova, reviewed by Noah Misch, slightly modified by me.
This commit is contained in:
parent
97c85098de
commit
2e46bf6711
@ -95,6 +95,7 @@ pgstatindex(PG_FUNCTION_ARGS)
|
|||||||
BlockNumber nblocks;
|
BlockNumber nblocks;
|
||||||
BlockNumber blkno;
|
BlockNumber blkno;
|
||||||
BTIndexStat indexStat;
|
BTIndexStat indexStat;
|
||||||
|
BufferAccessStrategy bstrategy = GetAccessStrategy(BAS_BULKREAD);
|
||||||
|
|
||||||
if (!superuser())
|
if (!superuser())
|
||||||
ereport(ERROR,
|
ereport(ERROR,
|
||||||
@ -122,7 +123,7 @@ pgstatindex(PG_FUNCTION_ARGS)
|
|||||||
* Read metapage
|
* Read metapage
|
||||||
*/
|
*/
|
||||||
{
|
{
|
||||||
Buffer buffer = ReadBuffer(rel, 0);
|
Buffer buffer = ReadBufferExtended(rel, MAIN_FORKNUM, 0, RBM_NORMAL, bstrategy);
|
||||||
Page page = BufferGetPage(buffer);
|
Page page = BufferGetPage(buffer);
|
||||||
BTMetaPageData *metad = BTPageGetMeta(page);
|
BTMetaPageData *metad = BTPageGetMeta(page);
|
||||||
|
|
||||||
@ -159,7 +160,7 @@ pgstatindex(PG_FUNCTION_ARGS)
|
|||||||
CHECK_FOR_INTERRUPTS();
|
CHECK_FOR_INTERRUPTS();
|
||||||
|
|
||||||
/* Read and lock buffer */
|
/* Read and lock buffer */
|
||||||
buffer = ReadBuffer(rel, blkno);
|
buffer = ReadBufferExtended(rel, MAIN_FORKNUM, blkno, RBM_NORMAL, bstrategy);
|
||||||
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
||||||
|
|
||||||
page = BufferGetPage(buffer);
|
page = BufferGetPage(buffer);
|
||||||
|
@ -61,18 +61,22 @@ typedef struct pgstattuple_type
|
|||||||
uint64 free_space; /* free/reusable space in bytes */
|
uint64 free_space; /* free/reusable space in bytes */
|
||||||
} pgstattuple_type;
|
} pgstattuple_type;
|
||||||
|
|
||||||
typedef void (*pgstat_page) (pgstattuple_type *, Relation, BlockNumber);
|
typedef void (*pgstat_page) (pgstattuple_type *, Relation, BlockNumber,
|
||||||
|
BufferAccessStrategy);
|
||||||
|
|
||||||
static Datum build_pgstattuple_type(pgstattuple_type *stat,
|
static Datum build_pgstattuple_type(pgstattuple_type *stat,
|
||||||
FunctionCallInfo fcinfo);
|
FunctionCallInfo fcinfo);
|
||||||
static Datum pgstat_relation(Relation rel, FunctionCallInfo fcinfo);
|
static Datum pgstat_relation(Relation rel, FunctionCallInfo fcinfo);
|
||||||
static Datum pgstat_heap(Relation rel, FunctionCallInfo fcinfo);
|
static Datum pgstat_heap(Relation rel, FunctionCallInfo fcinfo);
|
||||||
static void pgstat_btree_page(pgstattuple_type *stat,
|
static void pgstat_btree_page(pgstattuple_type *stat,
|
||||||
Relation rel, BlockNumber blkno);
|
Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy);
|
||||||
static void pgstat_hash_page(pgstattuple_type *stat,
|
static void pgstat_hash_page(pgstattuple_type *stat,
|
||||||
Relation rel, BlockNumber blkno);
|
Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy);
|
||||||
static void pgstat_gist_page(pgstattuple_type *stat,
|
static void pgstat_gist_page(pgstattuple_type *stat,
|
||||||
Relation rel, BlockNumber blkno);
|
Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy);
|
||||||
static Datum pgstat_index(Relation rel, BlockNumber start,
|
static Datum pgstat_index(Relation rel, BlockNumber start,
|
||||||
pgstat_page pagefn, FunctionCallInfo fcinfo);
|
pgstat_page pagefn, FunctionCallInfo fcinfo);
|
||||||
static void pgstat_index_page(pgstattuple_type *stat, Page page,
|
static void pgstat_index_page(pgstattuple_type *stat, Page page,
|
||||||
@ -273,12 +277,17 @@ pgstat_heap(Relation rel, FunctionCallInfo fcinfo)
|
|||||||
BlockNumber tupblock;
|
BlockNumber tupblock;
|
||||||
Buffer buffer;
|
Buffer buffer;
|
||||||
pgstattuple_type stat = {0};
|
pgstattuple_type stat = {0};
|
||||||
|
BufferAccessStrategy bstrategy;
|
||||||
|
|
||||||
/* Disable syncscan because we assume we scan from block zero upwards */
|
/* Disable syncscan because we assume we scan from block zero upwards */
|
||||||
scan = heap_beginscan_strat(rel, SnapshotAny, 0, NULL, true, false);
|
scan = heap_beginscan_strat(rel, SnapshotAny, 0, NULL, true, false);
|
||||||
|
|
||||||
nblocks = scan->rs_nblocks; /* # blocks to be scanned */
|
nblocks = scan->rs_nblocks; /* # blocks to be scanned */
|
||||||
|
|
||||||
|
/* prepare access strategy for this table */
|
||||||
|
bstrategy = GetAccessStrategy(BAS_BULKREAD);
|
||||||
|
scan->rs_strategy = bstrategy;
|
||||||
|
|
||||||
/* scan the relation */
|
/* scan the relation */
|
||||||
while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL)
|
while ((tuple = heap_getnext(scan, ForwardScanDirection)) != NULL)
|
||||||
{
|
{
|
||||||
@ -312,7 +321,7 @@ pgstat_heap(Relation rel, FunctionCallInfo fcinfo)
|
|||||||
{
|
{
|
||||||
CHECK_FOR_INTERRUPTS();
|
CHECK_FOR_INTERRUPTS();
|
||||||
|
|
||||||
buffer = ReadBuffer(rel, block);
|
buffer = ReadBufferExtended(rel, MAIN_FORKNUM, block, RBM_NORMAL, bstrategy);
|
||||||
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
||||||
stat.free_space += PageGetHeapFreeSpace((Page) BufferGetPage(buffer));
|
stat.free_space += PageGetHeapFreeSpace((Page) BufferGetPage(buffer));
|
||||||
UnlockReleaseBuffer(buffer);
|
UnlockReleaseBuffer(buffer);
|
||||||
@ -325,7 +334,7 @@ pgstat_heap(Relation rel, FunctionCallInfo fcinfo)
|
|||||||
{
|
{
|
||||||
CHECK_FOR_INTERRUPTS();
|
CHECK_FOR_INTERRUPTS();
|
||||||
|
|
||||||
buffer = ReadBuffer(rel, block);
|
buffer = ReadBufferExtended(rel, MAIN_FORKNUM, block, RBM_NORMAL, bstrategy);
|
||||||
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
LockBuffer(buffer, BUFFER_LOCK_SHARE);
|
||||||
stat.free_space += PageGetHeapFreeSpace((Page) BufferGetPage(buffer));
|
stat.free_space += PageGetHeapFreeSpace((Page) BufferGetPage(buffer));
|
||||||
UnlockReleaseBuffer(buffer);
|
UnlockReleaseBuffer(buffer);
|
||||||
@ -343,12 +352,13 @@ pgstat_heap(Relation rel, FunctionCallInfo fcinfo)
|
|||||||
* pgstat_btree_page -- check tuples in a btree page
|
* pgstat_btree_page -- check tuples in a btree page
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
pgstat_btree_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno)
|
pgstat_btree_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy)
|
||||||
{
|
{
|
||||||
Buffer buf;
|
Buffer buf;
|
||||||
Page page;
|
Page page;
|
||||||
|
|
||||||
buf = ReadBuffer(rel, blkno);
|
buf = ReadBufferExtended(rel, MAIN_FORKNUM, blkno, RBM_NORMAL, bstrategy);
|
||||||
LockBuffer(buf, BT_READ);
|
LockBuffer(buf, BT_READ);
|
||||||
page = BufferGetPage(buf);
|
page = BufferGetPage(buf);
|
||||||
|
|
||||||
@ -386,13 +396,14 @@ pgstat_btree_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno)
|
|||||||
* pgstat_hash_page -- check tuples in a hash page
|
* pgstat_hash_page -- check tuples in a hash page
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
pgstat_hash_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno)
|
pgstat_hash_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy)
|
||||||
{
|
{
|
||||||
Buffer buf;
|
Buffer buf;
|
||||||
Page page;
|
Page page;
|
||||||
|
|
||||||
_hash_getlock(rel, blkno, HASH_SHARE);
|
_hash_getlock(rel, blkno, HASH_SHARE);
|
||||||
buf = _hash_getbuf(rel, blkno, HASH_READ, 0);
|
buf = _hash_getbuf_with_strategy(rel, blkno, HASH_READ, 0, bstrategy);
|
||||||
page = BufferGetPage(buf);
|
page = BufferGetPage(buf);
|
||||||
|
|
||||||
if (PageGetSpecialSize(page) == MAXALIGN(sizeof(HashPageOpaqueData)))
|
if (PageGetSpecialSize(page) == MAXALIGN(sizeof(HashPageOpaqueData)))
|
||||||
@ -429,12 +440,13 @@ pgstat_hash_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno)
|
|||||||
* pgstat_gist_page -- check tuples in a gist page
|
* pgstat_gist_page -- check tuples in a gist page
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
pgstat_gist_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno)
|
pgstat_gist_page(pgstattuple_type *stat, Relation rel, BlockNumber blkno,
|
||||||
|
BufferAccessStrategy bstrategy)
|
||||||
{
|
{
|
||||||
Buffer buf;
|
Buffer buf;
|
||||||
Page page;
|
Page page;
|
||||||
|
|
||||||
buf = ReadBuffer(rel, blkno);
|
buf = ReadBufferExtended(rel, MAIN_FORKNUM, blkno, RBM_NORMAL, bstrategy);
|
||||||
LockBuffer(buf, GIST_SHARE);
|
LockBuffer(buf, GIST_SHARE);
|
||||||
gistcheckpage(rel, buf);
|
gistcheckpage(rel, buf);
|
||||||
page = BufferGetPage(buf);
|
page = BufferGetPage(buf);
|
||||||
@ -461,8 +473,12 @@ pgstat_index(Relation rel, BlockNumber start, pgstat_page pagefn,
|
|||||||
{
|
{
|
||||||
BlockNumber nblocks;
|
BlockNumber nblocks;
|
||||||
BlockNumber blkno;
|
BlockNumber blkno;
|
||||||
|
BufferAccessStrategy bstrategy;
|
||||||
pgstattuple_type stat = {0};
|
pgstattuple_type stat = {0};
|
||||||
|
|
||||||
|
/* prepare access strategy for this index */
|
||||||
|
bstrategy = GetAccessStrategy(BAS_BULKREAD);
|
||||||
|
|
||||||
blkno = start;
|
blkno = start;
|
||||||
for (;;)
|
for (;;)
|
||||||
{
|
{
|
||||||
@ -483,7 +499,7 @@ pgstat_index(Relation rel, BlockNumber start, pgstat_page pagefn,
|
|||||||
{
|
{
|
||||||
CHECK_FOR_INTERRUPTS();
|
CHECK_FOR_INTERRUPTS();
|
||||||
|
|
||||||
pagefn(&stat, rel, blkno);
|
pagefn(&stat, rel, blkno, bstrategy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user