|
|
@ -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); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|