Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
P
Postgres FD Implementation
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Abuhujair Javed
Postgres FD Implementation
Commits
f2bfe8a2
Commit
f2bfe8a2
authored
Sep 07, 2000
by
Vadim B. Mikheev
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Heap redo/undo (except for tuple moving used by vacuum).
parent
c18c3213
Changes
7
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
497 additions
and
89 deletions
+497
-89
src/backend/access/heap/heapam.c
src/backend/access/heap/heapam.c
+441
-44
src/backend/access/heap/hio.c
src/backend/access/heap/hio.c
+15
-28
src/include/access/hio.h
src/include/access/hio.h
+2
-2
src/include/access/htup.h
src/include/access/htup.h
+13
-10
src/include/storage/bufpage.h
src/include/storage/bufpage.h
+14
-1
src/include/storage/itemid.h
src/include/storage/itemid.h
+9
-3
src/include/utils/rel.h
src/include/utils/rel.h
+3
-1
No files found.
src/backend/access/heap/heapam.c
View file @
f2bfe8a2
...
@@ -8,7 +8,7 @@
...
@@ -8,7 +8,7 @@
*
*
*
*
* IDENTIFICATION
* IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/access/heap/heapam.c,v 1.8
4 2000/08/04 04:16:06 tgl
Exp $
* $Header: /cvsroot/pgsql/src/backend/access/heap/heapam.c,v 1.8
5 2000/09/07 09:58:34 vadim
Exp $
*
*
*
*
* INTERFACE ROUTINES
* INTERFACE ROUTINES
...
@@ -86,6 +86,10 @@
...
@@ -86,6 +86,10 @@
#include "utils/inval.h"
#include "utils/inval.h"
#include "utils/relcache.h"
#include "utils/relcache.h"
#ifdef XLOG
/* comments are in _heap_update */
static
ItemPointerData
_locked_tuple
;
#endif
/* ----------------------------------------------------------------
/* ----------------------------------------------------------------
* heap support routines
* heap support routines
...
@@ -1367,7 +1371,7 @@ heap_insert(Relation relation, HeapTuple tup)
...
@@ -1367,7 +1371,7 @@ heap_insert(Relation relation, HeapTuple tup)
#endif
#endif
/* Find buffer for this tuple */
/* Find buffer for this tuple */
buffer
=
RelationGetBufferForTuple
(
relation
,
tup
->
t_len
,
InvalidBuffer
);
buffer
=
RelationGetBufferForTuple
(
relation
,
tup
->
t_len
);
/* NO ELOG(ERROR) from here till changes are logged */
/* NO ELOG(ERROR) from here till changes are logged */
RelationPutHeapTuple
(
relation
,
buffer
,
tup
);
RelationPutHeapTuple
(
relation
,
buffer
,
tup
);
...
@@ -1376,10 +1380,9 @@ heap_insert(Relation relation, HeapTuple tup)
...
@@ -1376,10 +1380,9 @@ heap_insert(Relation relation, HeapTuple tup)
/* XLOG stuff */
/* XLOG stuff */
{
{
xl_heap_insert
xlrec
;
xl_heap_insert
xlrec
;
xlrec
.
itid
.
dbId
=
relation
->
rd_lockInfo
.
lockRelId
.
dbId
;
xlrec
.
target
.
node
=
relation
->
rd_node
;
xlrec
.
itid
.
relId
=
relation
->
rd_lockInfo
.
lockRelId
.
relId
;
xlrec
.
target
.
cid
=
GetCurrentCommandId
();
xlrec
.
itid
.
cid
=
GetCurrentCommandId
();
xlrec
.
target
.
tid
=
tup
->
t_self
;
xlrec
.
itid
.
tid
=
tup
->
t_self
;
xlrec
.
t_natts
=
tup
->
t_data
->
t_natts
;
xlrec
.
t_natts
=
tup
->
t_data
->
t_natts
;
xlrec
.
t_oid
=
tup
->
t_data
->
t_oid
;
xlrec
.
t_oid
=
tup
->
t_data
->
t_oid
;
xlrec
.
t_hoff
=
tup
->
t_data
->
t_hoff
;
xlrec
.
t_hoff
=
tup
->
t_data
->
t_hoff
;
...
@@ -1390,8 +1393,8 @@ heap_insert(Relation relation, HeapTuple tup)
...
@@ -1390,8 +1393,8 @@ heap_insert(Relation relation, HeapTuple tup)
(
char
*
)
tup
->
t_data
+
offsetof
(
HeapTupleHeaderData
,
t_bits
),
(
char
*
)
tup
->
t_data
+
offsetof
(
HeapTupleHeaderData
,
t_bits
),
tup
->
t_len
-
offsetof
(
HeapTupleHeaderData
,
t_bits
));
tup
->
t_len
-
offsetof
(
HeapTupleHeaderData
,
t_bits
));
((
PageHeader
)
BufferGetPage
(
buffer
))
->
pd_lsn
=
recptr
;
PageSetLSN
(
BufferGetPage
(
buffer
),
recptr
)
;
((
PageHeader
)
BufferGetPage
(
buffer
))
->
pd_sui
=
ThisStartUpID
;
PageSetSUI
(
BufferGetPage
(
buffer
),
ThisStartUpID
)
;
}
}
#endif
#endif
...
@@ -1490,15 +1493,14 @@ l1:
...
@@ -1490,15 +1493,14 @@ l1:
/* XLOG stuff */
/* XLOG stuff */
{
{
xl_heap_delete
xlrec
;
xl_heap_delete
xlrec
;
xlrec
.
dtid
.
dbId
=
relation
->
rd_lockInfo
.
lockRelId
.
dbId
;
xlrec
.
target
.
node
=
relation
->
rd_node
;
xlrec
.
dtid
.
relId
=
relation
->
rd_lockInfo
.
lockRelId
.
relId
;
xlrec
.
target
.
cid
=
GetCurrentCommandId
();
xlrec
.
dtid
.
cid
=
GetCurrentCommandId
();
xlrec
.
target
.
tid
=
tp
.
t_self
;
xlrec
.
dtid
.
tid
=
tp
.
t_self
;
XLogRecPtr
recptr
=
XLogInsert
(
RM_HEAP_ID
,
XLOG_HEAP_DELETE
,
XLogRecPtr
recptr
=
XLogInsert
(
RM_HEAP_ID
,
XLOG_HEAP_DELETE
,
(
char
*
)
xlrec
,
SizeOfHeapDelete
,
NULL
,
0
);
(
char
*
)
xlrec
,
SizeOfHeapDelete
,
NULL
,
0
);
dp
->
pd_lsn
=
recptr
;
PageSetLSN
(
dp
,
recptr
)
;
dp
->
pd_sui
=
ThisStartUpID
;
PageSetSUI
(
dp
,
ThisStartUpID
)
;
}
}
#endif
#endif
...
@@ -1638,18 +1640,49 @@ l2:
...
@@ -1638,18 +1640,49 @@ l2:
if
((
unsigned
)
MAXALIGN
(
newtup
->
t_len
)
<=
PageGetFreeSpace
((
Page
)
dp
))
if
((
unsigned
)
MAXALIGN
(
newtup
->
t_len
)
<=
PageGetFreeSpace
((
Page
)
dp
))
newbuf
=
buffer
;
newbuf
=
buffer
;
else
else
newbuf
=
RelationGetBufferForTuple
(
relation
,
newtup
->
t_len
,
buffer
);
{
#ifdef XLOG
/*
* We have to unlock old tuple buffer before extending table
* file but have to keep lock on the old tuple. To avoid second
* XLOG log record we use xact mngr hook to unlock old tuple
* without reading log if xact will abort before update is logged.
* In the event of crash prio logging, TQUAL routines will see
* HEAP_XMAX_UNLOGGED flag...
*/
_locked_tuple
=
*
otid
;
XactPushRollback
(
_heap_unlock_tuple
,
(
void
*
)
&
_locked_tuple
);
#endif
TransactionIdStore
(
GetCurrentTransactionId
(),
&
(
oldtup
.
t_data
->
t_xmax
));
oldtup
.
t_data
->
t_cmax
=
GetCurrentCommandId
();
oldtup
.
t_data
->
t_infomask
&=
~
(
HEAP_XMAX_COMMITTED
|
HEAP_XMAX_INVALID
|
HEAP_MARKED_FOR_UPDATE
);
oldtup
.
t_data
->
t_infomask
|=
HEAP_XMAX_UNLOGGED
;
LockBuffer
(
buffer
,
BUFFER_LOCK_UNLOCK
);
newbuf
=
RelationGetBufferForTuple
(
relation
,
newtup
->
t_len
);
/* this seems to be deadlock free... */
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
}
/* NO ELOG(ERROR) from here till changes are logged */
/* NO ELOG(ERROR) from here till changes are logged */
/* insert new tuple */
/* insert new tuple */
RelationPutHeapTuple
(
relation
,
newbuf
,
newtup
);
RelationPutHeapTuple
(
relation
,
newbuf
,
newtup
);
/* logically delete old tuple */
if
(
buffer
==
newbuf
)
{
TransactionIdStore
(
GetCurrentTransactionId
(),
&
(
oldtup
.
t_data
->
t_xmax
));
TransactionIdStore
(
GetCurrentTransactionId
(),
&
(
oldtup
.
t_data
->
t_xmax
));
oldtup
.
t_data
->
t_cmax
=
GetCurrentCommandId
();
oldtup
.
t_data
->
t_cmax
=
GetCurrentCommandId
();
oldtup
.
t_data
->
t_infomask
&=
~
(
HEAP_XMAX_COMMITTED
|
oldtup
.
t_data
->
t_infomask
&=
~
(
HEAP_XMAX_COMMITTED
|
HEAP_XMAX_INVALID
|
HEAP_MARKED_FOR_UPDATE
);
HEAP_XMAX_INVALID
|
HEAP_MARKED_FOR_UPDATE
);
}
else
{
oldtup
.
t_data
->
t_infomask
&=
~
HEAP_XMAX_UNLOGGED
;
#ifdef XLOG
XactPopRollback
();
#endif
}
/* record address of new tuple in t_ctid of old one */
/* record address of new tuple in t_ctid of old one */
oldtup
.
t_data
->
t_ctid
=
newtup
->
t_self
;
oldtup
.
t_data
->
t_ctid
=
newtup
->
t_self
;
...
@@ -1658,10 +1691,10 @@ l2:
...
@@ -1658,10 +1691,10 @@ l2:
/* XLOG stuff */
/* XLOG stuff */
{
{
xl_heap_update
xlrec
;
xl_heap_update
xlrec
;
xlrec
.
dtid
.
dbId
=
relation
->
rd_lockInfo
.
lockRelId
.
dbId
;
xlrec
.
target
.
node
=
relation
->
rd_node
;
xlrec
.
dtid
.
relId
=
relation
->
rd_lockInfo
.
lockRelId
.
relId
;
xlrec
.
target
.
cid
=
GetCurrentCommandId
()
;
xlrec
.
dtid
.
cid
=
GetCurrentCommandId
()
;
xlrec
.
target
.
tid
=
oldtup
.
t_self
;
xlrec
.
i
tid
.
tid
=
newtup
->
t_self
;
xlrec
.
new
tid
.
tid
=
newtup
->
t_self
;
xlrec
.
t_natts
=
newtup
->
t_data
->
t_natts
;
xlrec
.
t_natts
=
newtup
->
t_data
->
t_natts
;
xlrec
.
t_hoff
=
newtup
->
t_data
->
t_hoff
;
xlrec
.
t_hoff
=
newtup
->
t_data
->
t_hoff
;
xlrec
.
mask
=
newtup
->
t_data
->
t_infomask
;
xlrec
.
mask
=
newtup
->
t_data
->
t_infomask
;
...
@@ -1673,11 +1706,11 @@ l2:
...
@@ -1673,11 +1706,11 @@ l2:
if
(
newbuf
!=
buffer
)
if
(
newbuf
!=
buffer
)
{
{
((
PageHeader
)
BufferGetPage
(
newbuf
))
->
pd_lsn
=
recptr
;
PageSetLSN
(
BufferGetPage
(
newbuf
),
recptr
)
;
((
PageHeader
)
BufferGetPage
(
newbuf
))
->
pd_sui
=
ThisStartUpID
;
PageSetSUI
(
BufferGetPage
(
newbuf
),
ThisStartUpID
)
;
}
}
((
PageHeader
)
BufferGetPage
(
buffer
))
->
pd_lsn
=
recptr
;
PageSetLSN
(
BufferGetPage
(
buffer
),
recptr
)
;
((
PageHeader
)
BufferGetPage
(
buffer
))
->
pd_sui
=
ThisStartUpID
;
PageSetSUI
(
BufferGetPage
(
buffer
),
ThisStartUpID
)
;
}
}
#endif
#endif
...
@@ -1982,25 +2015,389 @@ void heap_redo(XLogRecPtr lsn, XLogRecord *record)
...
@@ -1982,25 +2015,389 @@ void heap_redo(XLogRecPtr lsn, XLogRecord *record)
elog
(
STOP
,
"heap_redo: unknown op code %u"
,
info
);
elog
(
STOP
,
"heap_redo: unknown op code %u"
,
info
);
}
}
void
heap_
undo
(
XLogRecPtr
lsn
,
XLogRecord
*
record
)
void
heap_
xlog_delete
(
bool
redo
,
XLogRecPtr
lsn
,
XLogRecord
*
record
)
{
{
uint8
info
=
record
->
xl_info
&
~
XLR_INFO_MASK
;
xl_heap_delete
*
xlrec
=
(
xl_heap_delete
*
)
XLogRecGetData
(
record
);
Relation
reln
=
XLogOpenRelation
(
redo
,
RM_HEAP_ID
,
xlrec
->
target
.
node
);
if
(
info
==
XLOG_HEAP_INSERT
)
if
(
!
RelationIsValid
(
reln
))
heap_xlog_insert
(
false
,
lsn
,
record
);
return
;
else
if
(
info
==
XLOG_HEAP_DELETE
)
Buffer
buffer
=
XLogReadBuffer
(
false
,
reln
,
heap_xlog_delete
(
false
,
lsn
,
record
);
ItemPointerGetBlockNumber
(
&
(
xlrec
->
target
.
tid
)));
else
if
(
info
==
XLOG_HEAP_UPDATE
)
if
(
!
BufferIsValid
(
buffer
))
heap_xlog_update
(
false
,
lsn
,
record
);
return
;
else
if
(
info
==
XLOG_HEAP_MOVE
)
heap_xlog_move
(
false
,
lsn
,
record
);
Page
page
=
(
Page
)
BufferGetPage
(
buffer
);
if
(
PageIsNew
((
PageHeader
)
page
))
{
PageInit
(
page
,
BufferGetPageSize
(
buffer
),
0
);
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
return
;
}
if
(
redo
)
{
if
(
XLByteLE
(
lsn
,
PageGetLSN
(
page
)))
/* changes are applied */
{
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
}
else
if
(
XLByteLT
(
PageGetLSN
(
page
),
lsn
))
/* changes are not applied ?! */
elog
(
STOP
,
"heap_delete_undo: bad page LSN"
);
OffsetNumber
offnum
=
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
target
.
tid
));
ItemId
lp
=
PageGetItemId
(
page
,
offnum
);
if
(
!
ItemIdIsUsed
(
lp
)
||
ItemIdDeleted
(
lp
))
{
if
(
redo
)
elog
(
STOP
,
"heap_delete_redo: unused/deleted target tuple"
);
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_delete_undo: unused/deleted target tuple in rollback"
);
if
(
ItemIdDeleted
(
lp
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
else
elog
(
STOP
,
"heap_undo: unknown op code %u"
,
info
);
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
HeapTupleHeader
htup
=
(
HeapTupleHeader
)
PageGetItem
(
page
,
lp
);
if
(
redo
)
{
htup
->
t_xmax
=
record
->
xl_xid
;
htup
->
t_cmax
=
xlrec
->
target
.
cid
;
htup
->
t_infomask
&=
~
(
HEAP_XMAX_INVALID
|
HEAP_MARKED_FOR_UPDATE
);
htup
->
t_infomask
|=
HEAP_XMAX_COMMITTED
;
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
return
;
}
/* undo... is it our tuple ? */
if
(
htup
->
t_xmax
!=
record
->
xl_xid
||
htup
->
t_cmax
!=
xlrec
->
target
.
cid
)
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_delete_undo: invalid target tuple in rollback"
);
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
else
/* undo DELETE */
{
htup
->
t_infomask
|=
HEAP_XMAX_INVALID
;
UnlockAndWriteBuffer
(
buffer
);
return
;
}
}
}
void
heap_xlog_insert
(
bool
redo
,
XLogRecPtr
lsn
,
XLogRecord
*
record
)
void
heap_xlog_insert
(
bool
redo
,
XLogRecPtr
lsn
,
XLogRecord
*
record
)
{
{
xl_heap_insert
xlrec
=
XLogRecGetData
(
record
);
xl_heap_insert
*
xlrec
=
(
xl_heap_insert
*
)
XLogRecGetData
(
record
);
Relation
reln
=
XLogOpenRelation
(
redo
,
RM_HEAP_ID
,
xlrec
->
target
.
node
);
if
(
!
RelationIsValid
(
reln
))
return
;
Buffer
buffer
=
XLogReadBuffer
((
redo
)
?
true
:
false
,
reln
,
ItemPointerGetBlockNumber
(
&
(
xlrec
->
target
.
tid
)));
if
(
!
BufferIsValid
(
buffer
))
return
;
Page
page
=
(
Page
)
BufferGetPage
(
buffer
);
if
(
PageIsNew
((
PageHeader
)
page
))
{
PageInit
(
page
,
BufferGetPageSize
(
buffer
),
0
);
if
(
!
redo
)
{
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
return
;
}
}
if
(
redo
)
{
if
(
XLByteLE
(
lsn
,
PageGetLSN
(
page
)))
/* changes are applied */
{
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
char
tbuf
[
MaxTupleSize
];
HeapTupleHeader
htup
=
(
HeapTupleHeader
)
tbuf
;
uint32
newlen
=
record
->
xl_len
-
SizeOfHeapInsert
;
memcpy
(
tbuf
+
offsetof
(
HeapTupleHeaderData
,
t_bits
),
(
char
*
)
xlrec
+
SizeOfHeapInsert
,
newlen
);
newlen
+=
offsetof
(
HeapTupleHeaderData
,
t_bits
);
htup
->
t_oid
=
xlrec
->
t_oid
;
htup
->
t_natts
=
xlrec
->
t_natts
;
htup
->
t_hoff
=
xlrec
->
t_hoff
;
htup
->
t_xmin
=
record
->
xl_xid
;
htup
->
t_cmin
=
xlrec
->
target
.
cid
;
htup
->
t_infomask
=
HEAP_XMAX_INVALID
|
HEAP_XMIN_COMMITTED
|
xlrec
->
mask
;
PageManagerModeSet
(
OverwritePageManagerMode
);
OffsetNumber
offnum
=
PageAddItem
(
page
,
htup
,
newlen
,
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
target
.
tid
)),
LP_USED
);
PageManagerModeSet
(
ShufflePageManagerMode
);
if
(
offnum
==
InvalidOffsetNumber
)
elog
(
STOP
,
"heap_insert_redo: failed to add tuple"
);
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
/* prev sui */
UnlockAndWriteBuffer
(
buffer
);
return
;
}
/* undo insert */
if
(
XLByteLT
(
PageGetLSN
(
page
),
lsn
))
/* changes are not applied ?! */
elog
(
STOP
,
"heap_insert_undo: bad page LSN"
);
OffsetNumber
offnum
=
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
target
.
tid
));
ItemId
lp
=
PageGetItemId
(
page
,
offnum
);
if
(
!
ItemIdIsUsed
(
lp
)
||
ItemIdDeleted
(
lp
))
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_insert_undo: unused/deleted target tuple in rollback"
);
if
(
ItemIdDeleted
(
lp
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
HeapTupleHeader
htup
=
(
HeapTupleHeader
)
PageGetItem
(
page
,
lp
);
/* is it our tuple ? */
if
(
htup
->
t_xmin
!=
record
->
xl_xid
||
htup
->
t_cmin
!=
xlrec
->
target
.
cid
)
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_insert_undo: invalid target tuple in rollback"
);
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
if
(
InRecovery
||
BufferIsUpdatable
(
buffer
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
/* we can't delete tuple right now */
{
lp
->
lp_flags
|=
LP_DELETE
;
/* mark for deletion */
MarkBufferForCleanup
(
buffer
,
PageCleanup
);
}
}
void
heap_xlog_update
(
bool
redo
,
XLogRecPtr
lsn
,
XLogRecord
*
record
)
{
xl_heap_update
*
xlrec
=
(
xl_heap_update
*
)
XLogRecGetData
(
record
);
Relation
reln
=
XLogOpenRelation
(
redo
,
RM_HEAP_ID
,
xlrec
->
target
.
node
);
if
(
!
RelationIsValid
(
reln
))
return
;
Buffer
buffer
;
Page
page
;
OffsetNumber
offnum
;
ItemId
lp
;
HeapTupleHeader
htup
;
/*
* Currently UPDATE is DELETE + INSERT and so code below are near
* exact sum of code in heap_xlog_delete & heap_xlog_insert. We could
* re-structure code better, but keeping in mind upcoming overwriting
* smgr separate heap_xlog_update code seems to be Good Thing.
*/
/* Deal with old tuple version */
buffer
=
XLogReadBuffer
(
false
,
reln
,
ItemPointerGetBlockNumber
(
&
(
xlrec
->
target
.
tid
)));
if
(
!
BufferIsValid
(
buffer
))
goto
newt
;
page
=
(
Page
)
BufferGetPage
(
buffer
);
if
(
PageIsNew
((
PageHeader
)
page
))
{
PageInit
(
page
,
BufferGetPageSize
(
buffer
),
0
);
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
goto
newt
;
}
if
(
redo
)
{
if
(
XLByteLE
(
lsn
,
PageGetLSN
(
page
)))
/* changes are applied */
{
UnlockAndReleaseBuffer
(
buffer
);
goto
newt
;
}
}
else
if
(
XLByteLT
(
PageGetLSN
(
page
),
lsn
))
/* changes are not applied ?! */
elog
(
STOP
,
"heap_update_undo: bad old tuple page LSN"
);
offnum
=
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
target
.
tid
));
lp
=
PageGetItemId
(
page
,
offnum
);
if
(
!
ItemIdIsUsed
(
lp
)
||
ItemIdDeleted
(
lp
))
{
if
(
redo
)
elog
(
STOP
,
"heap_update_redo: unused/deleted old tuple"
);
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_update_undo: unused/deleted old tuple in rollback"
);
if
(
ItemIdDeleted
(
lp
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
UnlockAndReleaseBuffer
(
buffer
);
goto
newt
;
}
htup
=
(
HeapTupleHeader
)
PageGetItem
(
page
,
lp
);
if
(
redo
)
{
htup
->
t_xmax
=
record
->
xl_xid
;
htup
->
t_cmax
=
xlrec
->
target
.
cid
;
htup
->
t_infomask
&=
~
(
HEAP_XMAX_INVALID
|
HEAP_MARKED_FOR_UPDATE
);
htup
->
t_infomask
|=
HEAP_XMAX_COMMITTED
;
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
goto
newt
;
}
/* undo... is it our tuple ? */
if
(
htup
->
t_xmax
!=
record
->
xl_xid
||
htup
->
t_cmax
!=
xlrec
->
target
.
cid
)
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_update_undo: invalid old tuple in rollback"
);
UnlockAndReleaseBuffer
(
buffer
);
}
else
/* undo */
{
htup
->
t_infomask
|=
HEAP_XMAX_INVALID
;
UnlockAndWriteBuffer
(
buffer
);
}
/* Deal with new tuple */
newt:
;
buffer
=
XLogReadBuffer
((
redo
)
?
true
:
false
,
reln
,
ItemPointerGetBlockNumber
(
&
(
xlrec
->
newtid
)));
if
(
!
BufferIsValid
(
buffer
))
return
;
page
=
(
Page
)
BufferGetPage
(
buffer
);
if
(
PageIsNew
((
PageHeader
)
page
))
{
PageInit
(
page
,
BufferGetPageSize
(
buffer
),
0
);
if
(
!
redo
)
{
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
UnlockAndWriteBuffer
(
buffer
);
return
;
}
}
if
(
redo
)
{
if
(
XLByteLE
(
lsn
,
PageGetLSN
(
page
)))
/* changes are applied */
{
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
char
tbuf
[
MaxTupleSize
];
uint32
newlen
=
record
->
xl_len
-
SizeOfHeapUpdate
;
htup
=
(
HeapTupleHeader
)
tbuf
;
memcpy
(
tbuf
+
offsetof
(
HeapTupleHeaderData
,
t_bits
),
(
char
*
)
xlrec
+
SizeOfHeapUpdate
,
newlen
);
newlen
+=
offsetof
(
HeapTupleHeaderData
,
t_bits
);
htup
->
t_oid
=
xlrec
->
t_oid
;
htup
->
t_natts
=
xlrec
->
t_natts
;
htup
->
t_hoff
=
xlrec
->
t_hoff
;
htup
->
t_xmin
=
record
->
xl_xid
;
htup
->
t_cmin
=
xlrec
->
target
.
cid
;
htup
->
t_infomask
=
HEAP_XMAX_INVALID
|
HEAP_XMIN_COMMITTED
|
xlrec
->
mask
;
PageManagerModeSet
(
OverwritePageManagerMode
);
OffsetNumber
offnum
=
PageAddItem
(
page
,
htup
,
newlen
,
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
newtid
)),
LP_USED
);
PageManagerModeSet
(
ShufflePageManagerMode
);
if
(
offnum
==
InvalidOffsetNumber
)
elog
(
STOP
,
"heap_update_redo: failed to add tuple"
);
PageSetLSN
(
page
,
lsn
);
PageSetSUI
(
page
,
ThisStartUpID
);
/* prev sui */
UnlockAndWriteBuffer
(
buffer
);
return
;
}
/* undo */
if
(
XLByteLT
(
PageGetLSN
(
page
),
lsn
))
/* changes are not applied ?! */
elog
(
STOP
,
"heap_update_undo: bad new tuple page LSN"
);
offnum
=
ItemPointerGetOffsetNumber
(
&
(
xlrec
->
newtid
));
lp
=
PageGetItemId
(
page
,
offnum
);
if
(
!
ItemIdIsUsed
(
lp
)
||
ItemIdDeleted
(
lp
))
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_update_undo: unused/deleted new tuple in rollback"
);
if
(
ItemIdDeleted
(
lp
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
htup
=
(
HeapTupleHeader
)
PageGetItem
(
page
,
lp
);
/* is it our tuple ? */
if
(
htup
->
t_xmin
!=
record
->
xl_xid
||
htup
->
t_cmin
!=
xlrec
->
target
.
cid
)
{
if
(
!
InRecovery
)
elog
(
STOP
,
"heap_update_undo: invalid new tuple in rollback"
);
UnlockAndReleaseBuffer
(
buffer
);
return
;
}
if
(
InRecovery
||
BufferIsUpdatable
(
buffer
))
{
lp
->
lp_flags
&=
~
LP_USED
;
PageRepairFragmentation
(
page
);
UnlockAndWriteBuffer
(
buffer
);
}
else
/* we can't delete tuple right now */
{
lp
->
lp_flags
|=
LP_DELETE
;
/* mark for deletion */
MarkBufferForCleanup
(
buffer
,
PageCleanup
);
}
}
}
#endif
/* XLOG */
#endif
/* XLOG */
src/backend/access/heap/hio.c
View file @
f2bfe8a2
...
@@ -8,7 +8,7 @@
...
@@ -8,7 +8,7 @@
*
*
*
*
* IDENTIFICATION
* IDENTIFICATION
* $Id: hio.c,v 1.3
2 2000/07/03 02:54:1
5 vadim Exp $
* $Id: hio.c,v 1.3
3 2000/09/07 09:58:3
5 vadim Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -67,16 +67,19 @@ RelationPutHeapTuple(Relation relation,
...
@@ -67,16 +67,19 @@ RelationPutHeapTuple(Relation relation,
/*
/*
* RelationGetBufferForTuple
* RelationGetBufferForTuple
*
*
* Returns (locked) buffer to add tuple with given len.
* Returns (locked) buffer with free space >= given len.
* If Ubuf is valid then no attempt to lock it should be made -
*
* this is for heap_update...
* Note that we use LockPage to lock relation for extension. We can
* do this as long as in all other places we use page-level locking
* for indices only. Alternatively, we could define pseudo-table as
* we do for transactions with XactLockTable.
*
*
* ELOG(ERROR) is allowed here, so this routine *must* be called
* ELOG(ERROR) is allowed here, so this routine *must* be called
* before any (unlogged) changes are made in buffer pool.
* before any (unlogged) changes are made in buffer pool.
*
*
*/
*/
Buffer
Buffer
RelationGetBufferForTuple
(
Relation
relation
,
Size
len
,
Buffer
Ubuf
)
RelationGetBufferForTuple
(
Relation
relation
,
Size
len
)
{
{
Buffer
buffer
;
Buffer
buffer
;
Page
pageHeader
;
Page
pageHeader
;
...
@@ -91,12 +94,6 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
...
@@ -91,12 +94,6 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
elog
(
ERROR
,
"Tuple is too big: size %u, max size %ld"
,
elog
(
ERROR
,
"Tuple is too big: size %u, max size %ld"
,
len
,
MaxTupleSize
);
len
,
MaxTupleSize
);
/*
* Lock relation for extension. We can use LockPage here as long as in
* all other places we use page-level locking for indices only.
* Alternatively, we could define pseudo-table as we do for
* transactions with XactLockTable.
*/
if
(
!
relation
->
rd_myxactonly
)
if
(
!
relation
->
rd_myxactonly
)
LockPage
(
relation
,
0
,
ExclusiveLock
);
LockPage
(
relation
,
0
,
ExclusiveLock
);
...
@@ -114,31 +111,29 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
...
@@ -114,31 +111,29 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
*/
*/
if
(
lastblock
==
0
)
if
(
lastblock
==
0
)
{
{
/* what exactly is this all about??? */
buffer
=
ReadBuffer
(
relation
,
P_NEW
);
buffer
=
ReadBuffer
(
relation
,
lastblock
);
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
Assert
(
PageIsNew
((
PageHeader
)
pageHeader
));
Assert
(
PageIsNew
((
PageHeader
)
pageHeader
));
buffer
=
ReleaseAndReadBuffer
(
buffer
,
relation
,
P_NEW
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
PageInit
(
pageHeader
,
BufferGetPageSize
(
buffer
),
0
);
PageInit
(
pageHeader
,
BufferGetPageSize
(
buffer
),
0
);
}
}
else
else
{
buffer
=
ReadBuffer
(
relation
,
lastblock
-
1
);
buffer
=
ReadBuffer
(
relation
,
lastblock
-
1
);
if
(
buffer
!=
Ubuf
)
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
}
/*
/*
* Is there room on the last existing page?
* Is there room on the last existing page?
*/
*/
if
(
len
>
PageGetFreeSpace
(
pageHeader
))
if
(
len
>
PageGetFreeSpace
(
pageHeader
))
{
{
if
(
buffer
!=
Ubuf
)
LockBuffer
(
buffer
,
BUFFER_LOCK_UNLOCK
);
LockBuffer
(
buffer
,
BUFFER_LOCK_UNLOCK
);
buffer
=
ReleaseAndReadBuffer
(
buffer
,
relation
,
P_NEW
);
buffer
=
ReleaseAndReadBuffer
(
buffer
,
relation
,
P_NEW
);
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
LockBuffer
(
buffer
,
BUFFER_LOCK_EXCLUSIVE
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
pageHeader
=
(
Page
)
BufferGetPage
(
buffer
);
Assert
(
PageIsNew
((
PageHeader
)
pageHeader
));
PageInit
(
pageHeader
,
BufferGetPageSize
(
buffer
),
0
);
PageInit
(
pageHeader
,
BufferGetPageSize
(
buffer
),
0
);
if
(
len
>
PageGetFreeSpace
(
pageHeader
))
if
(
len
>
PageGetFreeSpace
(
pageHeader
))
...
@@ -147,14 +142,6 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
...
@@ -147,14 +142,6 @@ RelationGetBufferForTuple(Relation relation, Size len, Buffer Ubuf)
elog
(
STOP
,
"Tuple is too big: size %u"
,
len
);
elog
(
STOP
,
"Tuple is too big: size %u"
,
len
);
}
}
}
}
/*
* Caller should check space in Ubuf but...
*/
else
if
(
buffer
==
Ubuf
)
{
ReleaseBuffer
(
buffer
);
buffer
=
Ubuf
;
}
if
(
!
relation
->
rd_myxactonly
)
if
(
!
relation
->
rd_myxactonly
)
UnlockPage
(
relation
,
0
,
ExclusiveLock
);
UnlockPage
(
relation
,
0
,
ExclusiveLock
);
...
...
src/include/access/hio.h
View file @
f2bfe8a2
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $Id: hio.h,v 1.1
5 2000/07/03 02:54:17
vadim Exp $
* $Id: hio.h,v 1.1
6 2000/09/07 09:58:35
vadim Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -18,6 +18,6 @@
...
@@ -18,6 +18,6 @@
extern
void
RelationPutHeapTuple
(
Relation
relation
,
Buffer
buffer
,
extern
void
RelationPutHeapTuple
(
Relation
relation
,
Buffer
buffer
,
HeapTuple
tuple
);
HeapTuple
tuple
);
extern
Buffer
RelationGetBufferForTuple
(
Relation
relation
,
Size
len
,
Buffer
Ubuf
);
extern
Buffer
RelationGetBufferForTuple
(
Relation
relation
,
Size
len
);
#endif
/* HIO_H */
#endif
/* HIO_H */
src/include/access/htup.h
View file @
f2bfe8a2
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $Id: htup.h,v 1.3
4 2000/08/07 20:15:40 tgl
Exp $
* $Id: htup.h,v 1.3
5 2000/09/07 09:58:35 vadim
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -15,6 +15,7 @@
...
@@ -15,6 +15,7 @@
#define HTUP_H
#define HTUP_H
#include "storage/bufpage.h"
#include "storage/bufpage.h"
#include "storage/relfilenode.h"
#define MinHeapTupleBitmapSize 32
/* 8 * 4 */
#define MinHeapTupleBitmapSize 32
/* 8 * 4 */
...
@@ -81,8 +82,7 @@ typedef HeapTupleHeaderData *HeapTupleHeader;
...
@@ -81,8 +82,7 @@ typedef HeapTupleHeaderData *HeapTupleHeader;
*/
*/
typedef
struct
xl_heaptid
typedef
struct
xl_heaptid
{
{
Oid
dbId
;
/* database */
RelFileNode
node
;
Oid
relId
;
/* relation */
CommandId
cid
;
/* this is for "better" tuple' */
CommandId
cid
;
/* this is for "better" tuple' */
/* identification - it allows to avoid */
/* identification - it allows to avoid */
/* "compensation" records for undo */
/* "compensation" records for undo */
...
@@ -92,7 +92,7 @@ typedef struct xl_heaptid
...
@@ -92,7 +92,7 @@ typedef struct xl_heaptid
/* This is what we need to know about delete - ALIGN(18) = 24 bytes */
/* This is what we need to know about delete - ALIGN(18) = 24 bytes */
typedef
struct
xl_heap_delete
typedef
struct
xl_heap_delete
{
{
xl_heaptid
dtid
;
/* deleted tuple id */
xl_heaptid
target
;
/* deleted tuple id */
}
xl_heap_delete
;
}
xl_heap_delete
;
#define SizeOfHeapDelete (offsetof(xl_heaptid, tid) + SizeOfIptrData))
#define SizeOfHeapDelete (offsetof(xl_heaptid, tid) + SizeOfIptrData))
...
@@ -100,7 +100,7 @@ typedef struct xl_heap_delete
...
@@ -100,7 +100,7 @@ typedef struct xl_heap_delete
/* This is what we need to know about insert - 26 + data */
/* This is what we need to know about insert - 26 + data */
typedef
struct
xl_heap_insert
typedef
struct
xl_heap_insert
{
{
xl_heaptid
itid
;
/* inserted tuple id */
xl_heaptid
target
;
/* inserted tuple id */
/* something from tuple header */
/* something from tuple header */
int16
t_natts
;
int16
t_natts
;
Oid
t_oid
;
Oid
t_oid
;
...
@@ -114,8 +114,8 @@ typedef struct xl_heap_insert
...
@@ -114,8 +114,8 @@ typedef struct xl_heap_insert
/* This is what we need to know about update - 28 + data */
/* This is what we need to know about update - 28 + data */
typedef
struct
xl_heap_update
typedef
struct
xl_heap_update
{
{
xl_heaptid
dtid
;
/* deleted tuple id */
xl_heaptid
target
;
/* deleted tuple id */
ItemPointerData
i
tid
;
/* new inserted tuple id */
ItemPointerData
new
tid
;
/* new inserted tuple id */
/* something from header of new tuple version */
/* something from header of new tuple version */
int16
t_natts
;
int16
t_natts
;
uint8
t_hoff
;
uint8
t_hoff
;
...
@@ -128,8 +128,8 @@ typedef struct xl_heap_update
...
@@ -128,8 +128,8 @@ typedef struct xl_heap_update
/* This is what we need to know about tuple move - 24 bytes */
/* This is what we need to know about tuple move - 24 bytes */
typedef
struct
xl_heap_move
typedef
struct
xl_heap_move
{
{
xl_heaptid
ftid
;
/* moved from */
xl_heaptid
target
;
/* moved from */
ItemPointerData
t
tid
;
/* moved to */
ItemPointerData
new
tid
;
/* moved to */
}
xl_heap_move
;
}
xl_heap_move
;
#define SizeOfHeapMove (offsetof(xl_heap_move, ttid) + SizeOfIptrData))
#define SizeOfHeapMove (offsetof(xl_heap_move, ttid) + SizeOfIptrData))
...
@@ -238,6 +238,9 @@ typedef HeapTupleData *HeapTuple;
...
@@ -238,6 +238,9 @@ typedef HeapTupleData *HeapTuple;
#define HEAP_HASCOMPRESSED 0x0008
/* has compressed stored */
#define HEAP_HASCOMPRESSED 0x0008
/* has compressed stored */
/* attribute(s) */
/* attribute(s) */
#define HEAP_HASEXTENDED 0x000C
/* the two above combined */
#define HEAP_HASEXTENDED 0x000C
/* the two above combined */
#define HEAP_XMAX_UNLOGGED 0x0080
/* to lock tuple for update */
/* without logging */
#define HEAP_XMIN_COMMITTED 0x0100
/* t_xmin committed */
#define HEAP_XMIN_COMMITTED 0x0100
/* t_xmin committed */
#define HEAP_XMIN_INVALID 0x0200
/* t_xmin invalid/aborted */
#define HEAP_XMIN_INVALID 0x0200
/* t_xmin invalid/aborted */
#define HEAP_XMAX_COMMITTED 0x0400
/* t_xmax committed */
#define HEAP_XMAX_COMMITTED 0x0400
/* t_xmax committed */
...
@@ -249,7 +252,7 @@ typedef HeapTupleData *HeapTuple;
...
@@ -249,7 +252,7 @@ typedef HeapTupleData *HeapTuple;
#define HEAP_MOVED_IN 0x8000
/* moved from another place by
#define HEAP_MOVED_IN 0x8000
/* moved from another place by
* vacuum */
* vacuum */
#define HEAP_XACT_MASK 0xFF
0
0
/* */
#define HEAP_XACT_MASK 0xFF
F
0
/* */
#define HeapTupleNoNulls(tuple) \
#define HeapTupleNoNulls(tuple) \
(!(((HeapTuple) (tuple))->t_data->t_infomask & HEAP_HASNULL))
(!(((HeapTuple) (tuple))->t_data->t_infomask & HEAP_HASNULL))
...
...
src/include/storage/bufpage.h
View file @
f2bfe8a2
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $Id: bufpage.h,v 1.3
1 2000/07/21 06:42:39 tgl
Exp $
* $Id: bufpage.h,v 1.3
2 2000/09/07 09:58:36 vadim
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -296,6 +296,19 @@ typedef enum
...
@@ -296,6 +296,19 @@ typedef enum
(sizeof(PageHeaderData) - sizeof(ItemIdData)))) \
(sizeof(PageHeaderData) - sizeof(ItemIdData)))) \
/ ((int) sizeof(ItemIdData)))
/ ((int) sizeof(ItemIdData)))
#ifdef XLOG
#define PageGetLSN(page) \
(((PageHeader) (page))->pd_lsn)
#define PageSetLSN(page, lsn) \
(((PageHeader) (page))->pd_lsn = (XLogRecPtr) (lsn))
#define PageGetSUI(page) \
(((PageHeader) (page))->pd_sui)
#define PageSetSUI(page, sui) \
(((PageHeader) (page))->pd_sui = (StartUpID) (sui))
#endif
/* ----------------------------------------------------------------
/* ----------------------------------------------------------------
* extern declarations
* extern declarations
...
...
src/include/storage/itemid.h
View file @
f2bfe8a2
...
@@ -7,14 +7,13 @@
...
@@ -7,14 +7,13 @@
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $Id: itemid.h,v 1.1
1 2000/08/07 20:15:50 tgl
Exp $
* $Id: itemid.h,v 1.1
2 2000/09/07 09:58:36 vadim
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
#ifndef ITEMID_H
#ifndef ITEMID_H
#define ITEMID_H
#define ITEMID_H
/*
/*
* An item pointer (also called line pointer) on a buffer page
* An item pointer (also called line pointer) on a buffer page
*/
*/
...
@@ -31,8 +30,15 @@ typedef ItemIdData *ItemId;
...
@@ -31,8 +30,15 @@ typedef ItemIdData *ItemId;
* lp_flags contains these flags:
* lp_flags contains these flags:
*/
*/
#define LP_USED 0x01
/* this line pointer is being used */
#define LP_USED 0x01
/* this line pointer is being used */
/* currently, there is one unused flag bit ... */
#ifdef XLOG
#define LP_DELETE 0x02
/* item is to be deleted */
#define ItemIdDeleted(itemId) \
(((itemId)->lp_flags & LP_DELETE) != 0)
#endif
/*
/*
* Item offsets, lengths, and flags are represented by these types when
* Item offsets, lengths, and flags are represented by these types when
...
...
src/include/utils/rel.h
View file @
f2bfe8a2
...
@@ -7,7 +7,7 @@
...
@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
* Portions Copyright (c) 1994, Regents of the University of California
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* $Id: rel.h,v 1.4
0 2000/07/14 22:18:02 tgl
Exp $
* $Id: rel.h,v 1.4
1 2000/09/07 09:58:38 vadim
Exp $
*
*
*-------------------------------------------------------------------------
*-------------------------------------------------------------------------
*/
*/
...
@@ -19,6 +19,7 @@
...
@@ -19,6 +19,7 @@
#include "catalog/pg_am.h"
#include "catalog/pg_am.h"
#include "catalog/pg_class.h"
#include "catalog/pg_class.h"
#include "rewrite/prs2lock.h"
#include "rewrite/prs2lock.h"
#include "storage/relfilenode.h"
#include "storage/fd.h"
#include "storage/fd.h"
/* added to prevent circular dependency. bjm 1999/11/15 */
/* added to prevent circular dependency. bjm 1999/11/15 */
...
@@ -86,6 +87,7 @@ typedef struct TriggerDesc
...
@@ -86,6 +87,7 @@ typedef struct TriggerDesc
typedef
struct
RelationData
typedef
struct
RelationData
{
{
File
rd_fd
;
/* open file descriptor, or -1 if none */
File
rd_fd
;
/* open file descriptor, or -1 if none */
RelFileNode
rd_node
;
/* relation file node */
int
rd_nblocks
;
/* number of blocks in rel */
int
rd_nblocks
;
/* number of blocks in rel */
uint16
rd_refcnt
;
/* reference count */
uint16
rd_refcnt
;
/* reference count */
bool
rd_myxactonly
;
/* rel uses the local buffer mgr */
bool
rd_myxactonly
;
/* rel uses the local buffer mgr */
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment