LCOV - code coverage report
Current view: top level - contrib/pageinspect - rawpage.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 107 117 91.5 %
Date: 2025-01-18 04:15:08 Functions: 18 18 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * rawpage.c
       4             :  *    Functions to extract a raw page as bytea and inspect it
       5             :  *
       6             :  * Access-method specific inspection functions are in separate files.
       7             :  *
       8             :  * Copyright (c) 2007-2025, PostgreSQL Global Development Group
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    contrib/pageinspect/rawpage.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : 
      16             : #include "postgres.h"
      17             : 
      18             : #include "access/htup_details.h"
      19             : #include "access/relation.h"
      20             : #include "catalog/namespace.h"
      21             : #include "catalog/pg_type.h"
      22             : #include "funcapi.h"
      23             : #include "miscadmin.h"
      24             : #include "pageinspect.h"
      25             : #include "storage/bufmgr.h"
      26             : #include "storage/checksum.h"
      27             : #include "utils/builtins.h"
      28             : #include "utils/pg_lsn.h"
      29             : #include "utils/rel.h"
      30             : #include "utils/varlena.h"
      31             : 
      32          50 : PG_MODULE_MAGIC;
      33             : 
      34             : static bytea *get_raw_page_internal(text *relname, ForkNumber forknum,
      35             :                                     BlockNumber blkno);
      36             : 
      37             : 
      38             : /*
      39             :  * get_raw_page
      40             :  *
      41             :  * Returns a copy of a page from shared buffers as a bytea
      42             :  */
      43          46 : PG_FUNCTION_INFO_V1(get_raw_page_1_9);
      44             : 
      45             : Datum
      46         234 : get_raw_page_1_9(PG_FUNCTION_ARGS)
      47             : {
      48         234 :     text       *relname = PG_GETARG_TEXT_PP(0);
      49         234 :     int64       blkno = PG_GETARG_INT64(1);
      50             :     bytea      *raw_page;
      51             : 
      52         234 :     if (blkno < 0 || blkno > MaxBlockNumber)
      53           2 :         ereport(ERROR,
      54             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
      55             :                  errmsg("invalid block number")));
      56             : 
      57         232 :     raw_page = get_raw_page_internal(relname, MAIN_FORKNUM, blkno);
      58             : 
      59         222 :     PG_RETURN_BYTEA_P(raw_page);
      60             : }
      61             : 
      62             : /*
      63             :  * entry point for old extension version
      64             :  */
      65          14 : PG_FUNCTION_INFO_V1(get_raw_page);
      66             : 
      67             : Datum
      68           4 : get_raw_page(PG_FUNCTION_ARGS)
      69             : {
      70           4 :     text       *relname = PG_GETARG_TEXT_PP(0);
      71           4 :     uint32      blkno = PG_GETARG_UINT32(1);
      72             :     bytea      *raw_page;
      73             : 
      74             :     /*
      75             :      * We don't normally bother to check the number of arguments to a C
      76             :      * function, but here it's needed for safety because early 8.4 beta
      77             :      * releases mistakenly redefined get_raw_page() as taking three arguments.
      78             :      */
      79           4 :     if (PG_NARGS() != 2)
      80           0 :         ereport(ERROR,
      81             :                 (errmsg("wrong number of arguments to get_raw_page()"),
      82             :                  errhint("Run the updated pageinspect.sql script.")));
      83             : 
      84           4 :     raw_page = get_raw_page_internal(relname, MAIN_FORKNUM, blkno);
      85             : 
      86           4 :     PG_RETURN_BYTEA_P(raw_page);
      87             : }
      88             : 
      89             : /*
      90             :  * get_raw_page_fork
      91             :  *
      92             :  * Same, for any fork
      93             :  */
      94          16 : PG_FUNCTION_INFO_V1(get_raw_page_fork_1_9);
      95             : 
      96             : Datum
      97          24 : get_raw_page_fork_1_9(PG_FUNCTION_ARGS)
      98             : {
      99          24 :     text       *relname = PG_GETARG_TEXT_PP(0);
     100          24 :     text       *forkname = PG_GETARG_TEXT_PP(1);
     101          24 :     int64       blkno = PG_GETARG_INT64(2);
     102             :     bytea      *raw_page;
     103             :     ForkNumber  forknum;
     104             : 
     105          24 :     forknum = forkname_to_number(text_to_cstring(forkname));
     106             : 
     107          22 :     if (blkno < 0 || blkno > MaxBlockNumber)
     108           2 :         ereport(ERROR,
     109             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     110             :                  errmsg("invalid block number")));
     111             : 
     112          20 :     raw_page = get_raw_page_internal(relname, forknum, blkno);
     113             : 
     114          14 :     PG_RETURN_BYTEA_P(raw_page);
     115             : }
     116             : 
     117             : /*
     118             :  * Entry point for old extension version
     119             :  */
     120          14 : PG_FUNCTION_INFO_V1(get_raw_page_fork);
     121             : 
     122             : Datum
     123           2 : get_raw_page_fork(PG_FUNCTION_ARGS)
     124             : {
     125           2 :     text       *relname = PG_GETARG_TEXT_PP(0);
     126           2 :     text       *forkname = PG_GETARG_TEXT_PP(1);
     127           2 :     uint32      blkno = PG_GETARG_UINT32(2);
     128             :     bytea      *raw_page;
     129             :     ForkNumber  forknum;
     130             : 
     131           2 :     forknum = forkname_to_number(text_to_cstring(forkname));
     132             : 
     133           2 :     raw_page = get_raw_page_internal(relname, forknum, blkno);
     134             : 
     135           2 :     PG_RETURN_BYTEA_P(raw_page);
     136             : }
     137             : 
     138             : /*
     139             :  * workhorse
     140             :  */
     141             : static bytea *
     142         258 : get_raw_page_internal(text *relname, ForkNumber forknum, BlockNumber blkno)
     143             : {
     144             :     bytea      *raw_page;
     145             :     RangeVar   *relrv;
     146             :     Relation    rel;
     147             :     char       *raw_page_data;
     148             :     Buffer      buf;
     149             : 
     150         258 :     if (!superuser())
     151           0 :         ereport(ERROR,
     152             :                 (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
     153             :                  errmsg("must be superuser to use raw page functions")));
     154             : 
     155         258 :     relrv = makeRangeVarFromNameList(textToQualifiedNameList(relname));
     156         258 :     rel = relation_openrv(relrv, AccessShareLock);
     157             : 
     158         256 :     if (!RELKIND_HAS_STORAGE(rel->rd_rel->relkind))
     159           4 :         ereport(ERROR,
     160             :                 (errcode(ERRCODE_WRONG_OBJECT_TYPE),
     161             :                  errmsg("cannot get raw page from relation \"%s\"",
     162             :                         RelationGetRelationName(rel)),
     163             :                  errdetail_relkind_not_supported(rel->rd_rel->relkind)));
     164             : 
     165             :     /*
     166             :      * Reject attempts to read non-local temporary relations; we would be
     167             :      * likely to get wrong data since we have no visibility into the owning
     168             :      * session's local buffers.
     169             :      */
     170         252 :     if (RELATION_IS_OTHER_TEMP(rel))
     171           0 :         ereport(ERROR,
     172             :                 (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     173             :                  errmsg("cannot access temporary tables of other sessions")));
     174             : 
     175         252 :     if (blkno >= RelationGetNumberOfBlocksInFork(rel, forknum))
     176          10 :         ereport(ERROR,
     177             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     178             :                  errmsg("block number %u is out of range for relation \"%s\"",
     179             :                         blkno, RelationGetRelationName(rel))));
     180             : 
     181             :     /* Initialize buffer to copy to */
     182         242 :     raw_page = (bytea *) palloc(BLCKSZ + VARHDRSZ);
     183         242 :     SET_VARSIZE(raw_page, BLCKSZ + VARHDRSZ);
     184         242 :     raw_page_data = VARDATA(raw_page);
     185             : 
     186             :     /* Take a verbatim copy of the page */
     187             : 
     188         242 :     buf = ReadBufferExtended(rel, forknum, blkno, RBM_NORMAL, NULL);
     189         242 :     LockBuffer(buf, BUFFER_LOCK_SHARE);
     190             : 
     191         242 :     memcpy(raw_page_data, BufferGetPage(buf), BLCKSZ);
     192             : 
     193         242 :     LockBuffer(buf, BUFFER_LOCK_UNLOCK);
     194         242 :     ReleaseBuffer(buf);
     195             : 
     196         242 :     relation_close(rel, AccessShareLock);
     197             : 
     198         242 :     return raw_page;
     199             : }
     200             : 
     201             : 
     202             : /*
     203             :  * get_page_from_raw
     204             :  *
     205             :  * Get a palloc'd, maxalign'ed page image from the result of get_raw_page()
     206             :  *
     207             :  * On machines with MAXALIGN = 8, the payload of a bytea is not maxaligned,
     208             :  * since it will start 4 bytes into a palloc'd value.  On alignment-picky
     209             :  * machines, this will cause failures in accesses to 8-byte-wide values
     210             :  * within the page.  We don't need to worry if accessing only 4-byte or
     211             :  * smaller fields, but when examining a struct that contains 8-byte fields,
     212             :  * use this function for safety.
     213             :  */
     214             : Page
     215         306 : get_page_from_raw(bytea *raw_page)
     216             : {
     217             :     Page        page;
     218             :     int         raw_page_size;
     219             : 
     220         306 :     raw_page_size = VARSIZE_ANY_EXHDR(raw_page);
     221             : 
     222         306 :     if (raw_page_size != BLCKSZ)
     223          28 :         ereport(ERROR,
     224             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     225             :                  errmsg("invalid page size"),
     226             :                  errdetail("Expected %d bytes, got %d.",
     227             :                            BLCKSZ, raw_page_size)));
     228             : 
     229         278 :     page = palloc(raw_page_size);
     230             : 
     231         278 :     memcpy(page, VARDATA_ANY(raw_page), raw_page_size);
     232             : 
     233         278 :     return page;
     234             : }
     235             : 
     236             : 
     237             : /*
     238             :  * page_header
     239             :  *
     240             :  * Allows inspection of page header fields of a raw page
     241             :  */
     242             : 
     243          28 : PG_FUNCTION_INFO_V1(page_header);
     244             : 
     245             : Datum
     246           8 : page_header(PG_FUNCTION_ARGS)
     247             : {
     248           8 :     bytea      *raw_page = PG_GETARG_BYTEA_P(0);
     249             : 
     250             :     TupleDesc   tupdesc;
     251             : 
     252             :     Datum       result;
     253             :     HeapTuple   tuple;
     254             :     Datum       values[9];
     255             :     bool        nulls[9];
     256             : 
     257             :     Page        page;
     258             :     PageHeader  pageheader;
     259             :     XLogRecPtr  lsn;
     260             : 
     261           8 :     if (!superuser())
     262           0 :         ereport(ERROR,
     263             :                 (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
     264             :                  errmsg("must be superuser to use raw page functions")));
     265             : 
     266           8 :     page = get_page_from_raw(raw_page);
     267           6 :     pageheader = (PageHeader) page;
     268             : 
     269             :     /* Build a tuple descriptor for our result type */
     270           6 :     if (get_call_result_type(fcinfo, NULL, &tupdesc) != TYPEFUNC_COMPOSITE)
     271           0 :         elog(ERROR, "return type must be a row type");
     272             : 
     273             :     /* Extract information from the page header */
     274             : 
     275           6 :     lsn = PageGetLSN(page);
     276             : 
     277             :     /* pageinspect >= 1.2 uses pg_lsn instead of text for the LSN field. */
     278           6 :     if (TupleDescAttr(tupdesc, 0)->atttypid == TEXTOID)
     279             :     {
     280             :         char        lsnchar[64];
     281             : 
     282           0 :         snprintf(lsnchar, sizeof(lsnchar), "%X/%X", LSN_FORMAT_ARGS(lsn));
     283           0 :         values[0] = CStringGetTextDatum(lsnchar);
     284             :     }
     285             :     else
     286           6 :         values[0] = LSNGetDatum(lsn);
     287           6 :     values[1] = UInt16GetDatum(pageheader->pd_checksum);
     288           6 :     values[2] = UInt16GetDatum(pageheader->pd_flags);
     289             : 
     290             :     /* pageinspect >= 1.10 uses int4 instead of int2 for those fields */
     291           6 :     switch (TupleDescAttr(tupdesc, 3)->atttypid)
     292             :     {
     293           2 :         case INT2OID:
     294             :             Assert(TupleDescAttr(tupdesc, 4)->atttypid == INT2OID &&
     295             :                    TupleDescAttr(tupdesc, 5)->atttypid == INT2OID &&
     296             :                    TupleDescAttr(tupdesc, 6)->atttypid == INT2OID);
     297           2 :             values[3] = UInt16GetDatum(pageheader->pd_lower);
     298           2 :             values[4] = UInt16GetDatum(pageheader->pd_upper);
     299           2 :             values[5] = UInt16GetDatum(pageheader->pd_special);
     300           2 :             values[6] = UInt16GetDatum(PageGetPageSize(page));
     301           2 :             break;
     302           4 :         case INT4OID:
     303             :             Assert(TupleDescAttr(tupdesc, 4)->atttypid == INT4OID &&
     304             :                    TupleDescAttr(tupdesc, 5)->atttypid == INT4OID &&
     305             :                    TupleDescAttr(tupdesc, 6)->atttypid == INT4OID);
     306           4 :             values[3] = Int32GetDatum(pageheader->pd_lower);
     307           4 :             values[4] = Int32GetDatum(pageheader->pd_upper);
     308           4 :             values[5] = Int32GetDatum(pageheader->pd_special);
     309           4 :             values[6] = Int32GetDatum(PageGetPageSize(page));
     310           4 :             break;
     311           0 :         default:
     312           0 :             elog(ERROR, "incorrect output types");
     313             :             break;
     314             :     }
     315             : 
     316           6 :     values[7] = UInt16GetDatum(PageGetPageLayoutVersion(page));
     317           6 :     values[8] = TransactionIdGetDatum(pageheader->pd_prune_xid);
     318             : 
     319             :     /* Build and return the tuple. */
     320             : 
     321           6 :     memset(nulls, 0, sizeof(nulls));
     322             : 
     323           6 :     tuple = heap_form_tuple(tupdesc, values, nulls);
     324           6 :     result = HeapTupleGetDatum(tuple);
     325             : 
     326           6 :     PG_RETURN_DATUM(result);
     327             : }
     328             : 
     329             : /*
     330             :  * page_checksum
     331             :  *
     332             :  * Compute checksum of a raw page
     333             :  */
     334             : 
     335          16 : PG_FUNCTION_INFO_V1(page_checksum_1_9);
     336          14 : PG_FUNCTION_INFO_V1(page_checksum);
     337             : 
     338             : static Datum
     339          46 : page_checksum_internal(PG_FUNCTION_ARGS, enum pageinspect_version ext_version)
     340             : {
     341          46 :     bytea      *raw_page = PG_GETARG_BYTEA_P(0);
     342          46 :     int64       blkno = (ext_version == PAGEINSPECT_V1_8 ? PG_GETARG_UINT32(1) : PG_GETARG_INT64(1));
     343             :     Page        page;
     344             : 
     345          46 :     if (!superuser())
     346           0 :         ereport(ERROR,
     347             :                 (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
     348             :                  errmsg("must be superuser to use raw page functions")));
     349             : 
     350          46 :     if (blkno < 0 || blkno > MaxBlockNumber)
     351           2 :         ereport(ERROR,
     352             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     353             :                  errmsg("invalid block number")));
     354             : 
     355          44 :     page = get_page_from_raw(raw_page);
     356             : 
     357          42 :     if (PageIsNew(page))
     358           2 :         PG_RETURN_NULL();
     359             : 
     360          40 :     PG_RETURN_INT16(pg_checksum_page((char *) page, blkno));
     361             : }
     362             : 
     363             : Datum
     364          44 : page_checksum_1_9(PG_FUNCTION_ARGS)
     365             : {
     366          44 :     return page_checksum_internal(fcinfo, PAGEINSPECT_V1_9);
     367             : }
     368             : 
     369             : /*
     370             :  * Entry point for old extension version
     371             :  */
     372             : Datum
     373           2 : page_checksum(PG_FUNCTION_ARGS)
     374             : {
     375           2 :     return page_checksum_internal(fcinfo, PAGEINSPECT_V1_8);
     376             : }

Generated by: LCOV version 1.14