blob: f32b0acd442ece7e4e4a2768a726de1f6c91745f [file] [log] [blame]
/*-------------------------------------------------------------------------
*
* itemptr.c
* POSTGRES disk item pointer code.
*
* Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* IDENTIFICATION
* src/backend/storage/page/itemptr.c
*
*-------------------------------------------------------------------------
*/
#include "postgres.h"
#include "storage/itemptr.h"
/*
* We really want ItemPointerData to be exactly 6 bytes.
*/
StaticAssertDecl(sizeof(ItemPointerData) == 3 * sizeof(uint16),
"ItemPointerData struct is improperly padded");
/*
* ItemPointerEquals
* Returns true if both item pointers point to the same item,
* otherwise returns false.
*
* Note:
* Asserts that the disk item pointers are both valid!
*/
bool
ItemPointerEquals(ItemPointer pointer1, ItemPointer pointer2)
{
if (ItemPointerGetBlockNumber(pointer1) ==
ItemPointerGetBlockNumber(pointer2) &&
ItemPointerGetOffsetNumber(pointer1) ==
ItemPointerGetOffsetNumber(pointer2))
return true;
else
return false;
}
/*
* ItemPointerCompare
* Generic btree-style comparison for item pointers.
*/
int32
ItemPointerCompare(ItemPointer arg1, ItemPointer arg2)
{
/*
* Use ItemPointerGet{Offset,Block}NumberNoCheck to avoid asserting
* ip_posid != 0, which may not be true for a user-supplied TID.
*/
BlockNumber b1 = ItemPointerGetBlockNumberNoCheck(arg1);
BlockNumber b2 = ItemPointerGetBlockNumberNoCheck(arg2);
if (b1 < b2)
return -1;
else if (b1 > b2)
return 1;
else if (ItemPointerGetOffsetNumberNoCheck(arg1) <
ItemPointerGetOffsetNumberNoCheck(arg2))
return -1;
else if (ItemPointerGetOffsetNumberNoCheck(arg1) >
ItemPointerGetOffsetNumberNoCheck(arg2))
return 1;
else
return 0;
}
static char *
ItemPointerToBuffer(char *buffer, ItemPointer tid)
{
// Do not assert valid ItemPointer -- it is ok if it is (0,0)...
BlockNumber blockNumber = BlockIdGetBlockNumber(&tid->ip_blkid);
OffsetNumber offsetNumber = tid->ip_posid;
sprintf(buffer,
"(%u,%u)",
blockNumber,
offsetNumber);
return buffer;
}
static char itemPointerBuffer[50];
static char itemPointerBuffer2[50];
char *
ItemPointerToString(ItemPointer tid)
{
return ItemPointerToBuffer(itemPointerBuffer, tid);
}
char *
ItemPointerToString2(ItemPointer tid)
{
return ItemPointerToBuffer(itemPointerBuffer2, tid);
}
/*
* ItemPointerInc
* Increment 'pointer' by 1 only paying attention to the ItemPointer's
* type's range limits and not MaxOffsetNumber and FirstOffsetNumber.
* This may result in 'pointer' becoming !OffsetNumberIsValid.
*
* If the pointer is already the maximum possible values permitted by the
* range of the ItemPointer's types, then do nothing.
*/
void
ItemPointerInc(ItemPointer pointer)
{
BlockNumber blk = ItemPointerGetBlockNumberNoCheck(pointer);
OffsetNumber off = ItemPointerGetOffsetNumberNoCheck(pointer);
if (off == PG_UINT16_MAX)
{
if (blk != InvalidBlockNumber)
{
off = 0;
blk++;
}
}
else
off++;
ItemPointerSet(pointer, blk, off);
}
/*
* ItemPointerDec
* Decrement 'pointer' by 1 only paying attention to the ItemPointer's
* type's range limits and not MaxOffsetNumber and FirstOffsetNumber.
* This may result in 'pointer' becoming !OffsetNumberIsValid.
*
* If the pointer is already the minimum possible values permitted by the
* range of the ItemPointer's types, then do nothing. This does rely on
* FirstOffsetNumber being 1 rather than 0.
*/
void
ItemPointerDec(ItemPointer pointer)
{
BlockNumber blk = ItemPointerGetBlockNumberNoCheck(pointer);
OffsetNumber off = ItemPointerGetOffsetNumberNoCheck(pointer);
if (off == 0)
{
if (blk != 0)
{
off = PG_UINT16_MAX;
blk--;
}
}
else
off--;
ItemPointerSet(pointer, blk, off);
}