86{
90 int ntids,
91 nblocks;
94 next_start_ptr;
96
99 (
errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
100 errmsg(
"recovery is in progress"),
101 errhint(
"Heap surgery functions cannot be executed during recovery.")));
102
103
105
107
108
109
110
111 if (!RELKIND_HAS_TABLE_AM(rel->
rd_rel->relkind))
113 (
errcode(ERRCODE_WRONG_OBJECT_TYPE),
114 errmsg(
"cannot operate on relation \"%s\"",
117
118 if (rel->
rd_rel->relam != HEAP_TABLE_AM_OID)
120 (
errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
121 errmsg(
"only heap AM is supported")));
122
123
128
130
131
132
133
134
135
136 if (ntids > 1)
138
139 curr_start_ptr = next_start_ptr = 0;
141
142
143
144
145 while (next_start_ptr != ntids)
146 {
154 bool did_modify_page = false;
155 bool did_modify_vm = false;
156
158
159
160
161
162
164
165
166 if (blkno >= nblocks)
167 {
168
169 curr_start_ptr = next_start_ptr;
170
172 (
errcode(ERRCODE_INVALID_PARAMETER_VALUE),
173 errmsg(
"skipping block %u for relation \"%s\" because the block number is out of range",
175 continue;
176 }
177
180
182
184
185
186
187
188
189 memset(include_this_tid, 0, sizeof(include_this_tid));
190 for (
i = curr_start_ptr;
i < next_start_ptr;
i++)
191 {
194
195
197 {
199 errmsg(
"skipping tid (%u, %u) for relation \"%s\" because the item number is out of range",
201 continue;
202 }
203
205
206
208 {
210 errmsg(
"skipping tid (%u, %u) for relation \"%s\" because it redirects to item %u",
213 continue;
214 }
216 {
218 (
errmsg(
"skipping tid (%u, %u) for relation \"%s\" because it is marked dead",
220 continue;
221 }
223 {
225 (
errmsg(
"skipping tid (%u, %u) for relation \"%s\" because it is marked unused",
227 continue;
228 }
229
230
232 include_this_tid[offno] = true;
233 }
234
235
236
237
238
241
242
244
247 {
249
250 if (!include_this_tid[curoff])
251 continue;
252
255
256 did_modify_page = true;
257
259 {
261
262
263
264
265
266
268 {
272 did_modify_vm = true;
273 }
274 }
275 else
276 {
278
280
282
283
284
285
286
287
288
293 {
296 else
298 }
299
300
301
302
303
304
309 }
310 }
311
312
313
314
315
316 if (did_modify_page)
317 {
318
320
321
324 }
325
326
329
331
333
336
337
338 curr_start_ptr = next_start_ptr;
339 }
340
342
344
346}
void aclcheck_error(AclResult aclerr, ObjectType objtype, const char *objectname)
bool object_ownercheck(Oid classid, Oid objectid, Oid roleid)
#define PG_GETARG_ARRAYTYPE_P_COPY(n)
void ReleaseBuffer(Buffer buffer)
void UnlockReleaseBuffer(Buffer buffer)
void MarkBufferDirty(Buffer buffer)
void LockBufferForCleanup(Buffer buffer)
Buffer ReadBuffer(Relation reln, BlockNumber blockNum)
#define RelationGetNumberOfBlocks(reln)
static Page BufferGetPage(Buffer buffer)
static bool PageIsAllVisible(const PageData *page)
static void PageClearAllVisible(Page page)
static Item PageGetItem(const PageData *page, const ItemIdData *itemId)
static ItemId PageGetItemId(Page page, OffsetNumber offsetNumber)
static OffsetNumber PageGetMaxOffsetNumber(const PageData *page)
int errhint(const char *fmt,...)
int errcode(int sqlerrcode)
int errmsg(const char *fmt,...)
#define ereport(elevel,...)
Assert(PointerIsAligned(start, uint64))
static int32 tidcmp(const void *a, const void *b)
static BlockNumber find_tids_one_page(ItemPointer tids, int ntids, OffsetNumber *next_start_ptr)
static void sanity_check_tid_array(ArrayType *ta, int *ntids)
HeapTupleHeaderData * HeapTupleHeader
static void HeapTupleHeaderSetXvac(HeapTupleHeaderData *tup, TransactionId xid)
#define HEAP_XMAX_INVALID
#define MaxHeapTuplesPerPage
static void HeapTupleHeaderSetXmin(HeapTupleHeaderData *tup, TransactionId xid)
static void HeapTupleHeaderSetXmax(HeapTupleHeaderData *tup, TransactionId xid)
#define ItemIdIsNormal(itemId)
#define ItemIdGetRedirect(itemId)
#define ItemIdIsDead(itemId)
#define ItemIdSetDead(itemId)
#define ItemIdIsUsed(itemId)
#define ItemIdIsRedirected(itemId)
static void ItemPointerSet(ItemPointerData *pointer, BlockNumber blockNumber, OffsetNumber offNum)
static OffsetNumber ItemPointerGetOffsetNumberNoCheck(const ItemPointerData *pointer)
ItemPointerData * ItemPointer
void pfree(void *pointer)
#define START_CRIT_SECTION()
#define CHECK_FOR_INTERRUPTS()
#define END_CRIT_SECTION()
ObjectType get_relkind_objtype(char relkind)
#define InvalidOffsetNumber
#define OffsetNumberNext(offsetNumber)
#define FirstOffsetNumber
int errdetail_relkind_not_supported(char relkind)
#define qsort(a, b, c, d)
#define RelationGetRelid(relation)
#define RelationGetRelationName(relation)
#define RelationNeedsWAL(relation)
void relation_close(Relation relation, LOCKMODE lockmode)
Relation relation_open(Oid relationId, LOCKMODE lockmode)
#define FrozenTransactionId
#define InvalidTransactionId
bool visibilitymap_clear(Relation rel, BlockNumber heapBlk, Buffer vmbuf, uint8 flags)
void visibilitymap_pin(Relation rel, BlockNumber heapBlk, Buffer *vmbuf)
#define VISIBILITYMAP_VALID_BITS
bool RecoveryInProgress(void)
XLogRecPtr log_newpage_buffer(Buffer buffer, bool page_std)