Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * xact.c
4 : : * top level transaction system support routines
5 : : *
6 : : * See src/backend/access/transam/README for more information.
7 : : *
8 : : * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
9 : : * Portions Copyright (c) 1994, Regents of the University of California
10 : : *
11 : : *
12 : : * IDENTIFICATION
13 : : * src/backend/access/transam/xact.c
14 : : *
15 : : *-------------------------------------------------------------------------
16 : : */
17 : :
18 : : #include "postgres.h"
19 : :
20 : : #include <time.h>
21 : : #include <unistd.h>
22 : :
23 : : #include "access/commit_ts.h"
24 : : #include "access/multixact.h"
25 : : #include "access/parallel.h"
26 : : #include "access/subtrans.h"
27 : : #include "access/transam.h"
28 : : #include "access/twophase.h"
29 : : #include "access/xact.h"
30 : : #include "access/xlog.h"
31 : : #include "access/xloginsert.h"
32 : : #include "access/xlogrecovery.h"
33 : : #include "access/xlogutils.h"
34 : : #include "access/xlogwait.h"
35 : : #include "catalog/index.h"
36 : : #include "catalog/namespace.h"
37 : : #include "catalog/pg_enum.h"
38 : : #include "catalog/storage.h"
39 : : #include "commands/async.h"
40 : : #include "commands/tablecmds.h"
41 : : #include "commands/trigger.h"
42 : : #include "common/pg_prng.h"
43 : : #include "executor/spi.h"
44 : : #include "libpq/be-fsstubs.h"
45 : : #include "libpq/pqsignal.h"
46 : : #include "miscadmin.h"
47 : : #include "pg_trace.h"
48 : : #include "pgstat.h"
49 : : #include "replication/logical.h"
50 : : #include "replication/logicallauncher.h"
51 : : #include "replication/logicalworker.h"
52 : : #include "replication/origin.h"
53 : : #include "replication/snapbuild.h"
54 : : #include "replication/syncrep.h"
55 : : #include "storage/aio_subsys.h"
56 : : #include "storage/condition_variable.h"
57 : : #include "storage/fd.h"
58 : : #include "storage/lmgr.h"
59 : : #include "storage/md.h"
60 : : #include "storage/predicate.h"
61 : : #include "storage/proc.h"
62 : : #include "storage/procarray.h"
63 : : #include "storage/sinvaladt.h"
64 : : #include "storage/smgr.h"
65 : : #include "utils/builtins.h"
66 : : #include "utils/combocid.h"
67 : : #include "utils/guc.h"
68 : : #include "utils/inval.h"
69 : : #include "utils/memutils.h"
70 : : #include "utils/relmapper.h"
71 : : #include "utils/snapmgr.h"
72 : : #include "utils/timeout.h"
73 : : #include "utils/timestamp.h"
74 : : #include "utils/typcache.h"
75 : : #include "utils/wait_event.h"
76 : :
77 : : /*
78 : : * User-tweakable parameters
79 : : */
80 : : int DefaultXactIsoLevel = XACT_READ_COMMITTED;
81 : : int XactIsoLevel = XACT_READ_COMMITTED;
82 : :
83 : : bool DefaultXactReadOnly = false;
84 : : bool XactReadOnly;
85 : :
86 : : bool DefaultXactDeferrable = false;
87 : : bool XactDeferrable;
88 : :
89 : : int synchronous_commit = SYNCHRONOUS_COMMIT_ON;
90 : :
91 : : /*
92 : : * CheckXidAlive is a xid value pointing to a possibly ongoing (sub)
93 : : * transaction. Currently, it is used in logical decoding. It's possible
94 : : * that such transactions can get aborted while the decoding is ongoing in
95 : : * which case we skip decoding that particular transaction. To ensure that we
96 : : * check whether the CheckXidAlive is aborted after fetching the tuple from
97 : : * system tables. We also ensure that during logical decoding we never
98 : : * directly access the tableam or heap APIs because we are checking for the
99 : : * concurrent aborts only in systable_* APIs.
100 : : */
101 : : TransactionId CheckXidAlive = InvalidTransactionId;
102 : : bool bsysscan = false;
103 : :
104 : : /*
105 : : * When running as a parallel worker, we place only a single
106 : : * TransactionStateData on the parallel worker's state stack, and the XID
107 : : * reflected there will be that of the *innermost* currently-active
108 : : * subtransaction in the backend that initiated parallelism. However,
109 : : * GetTopTransactionId() and TransactionIdIsCurrentTransactionId()
110 : : * need to return the same answers in the parallel worker as they would have
111 : : * in the user backend, so we need some additional bookkeeping.
112 : : *
113 : : * XactTopFullTransactionId stores the XID of our toplevel transaction, which
114 : : * will be the same as TopTransactionStateData.fullTransactionId in an
115 : : * ordinary backend; but in a parallel backend, which does not have the entire
116 : : * transaction state, it will instead be copied from the backend that started
117 : : * the parallel operation.
118 : : *
119 : : * nParallelCurrentXids will be 0 and ParallelCurrentXids NULL in an ordinary
120 : : * backend, but in a parallel backend, nParallelCurrentXids will contain the
121 : : * number of XIDs that need to be considered current, and ParallelCurrentXids
122 : : * will contain the XIDs themselves. This includes all XIDs that were current
123 : : * or sub-committed in the parent at the time the parallel operation began.
124 : : * The XIDs are stored sorted in numerical order (not logical order) to make
125 : : * lookups as fast as possible.
126 : : */
127 : : static FullTransactionId XactTopFullTransactionId = {InvalidTransactionId};
128 : : static int nParallelCurrentXids = 0;
129 : : static TransactionId *ParallelCurrentXids;
130 : :
131 : : /*
132 : : * Miscellaneous flag bits to record events which occur on the top level
133 : : * transaction. These flags are only persisted in MyXactFlags and are intended
134 : : * so we remember to do certain things later on in the transaction. This is
135 : : * globally accessible, so can be set from anywhere in the code that requires
136 : : * recording flags.
137 : : */
138 : : int MyXactFlags;
139 : :
140 : : /*
141 : : * transaction states - transaction state from server perspective
142 : : */
143 : : typedef enum TransState
144 : : {
145 : : TRANS_DEFAULT, /* idle */
146 : : TRANS_START, /* transaction starting */
147 : : TRANS_INPROGRESS, /* inside a valid transaction */
148 : : TRANS_COMMIT, /* commit in progress */
149 : : TRANS_ABORT, /* abort in progress */
150 : : TRANS_PREPARE, /* prepare in progress */
151 : : } TransState;
152 : :
153 : : /*
154 : : * transaction block states - transaction state of client queries
155 : : *
156 : : * Note: the subtransaction states are used only for non-topmost
157 : : * transactions; the others appear only in the topmost transaction.
158 : : */
159 : : typedef enum TBlockState
160 : : {
161 : : /* not-in-transaction-block states */
162 : : TBLOCK_DEFAULT, /* idle */
163 : : TBLOCK_STARTED, /* running single-query transaction */
164 : :
165 : : /* transaction block states */
166 : : TBLOCK_BEGIN, /* starting transaction block */
167 : : TBLOCK_INPROGRESS, /* live transaction */
168 : : TBLOCK_IMPLICIT_INPROGRESS, /* live transaction after implicit BEGIN */
169 : : TBLOCK_PARALLEL_INPROGRESS, /* live transaction inside parallel worker */
170 : : TBLOCK_END, /* COMMIT received */
171 : : TBLOCK_ABORT, /* failed xact, awaiting ROLLBACK */
172 : : TBLOCK_ABORT_END, /* failed xact, ROLLBACK received */
173 : : TBLOCK_ABORT_PENDING, /* live xact, ROLLBACK received */
174 : : TBLOCK_PREPARE, /* live xact, PREPARE received */
175 : :
176 : : /* subtransaction states */
177 : : TBLOCK_SUBBEGIN, /* starting a subtransaction */
178 : : TBLOCK_SUBINPROGRESS, /* live subtransaction */
179 : : TBLOCK_SUBRELEASE, /* RELEASE received */
180 : : TBLOCK_SUBCOMMIT, /* COMMIT received while TBLOCK_SUBINPROGRESS */
181 : : TBLOCK_SUBABORT, /* failed subxact, awaiting ROLLBACK */
182 : : TBLOCK_SUBABORT_END, /* failed subxact, ROLLBACK received */
183 : : TBLOCK_SUBABORT_PENDING, /* live subxact, ROLLBACK received */
184 : : TBLOCK_SUBRESTART, /* live subxact, ROLLBACK TO received */
185 : : TBLOCK_SUBABORT_RESTART, /* failed subxact, ROLLBACK TO received */
186 : : } TBlockState;
187 : :
188 : : /*
189 : : * transaction state structure
190 : : *
191 : : * Note: parallelModeLevel counts the number of unmatched EnterParallelMode
192 : : * calls done at this transaction level. parallelChildXact is true if any
193 : : * upper transaction level has nonzero parallelModeLevel.
194 : : */
195 : : typedef struct TransactionStateData
196 : : {
197 : : FullTransactionId fullTransactionId; /* my FullTransactionId */
198 : : SubTransactionId subTransactionId; /* my subxact ID */
199 : : char *name; /* savepoint name, if any */
200 : : int savepointLevel; /* savepoint level */
201 : : TransState state; /* low-level state */
202 : : TBlockState blockState; /* high-level state */
203 : : int nestingLevel; /* transaction nesting depth */
204 : : int gucNestLevel; /* GUC context nesting depth */
205 : : MemoryContext curTransactionContext; /* my xact-lifetime context */
206 : : ResourceOwner curTransactionOwner; /* my query resources */
207 : : MemoryContext priorContext; /* CurrentMemoryContext before xact started */
208 : : TransactionId *childXids; /* subcommitted child XIDs, in XID order */
209 : : int nChildXids; /* # of subcommitted child XIDs */
210 : : int maxChildXids; /* allocated size of childXids[] */
211 : : Oid prevUser; /* previous CurrentUserId setting */
212 : : int prevSecContext; /* previous SecurityRestrictionContext */
213 : : bool prevXactReadOnly; /* entry-time xact r/o state */
214 : : bool startedInRecovery; /* did we start in recovery? */
215 : : bool didLogXid; /* has xid been included in WAL record? */
216 : : int parallelModeLevel; /* Enter/ExitParallelMode counter */
217 : : bool parallelChildXact; /* is any parent transaction parallel? */
218 : : bool chain; /* start a new block after this one */
219 : : bool topXidLogged; /* for a subxact: is top-level XID logged? */
220 : : struct TransactionStateData *parent; /* back link to parent */
221 : : } TransactionStateData;
222 : :
223 : : typedef TransactionStateData *TransactionState;
224 : :
225 : : /*
226 : : * Serialized representation used to transmit transaction state to parallel
227 : : * workers through shared memory.
228 : : */
229 : : typedef struct SerializedTransactionState
230 : : {
231 : : int xactIsoLevel;
232 : : bool xactDeferrable;
233 : : FullTransactionId topFullTransactionId;
234 : : FullTransactionId currentFullTransactionId;
235 : : CommandId currentCommandId;
236 : : int nParallelCurrentXids;
237 : : TransactionId parallelCurrentXids[FLEXIBLE_ARRAY_MEMBER];
238 : : } SerializedTransactionState;
239 : :
240 : : /* The size of SerializedTransactionState, not including the final array. */
241 : : #define SerializedTransactionStateHeaderSize \
242 : : offsetof(SerializedTransactionState, parallelCurrentXids)
243 : :
244 : : /*
245 : : * CurrentTransactionState always points to the current transaction state
246 : : * block. It will point to TopTransactionStateData when not in a
247 : : * transaction at all, or when in a top-level transaction.
248 : : */
249 : : static TransactionStateData TopTransactionStateData = {
250 : : .state = TRANS_DEFAULT,
251 : : .blockState = TBLOCK_DEFAULT,
252 : : .topXidLogged = false,
253 : : };
254 : :
255 : : /*
256 : : * unreportedXids holds XIDs of all subtransactions that have not yet been
257 : : * reported in an XLOG_XACT_ASSIGNMENT record.
258 : : */
259 : : static int nUnreportedXids;
260 : : static TransactionId unreportedXids[PGPROC_MAX_CACHED_SUBXIDS];
261 : :
262 : : static TransactionState CurrentTransactionState = &TopTransactionStateData;
263 : :
264 : : /*
265 : : * The subtransaction ID and command ID assignment counters are global
266 : : * to a whole transaction, so we do not keep them in the state stack.
267 : : */
268 : : static SubTransactionId currentSubTransactionId;
269 : : static CommandId currentCommandId;
270 : : static bool currentCommandIdUsed;
271 : :
272 : : /*
273 : : * xactStartTimestamp is the value of transaction_timestamp().
274 : : * stmtStartTimestamp is the value of statement_timestamp().
275 : : * xactStopTimestamp is the time at which we log a commit / abort WAL record,
276 : : * or if that was skipped, the time of the first subsequent
277 : : * GetCurrentTransactionStopTimestamp() call.
278 : : *
279 : : * These do not change as we enter and exit subtransactions, so we don't
280 : : * keep them inside the TransactionState stack.
281 : : */
282 : : static TimestampTz xactStartTimestamp;
283 : : static TimestampTz stmtStartTimestamp;
284 : : static TimestampTz xactStopTimestamp;
285 : :
286 : : /*
287 : : * GID to be used for preparing the current transaction. This is also
288 : : * global to a whole transaction, so we don't keep it in the state stack.
289 : : */
290 : : static char *prepareGID;
291 : :
292 : : /*
293 : : * Some commands want to force synchronous commit.
294 : : */
295 : : static bool forceSyncCommit = false;
296 : :
297 : : /* Flag for logging statements in a transaction. */
298 : : bool xact_is_sampled = false;
299 : :
300 : : /*
301 : : * Private context for transaction-abort work --- we reserve space for this
302 : : * at startup to ensure that AbortTransaction and AbortSubTransaction can work
303 : : * when we've run out of memory.
304 : : */
305 : : static MemoryContext TransactionAbortContext = NULL;
306 : :
307 : : /*
308 : : * List of add-on start- and end-of-xact callbacks
309 : : */
310 : : typedef struct XactCallbackItem
311 : : {
312 : : struct XactCallbackItem *next;
313 : : XactCallback callback;
314 : : void *arg;
315 : : } XactCallbackItem;
316 : :
317 : : static XactCallbackItem *Xact_callbacks = NULL;
318 : :
319 : : /*
320 : : * List of add-on start- and end-of-subxact callbacks
321 : : */
322 : : typedef struct SubXactCallbackItem
323 : : {
324 : : struct SubXactCallbackItem *next;
325 : : SubXactCallback callback;
326 : : void *arg;
327 : : } SubXactCallbackItem;
328 : :
329 : : static SubXactCallbackItem *SubXact_callbacks = NULL;
330 : :
331 : :
332 : : /* local function prototypes */
333 : : static void AssignTransactionId(TransactionState s);
334 : : static void AbortTransaction(void);
335 : : static void AtAbort_Memory(void);
336 : : static void AtCleanup_Memory(void);
337 : : static void AtAbort_ResourceOwner(void);
338 : : static void AtCCI_LocalCache(void);
339 : : static void AtCommit_Memory(void);
340 : : static void AtStart_Cache(void);
341 : : static void AtStart_Memory(void);
342 : : static void AtStart_ResourceOwner(void);
343 : : static void CallXactCallbacks(XactEvent event);
344 : : static void CallSubXactCallbacks(SubXactEvent event,
345 : : SubTransactionId mySubid,
346 : : SubTransactionId parentSubid);
347 : : static void CleanupTransaction(void);
348 : : static void CheckTransactionBlock(bool isTopLevel, bool throwError,
349 : : const char *stmtType);
350 : : static void CommitTransaction(void);
351 : : static TransactionId RecordTransactionAbort(bool isSubXact);
352 : : static void StartTransaction(void);
353 : :
354 : : static bool CommitTransactionCommandInternal(void);
355 : : static bool AbortCurrentTransactionInternal(void);
356 : :
357 : : static void StartSubTransaction(void);
358 : : static void CommitSubTransaction(void);
359 : : static void AbortSubTransaction(void);
360 : : static void CleanupSubTransaction(void);
361 : : static void PushTransaction(void);
362 : : static void PopTransaction(void);
363 : :
364 : : static void AtSubAbort_Memory(void);
365 : : static void AtSubCleanup_Memory(void);
366 : : static void AtSubAbort_ResourceOwner(void);
367 : : static void AtSubCommit_Memory(void);
368 : : static void AtSubStart_Memory(void);
369 : : static void AtSubStart_ResourceOwner(void);
370 : :
371 : : static void ShowTransactionState(const char *str);
372 : : static void ShowTransactionStateRec(const char *str, TransactionState s);
373 : : static const char *BlockStateAsString(TBlockState blockState);
374 : : static const char *TransStateAsString(TransState state);
375 : :
376 : :
377 : : /* ----------------------------------------------------------------
378 : : * transaction state accessors
379 : : * ----------------------------------------------------------------
380 : : */
381 : :
382 : : /*
383 : : * IsTransactionState
384 : : *
385 : : * This returns true if we are inside a valid transaction; that is,
386 : : * it is safe to initiate database access, take heavyweight locks, etc.
387 : : */
388 : : bool
10824 bruce@momjian.us 389 :CBC 195379718 : IsTransactionState(void)
390 : : {
10523 391 : 195379718 : TransactionState s = CurrentTransactionState;
392 : :
393 : : /*
394 : : * TRANS_DEFAULT and TRANS_ABORT are obviously unsafe states. However, we
395 : : * also reject the startup/shutdown states TRANS_START, TRANS_COMMIT,
396 : : * TRANS_PREPARE since it might be too soon or too late within those
397 : : * transition states to do anything interesting. Hence, the only "valid"
398 : : * state is TRANS_INPROGRESS.
399 : : */
6963 tgl@sss.pgh.pa.us 400 : 195379718 : return (s->state == TRANS_INPROGRESS);
401 : : }
402 : :
403 : : /*
404 : : * IsAbortedTransactionBlockState
405 : : *
406 : : * This returns true if we are within an aborted transaction block.
407 : : */
408 : : bool
9380 409 : 946914 : IsAbortedTransactionBlockState(void)
410 : : {
10523 bruce@momjian.us 411 : 946914 : TransactionState s = CurrentTransactionState;
412 : :
7975 413 [ + + ]: 946914 : if (s->blockState == TBLOCK_ABORT ||
8034 tgl@sss.pgh.pa.us 414 [ + + ]: 944932 : s->blockState == TBLOCK_SUBABORT)
10523 bruce@momjian.us 415 : 2396 : return true;
416 : :
417 : 944518 : return false;
418 : : }
419 : :
420 : :
421 : : /*
422 : : * GetTopTransactionId
423 : : *
424 : : * This will return the XID of the main transaction, assigning one if
425 : : * it's not yet set. Be careful to call this only inside a valid xact.
426 : : */
427 : : TransactionId
8034 tgl@sss.pgh.pa.us 428 : 29798 : GetTopTransactionId(void)
429 : : {
2651 tmunro@postgresql.or 430 [ + + ]: 29798 : if (!FullTransactionIdIsValid(XactTopFullTransactionId))
6873 tgl@sss.pgh.pa.us 431 : 580 : AssignTransactionId(&TopTransactionStateData);
2651 tmunro@postgresql.or 432 : 29798 : return XidFromFullTransactionId(XactTopFullTransactionId);
433 : : }
434 : :
435 : : /*
436 : : * GetTopTransactionIdIfAny
437 : : *
438 : : * This will return the XID of the main transaction, if one is assigned.
439 : : * It will return InvalidTransactionId if we are not currently inside a
440 : : * transaction, or inside a transaction that hasn't yet been assigned an XID.
441 : : */
442 : : TransactionId
6873 tgl@sss.pgh.pa.us 443 : 90295860 : GetTopTransactionIdIfAny(void)
444 : : {
2651 tmunro@postgresql.or 445 : 90295860 : return XidFromFullTransactionId(XactTopFullTransactionId);
446 : : }
447 : :
448 : : /*
449 : : * GetCurrentTransactionId
450 : : *
451 : : * This will return the XID of the current transaction (main or sub
452 : : * transaction), assigning one if it's not yet set. Be careful to call this
453 : : * only inside a valid xact.
454 : : */
455 : : TransactionId
9380 tgl@sss.pgh.pa.us 456 : 16978859 : GetCurrentTransactionId(void)
457 : : {
10523 bruce@momjian.us 458 : 16978859 : TransactionState s = CurrentTransactionState;
459 : :
2651 tmunro@postgresql.or 460 [ + + ]: 16978859 : if (!FullTransactionIdIsValid(s->fullTransactionId))
6873 tgl@sss.pgh.pa.us 461 : 168560 : AssignTransactionId(s);
2651 tmunro@postgresql.or 462 : 16978859 : return XidFromFullTransactionId(s->fullTransactionId);
463 : : }
464 : :
465 : : /*
466 : : * GetCurrentTransactionIdIfAny
467 : : *
468 : : * This will return the XID of the current sub xact, if one is assigned.
469 : : * It will return InvalidTransactionId if we are not currently inside a
470 : : * transaction, or inside a transaction that hasn't been assigned an XID yet.
471 : : */
472 : : TransactionId
6873 tgl@sss.pgh.pa.us 473 : 24607248 : GetCurrentTransactionIdIfAny(void)
474 : : {
2651 tmunro@postgresql.or 475 : 24607248 : return XidFromFullTransactionId(CurrentTransactionState->fullTransactionId);
476 : : }
477 : :
478 : : /*
479 : : * GetTopFullTransactionId
480 : : *
481 : : * This will return the FullTransactionId of the main transaction, assigning
482 : : * one if it's not yet set. Be careful to call this only inside a valid xact.
483 : : */
484 : : FullTransactionId
485 : 3462 : GetTopFullTransactionId(void)
486 : : {
487 [ + + ]: 3462 : if (!FullTransactionIdIsValid(XactTopFullTransactionId))
488 : 2104 : AssignTransactionId(&TopTransactionStateData);
489 : 3462 : return XactTopFullTransactionId;
490 : : }
491 : :
492 : : /*
493 : : * GetTopFullTransactionIdIfAny
494 : : *
495 : : * This will return the FullTransactionId of the main transaction, if one is
496 : : * assigned. It will return InvalidFullTransactionId if we are not currently
497 : : * inside a transaction, or inside a transaction that hasn't yet been assigned
498 : : * one.
499 : : */
500 : : FullTransactionId
501 : 16 : GetTopFullTransactionIdIfAny(void)
502 : : {
503 : 16 : return XactTopFullTransactionId;
504 : : }
505 : :
506 : : /*
507 : : * GetCurrentFullTransactionId
508 : : *
509 : : * This will return the FullTransactionId of the current transaction (main or
510 : : * sub transaction), assigning one if it's not yet set. Be careful to call
511 : : * this only inside a valid xact.
512 : : */
513 : : FullTransactionId
2651 tmunro@postgresql.or 514 :GBC 371 : GetCurrentFullTransactionId(void)
515 : : {
516 : 371 : TransactionState s = CurrentTransactionState;
517 : :
518 [ + + ]: 371 : if (!FullTransactionIdIsValid(s->fullTransactionId))
519 : 14 : AssignTransactionId(s);
520 : 371 : return s->fullTransactionId;
521 : : }
522 : :
523 : : /*
524 : : * GetCurrentFullTransactionIdIfAny
525 : : *
526 : : * This will return the FullTransactionId of the current sub xact, if one is
527 : : * assigned. It will return InvalidFullTransactionId if we are not currently
528 : : * inside a transaction, or inside a transaction that hasn't been assigned one
529 : : * yet.
530 : : */
531 : : FullTransactionId
2651 tmunro@postgresql.or 532 :UBC 0 : GetCurrentFullTransactionIdIfAny(void)
533 : : {
534 : 0 : return CurrentTransactionState->fullTransactionId;
535 : : }
536 : :
537 : : /*
538 : : * MarkCurrentTransactionIdLoggedIfAny
539 : : *
540 : : * Remember that the current xid - if it is assigned - now has been wal logged.
541 : : */
542 : : void
4585 rhaas@postgresql.org 543 :CBC 24555411 : MarkCurrentTransactionIdLoggedIfAny(void)
544 : : {
2651 tmunro@postgresql.or 545 [ + + ]: 24555411 : if (FullTransactionIdIsValid(CurrentTransactionState->fullTransactionId))
4585 rhaas@postgresql.org 546 : 24191692 : CurrentTransactionState->didLogXid = true;
547 : 24555411 : }
548 : :
549 : : /*
550 : : * IsSubxactTopXidLogPending
551 : : *
552 : : * This is used to decide whether we need to WAL log the top-level XID for
553 : : * operation in a subtransaction. We require that for logical decoding, see
554 : : * LogicalDecodingProcessRecord.
555 : : *
556 : : * This returns true if effective_wal_level is logical and we are inside
557 : : * a valid subtransaction, for which the assignment was not yet written to
558 : : * any WAL record.
559 : : */
560 : : bool
1701 akapila@postgresql.o 561 : 24563988 : IsSubxactTopXidLogPending(void)
562 : : {
563 : : /* check whether it is already logged */
564 [ + + ]: 24563988 : if (CurrentTransactionState->topXidLogged)
565 : 102135 : return false;
566 : :
567 : : /* effective_wal_level has to be logical */
568 [ + + + + ]: 24461853 : if (!XLogLogicalInfoActive())
569 : 23872902 : return false;
570 : :
571 : : /* we need to be in a transaction state */
572 [ + + ]: 588951 : if (!IsTransactionState())
573 : 4251 : return false;
574 : :
575 : : /* it has to be a subtransaction */
576 [ + + ]: 584700 : if (!IsSubTransaction())
577 : 584244 : return false;
578 : :
579 : : /* the subtransaction has to have a XID assigned */
580 [ + + ]: 456 : if (!TransactionIdIsValid(GetCurrentTransactionIdIfAny()))
581 : 8 : return false;
582 : :
583 : 448 : return true;
584 : : }
585 : :
586 : : /*
587 : : * MarkSubxactTopXidLogged
588 : : *
589 : : * Remember that the top transaction id for the current subtransaction is WAL
590 : : * logged now.
591 : : */
592 : : void
593 : 223 : MarkSubxactTopXidLogged(void)
594 : : {
595 [ - + ]: 223 : Assert(IsSubxactTopXidLogPending());
596 : :
597 : 223 : CurrentTransactionState->topXidLogged = true;
598 : 223 : }
599 : :
600 : : /*
601 : : * GetStableLatestTransactionId
602 : : *
603 : : * Get the transaction's XID if it has one, else read the next-to-be-assigned
604 : : * XID. Once we have a value, return that same value for the remainder of the
605 : : * current transaction. This is meant to provide the reference point for the
606 : : * age(xid) function, but might be useful for other maintenance tasks as well.
607 : : */
608 : : TransactionId
5163 simon@2ndQuadrant.co 609 : 126 : GetStableLatestTransactionId(void)
610 : : {
611 : : static LocalTransactionId lxid = InvalidLocalTransactionId;
612 : : static TransactionId stablexid = InvalidTransactionId;
613 : :
849 heikki.linnakangas@i 614 [ + + ]: 126 : if (lxid != MyProc->vxid.lxid)
615 : : {
616 : 2 : lxid = MyProc->vxid.lxid;
5162 simon@2ndQuadrant.co 617 : 2 : stablexid = GetTopTransactionIdIfAny();
618 [ + - ]: 2 : if (!TransactionIdIsValid(stablexid))
1961 tmunro@postgresql.or 619 : 2 : stablexid = ReadNextTransactionId();
620 : : }
621 : :
5162 simon@2ndQuadrant.co 622 [ - + ]: 126 : Assert(TransactionIdIsValid(stablexid));
623 : :
5163 624 : 126 : return stablexid;
625 : : }
626 : :
627 : : /*
628 : : * AssignTransactionId
629 : : *
630 : : * Assigns a new permanent FullTransactionId to the given TransactionState.
631 : : * We do not assign XIDs to transactions until/unless this is called.
632 : : * Also, any parent TransactionStates that don't yet have XIDs are assigned
633 : : * one; this maintains the invariant that a child transaction has an XID
634 : : * following its parent's.
635 : : */
636 : : static void
6873 tgl@sss.pgh.pa.us 637 : 172476 : AssignTransactionId(TransactionState s)
638 : : {
6802 bruce@momjian.us 639 : 172476 : bool isSubXact = (s->parent != NULL);
640 : : ResourceOwner currentOwner;
4438 641 : 172476 : bool log_unknown_top = false;
642 : :
643 : : /* Assert that caller didn't screw up */
2651 tmunro@postgresql.or 644 [ - + ]: 172476 : Assert(!FullTransactionIdIsValid(s->fullTransactionId));
7957 tgl@sss.pgh.pa.us 645 [ - + ]: 172476 : Assert(s->state == TRANS_INPROGRESS);
646 : :
647 : : /*
648 : : * Workers synchronize transaction state at the beginning of each parallel
649 : : * operation, so we can't account for new XIDs at this point.
650 : : */
3910 rhaas@postgresql.org 651 [ + - - + ]: 172476 : if (IsInParallelMode() || IsParallelWorker())
824 tgl@sss.pgh.pa.us 652 [ # # ]:UBC 0 : ereport(ERROR,
653 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
654 : : errmsg("cannot assign transaction IDs during a parallel operation")));
655 : :
656 : : /*
657 : : * Ensure parent(s) have XIDs, so that a child always has an XID later
658 : : * than its parent. Mustn't recurse here, or we might get a stack
659 : : * overflow if we're at the bottom of a huge stack of subtransactions none
660 : : * of which have XIDs yet.
661 : : */
2651 tmunro@postgresql.or 662 [ + + + + ]:CBC 172476 : if (isSubXact && !FullTransactionIdIsValid(s->parent->fullTransactionId))
663 : : {
5560 bruce@momjian.us 664 : 625 : TransactionState p = s->parent;
665 : : TransactionState *parents;
666 : 625 : size_t parentOffset = 0;
667 : :
202 michael@paquier.xyz 668 :GNC 625 : parents = palloc_array(TransactionState, s->nestingLevel);
2651 tmunro@postgresql.or 669 [ + + + + ]:CBC 1843 : while (p != NULL && !FullTransactionIdIsValid(p->fullTransactionId))
670 : : {
5821 rhaas@postgresql.org 671 : 1218 : parents[parentOffset++] = p;
672 : 1218 : p = p->parent;
673 : : }
674 : :
675 : : /*
676 : : * This is technically a recursive call, but the recursion will never
677 : : * be more than one layer deep.
678 : : */
679 [ + + ]: 1843 : while (parentOffset != 0)
680 : 1218 : AssignTransactionId(parents[--parentOffset]);
681 : :
682 : 625 : pfree(parents);
683 : : }
684 : :
685 : : /*
686 : : * When effective_wal_level is logical, guarantee that a subtransaction's
687 : : * xid can only be seen in the WAL stream if its toplevel xid has been
688 : : * logged before. If necessary we log an xact_assignment record with fewer
689 : : * than PGPROC_MAX_CACHED_SUBXIDS. Note that it is fine if didLogXid isn't
690 : : * set for a transaction even though it appears in a WAL record, we just
691 : : * might superfluously log something. That can happen when an xid is
692 : : * included somewhere inside a wal record, but not in XLogRecord->xl_xid,
693 : : * like in xl_standby_locks.
694 : : */
4585 695 [ + + + + : 172476 : if (isSubXact && XLogLogicalInfoActive() &&
+ + ]
696 [ + + ]: 304 : !TopTransactionStateData.didLogXid)
697 : 25 : log_unknown_top = true;
698 : :
699 : : /*
700 : : * Generate a new FullTransactionId and record its xid in PGPROC and
701 : : * pg_subtrans.
702 : : *
703 : : * NB: we must make the subtrans entry BEFORE the Xid appears anywhere in
704 : : * shared storage other than PGPROC; because if there's no room for it in
705 : : * PGPROC, the subtrans entry is needed to ensure that other backends see
706 : : * the Xid as "running". See GetNewTransactionId.
707 : : */
2651 tmunro@postgresql.or 708 : 172476 : s->fullTransactionId = GetNewTransactionId(isSubXact);
4079 rhaas@postgresql.org 709 [ + + ]: 172476 : if (!isSubXact)
2651 tmunro@postgresql.or 710 : 166264 : XactTopFullTransactionId = s->fullTransactionId;
711 : :
6873 tgl@sss.pgh.pa.us 712 [ + + ]: 172476 : if (isSubXact)
2651 tmunro@postgresql.or 713 : 6212 : SubTransSetParent(XidFromFullTransactionId(s->fullTransactionId),
714 : 6212 : XidFromFullTransactionId(s->parent->fullTransactionId));
715 : :
716 : : /*
717 : : * If it's a top-level transaction, the predicate locking system needs to
718 : : * be told about it too.
719 : : */
5534 tgl@sss.pgh.pa.us 720 [ + + ]: 172476 : if (!isSubXact)
2651 tmunro@postgresql.or 721 : 166264 : RegisterPredicateLockingXid(XidFromFullTransactionId(s->fullTransactionId));
722 : :
723 : : /*
724 : : * Acquire lock on the transaction XID. (We assume this cannot block.) We
725 : : * have to ensure that the lock is assigned to the transaction's own
726 : : * ResourceOwner.
727 : : */
7957 tgl@sss.pgh.pa.us 728 : 172476 : currentOwner = CurrentResourceOwner;
3184 729 : 172476 : CurrentResourceOwner = s->curTransactionOwner;
730 : :
2651 tmunro@postgresql.or 731 : 172476 : XactLockTableInsert(XidFromFullTransactionId(s->fullTransactionId));
732 : :
7957 tgl@sss.pgh.pa.us 733 : 172476 : CurrentResourceOwner = currentOwner;
734 : :
735 : : /*
736 : : * Every PGPROC_MAX_CACHED_SUBXIDS assigned transaction ids within each
737 : : * top-level transaction we issue a WAL record for the assignment. We
738 : : * include the top-level xid and all the subxids that have not yet been
739 : : * reported using XLOG_XACT_ASSIGNMENT records.
740 : : *
741 : : * This is required to limit the amount of shared memory required in a hot
742 : : * standby server to keep track of in-progress XIDs. See notes for
743 : : * RecordKnownAssignedTransactionIds().
744 : : *
745 : : * We don't keep track of the immediate parent of each subxid, only the
746 : : * top-level transaction that each subxact belongs to. This is correct in
747 : : * recovery only because aborted subtransactions are separately WAL
748 : : * logged.
749 : : *
750 : : * This is correct even for the case where several levels above us didn't
751 : : * have an xid assigned as we recursed up to them beforehand.
752 : : */
6037 simon@2ndQuadrant.co 753 [ + + + + ]: 172476 : if (isSubXact && XLogStandbyInfoActive())
754 : : {
2651 tmunro@postgresql.or 755 : 5933 : unreportedXids[nUnreportedXids] = XidFromFullTransactionId(s->fullTransactionId);
6037 simon@2ndQuadrant.co 756 : 5933 : nUnreportedXids++;
757 : :
758 : : /*
759 : : * ensure this test matches similar one in
760 : : * RecoverPreparedTransactions()
761 : : */
4585 rhaas@postgresql.org 762 [ + + + + ]: 5933 : if (nUnreportedXids >= PGPROC_MAX_CACHED_SUBXIDS ||
763 : : log_unknown_top)
764 : : {
765 : : xl_xact_assignment xlrec;
766 : :
767 : : /*
768 : : * xtop is always set by now because we recurse up transaction
769 : : * stack to the highest unassigned xid and then come back down
770 : : */
6037 simon@2ndQuadrant.co 771 : 94 : xlrec.xtop = GetTopTransactionId();
772 [ - + ]: 94 : Assert(TransactionIdIsValid(xlrec.xtop));
773 : 94 : xlrec.nsubxacts = nUnreportedXids;
774 : :
4240 heikki.linnakangas@i 775 : 94 : XLogBeginInsert();
504 peter@eisentraut.org 776 : 94 : XLogRegisterData(&xlrec, MinSizeOfXactAssignment);
777 : 94 : XLogRegisterData(unreportedXids,
778 : : nUnreportedXids * sizeof(TransactionId));
779 : :
4240 heikki.linnakangas@i 780 : 94 : (void) XLogInsert(RM_XACT_ID, XLOG_XACT_ASSIGNMENT);
781 : :
6037 simon@2ndQuadrant.co 782 : 94 : nUnreportedXids = 0;
783 : : /* mark top, not current xact as having been logged */
4585 rhaas@postgresql.org 784 : 94 : TopTransactionStateData.didLogXid = true;
785 : : }
786 : : }
6037 simon@2ndQuadrant.co 787 : 172476 : }
788 : :
789 : : /*
790 : : * GetCurrentSubTransactionId
791 : : */
792 : : SubTransactionId
7957 tgl@sss.pgh.pa.us 793 : 1977605 : GetCurrentSubTransactionId(void)
794 : : {
795 : 1977605 : TransactionState s = CurrentTransactionState;
796 : :
797 : 1977605 : return s->subTransactionId;
798 : : }
799 : :
800 : : /*
801 : : * SubTransactionIsActive
802 : : *
803 : : * Test if the specified subxact ID is still active. Note caller is
804 : : * responsible for checking whether this ID is relevant to the current xact.
805 : : */
806 : : bool
4867 tgl@sss.pgh.pa.us 807 :UBC 0 : SubTransactionIsActive(SubTransactionId subxid)
808 : : {
809 : : TransactionState s;
810 : :
811 [ # # ]: 0 : for (s = CurrentTransactionState; s != NULL; s = s->parent)
812 : : {
813 [ # # ]: 0 : if (s->state == TRANS_ABORT)
814 : 0 : continue;
815 [ # # ]: 0 : if (s->subTransactionId == subxid)
816 : 0 : return true;
817 : : }
818 : 0 : return false;
819 : : }
820 : :
821 : :
822 : : /*
823 : : * GetCurrentCommandId
824 : : *
825 : : * "used" must be true if the caller intends to use the command ID to mark
826 : : * inserted/updated/deleted tuples. false means the ID is being fetched
827 : : * for read-only purposes (ie, as a snapshot validity cutoff). See
828 : : * CommandCounterIncrement() for discussion.
829 : : */
830 : : CommandId
6787 tgl@sss.pgh.pa.us 831 :CBC 7349242 : GetCurrentCommandId(bool used)
832 : : {
833 : : /* this is global to a transaction, not subtransaction-local */
834 [ + + ]: 7349242 : if (used)
835 : : {
836 : : /*
837 : : * Forbid setting currentCommandIdUsed in a parallel worker, because
838 : : * we have no provision for communicating this back to the leader. We
839 : : * could relax this restriction when currentCommandIdUsed was already
840 : : * true at the start of the parallel operation.
841 : : */
824 842 [ - + ]: 4324717 : if (IsParallelWorker())
824 tgl@sss.pgh.pa.us 843 [ # # ]:UBC 0 : ereport(ERROR,
844 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
845 : : errmsg("cannot modify data in a parallel worker")));
846 : :
6787 tgl@sss.pgh.pa.us 847 :CBC 4324717 : currentCommandIdUsed = true;
848 : : }
7957 849 : 7349242 : return currentCommandId;
850 : : }
851 : :
852 : : /*
853 : : * SetParallelStartTimestamps
854 : : *
855 : : * In a parallel worker, we should inherit the parent transaction's
856 : : * timestamps rather than setting our own. The parallel worker
857 : : * infrastructure must call this to provide those values before
858 : : * calling StartTransaction() or SetCurrentStatementStartTimestamp().
859 : : */
860 : : void
2824 861 : 2012 : SetParallelStartTimestamps(TimestampTz xact_ts, TimestampTz stmt_ts)
862 : : {
863 [ - + ]: 2012 : Assert(IsParallelWorker());
864 : 2012 : xactStartTimestamp = xact_ts;
865 : 2012 : stmtStartTimestamp = stmt_ts;
866 : 2012 : }
867 : :
868 : : /*
869 : : * GetCurrentTransactionStartTimestamp
870 : : */
871 : : TimestampTz
7671 872 : 45367 : GetCurrentTransactionStartTimestamp(void)
873 : : {
874 : 45367 : return xactStartTimestamp;
875 : : }
876 : :
877 : : /*
878 : : * GetCurrentStatementStartTimestamp
879 : : */
880 : : TimestampTz
7371 bruce@momjian.us 881 : 1398440 : GetCurrentStatementStartTimestamp(void)
882 : : {
883 : 1398440 : return stmtStartTimestamp;
884 : : }
885 : :
886 : : /*
887 : : * GetCurrentTransactionStopTimestamp
888 : : *
889 : : * If the transaction stop time hasn't already been set, which can happen if
890 : : * we decided we don't need to log an XLOG record, set xactStopTimestamp.
891 : : */
892 : : TimestampTz
7001 tgl@sss.pgh.pa.us 893 : 1325877 : GetCurrentTransactionStopTimestamp(void)
894 : : {
1355 andres@anarazel.de 895 : 1325877 : TransactionState s PG_USED_FOR_ASSERTS_ONLY = CurrentTransactionState;
896 : :
897 : : /* should only be called after commit / abort processing */
898 [ + + + + : 1325877 : Assert(s->state == TRANS_DEFAULT ||
- + - - ]
899 : : s->state == TRANS_COMMIT ||
900 : : s->state == TRANS_ABORT ||
901 : : s->state == TRANS_PREPARE);
902 : :
903 [ + + ]: 1325877 : if (xactStopTimestamp == 0)
904 : 374735 : xactStopTimestamp = GetCurrentTimestamp();
905 : :
906 : 1325877 : return xactStopTimestamp;
907 : : }
908 : :
909 : : /*
910 : : * SetCurrentStatementStartTimestamp
911 : : *
912 : : * In a parallel worker, this should already have been provided by a call
913 : : * to SetParallelStartTimestamps().
914 : : */
915 : : void
7371 bruce@momjian.us 916 : 656639 : SetCurrentStatementStartTimestamp(void)
917 : : {
2824 tgl@sss.pgh.pa.us 918 [ + + ]: 656639 : if (!IsParallelWorker())
919 : 654627 : stmtStartTimestamp = GetCurrentTimestamp();
920 : : else
921 [ - + ]: 2012 : Assert(stmtStartTimestamp != 0);
7371 bruce@momjian.us 922 : 656639 : }
923 : :
924 : : /*
925 : : * GetCurrentTransactionNestLevel
926 : : *
927 : : * Note: this will return zero when not inside any transaction, one when
928 : : * inside a top-level transaction, etc.
929 : : */
930 : : int
8034 tgl@sss.pgh.pa.us 931 : 25716362 : GetCurrentTransactionNestLevel(void)
932 : : {
933 : 25716362 : TransactionState s = CurrentTransactionState;
934 : :
935 : 25716362 : return s->nestingLevel;
936 : : }
937 : :
938 : :
939 : : /*
940 : : * TransactionIdIsCurrentTransactionId
941 : : */
942 : : bool
10948 scrappy@hub.org 943 : 90464731 : TransactionIdIsCurrentTransactionId(TransactionId xid)
944 : : {
945 : : TransactionState s;
946 : :
947 : : /*
948 : : * We always say that BootstrapTransactionId is "not my transaction ID"
949 : : * even when it is (ie, during bootstrap). Along with the fact that
950 : : * transam.c always treats BootstrapTransactionId as already committed,
951 : : * this causes the heapam_visibility.c routines to see all tuples as
952 : : * committed, which is what we need during bootstrap. (Bootstrap mode
953 : : * only inserts tuples, it never updates or deletes them, so all tuples
954 : : * can be presumed good immediately.)
955 : : *
956 : : * Likewise, InvalidTransactionId and FrozenTransactionId are certainly
957 : : * not my transaction ID, so we can just return "false" immediately for
958 : : * any non-normal XID.
959 : : */
7177 tgl@sss.pgh.pa.us 960 [ + + ]: 90464731 : if (!TransactionIdIsNormal(xid))
10523 bruce@momjian.us 961 : 724887 : return false;
962 : :
2423 tmunro@postgresql.or 963 [ + + ]: 89739844 : if (TransactionIdEquals(xid, GetTopTransactionIdIfAny()))
964 : 61342040 : return true;
965 : :
966 : : /*
967 : : * In parallel workers, the XIDs we must consider as current are stored in
968 : : * ParallelCurrentXids rather than the transaction-state stack. Note that
969 : : * the XIDs in this array are sorted numerically rather than according to
970 : : * transactionIdPrecedes order.
971 : : */
4079 rhaas@postgresql.org 972 [ + + ]: 28397804 : if (nParallelCurrentXids > 0)
973 : : {
974 : : int low,
975 : : high;
976 : :
977 : 4959126 : low = 0;
978 : 4959126 : high = nParallelCurrentXids - 1;
979 [ + + ]: 19532404 : while (low <= high)
980 : : {
981 : : int middle;
982 : : TransactionId probe;
983 : :
984 : 19380470 : middle = low + (high - low) / 2;
985 : 19380470 : probe = ParallelCurrentXids[middle];
986 [ + + ]: 19380470 : if (probe == xid)
987 : 4807192 : return true;
988 [ + + ]: 14573278 : else if (probe < xid)
989 : 14421344 : low = middle + 1;
990 : : else
991 : 151934 : high = middle - 1;
992 : : }
993 : 151934 : return false;
994 : : }
995 : :
996 : : /*
997 : : * We will return true for the Xid of the current subtransaction, any of
998 : : * its subcommitted children, any of its parents, or any of their
999 : : * previously subcommitted children. However, a transaction being aborted
1000 : : * is no longer "current", even though it may still have an entry on the
1001 : : * state stack.
1002 : : */
7976 tgl@sss.pgh.pa.us 1003 [ + + ]: 46801700 : for (s = CurrentTransactionState; s != NULL; s = s->parent)
1004 : : {
1005 : : int low,
1006 : : high;
1007 : :
1008 [ - + ]: 23521416 : if (s->state == TRANS_ABORT)
7976 tgl@sss.pgh.pa.us 1009 :UBC 0 : continue;
2651 tmunro@postgresql.or 1010 [ + + ]:CBC 23521416 : if (!FullTransactionIdIsValid(s->fullTransactionId))
7957 tgl@sss.pgh.pa.us 1011 : 10125255 : continue; /* it can't have any child XIDs either */
2651 tmunro@postgresql.or 1012 [ + + ]: 13396161 : if (TransactionIdEquals(xid, XidFromFullTransactionId(s->fullTransactionId)))
8034 tgl@sss.pgh.pa.us 1013 : 155334 : return true;
1014 : : /* As the childXids array is ordered, we can use binary search */
6679 1015 : 13240827 : low = 0;
1016 : 13240827 : high = s->nChildXids - 1;
1017 [ + + ]: 13241723 : while (low <= high)
1018 : : {
1019 : : int middle;
1020 : : TransactionId probe;
1021 : :
1022 : 3956 : middle = low + (high - low) / 2;
1023 : 3956 : probe = s->childXids[middle];
1024 [ + + ]: 3956 : if (TransactionIdEquals(probe, xid))
8034 1025 : 3060 : return true;
6679 1026 [ + + ]: 896 : else if (TransactionIdPrecedes(probe, xid))
1027 : 803 : low = middle + 1;
1028 : : else
1029 : 93 : high = middle - 1;
1030 : : }
1031 : : }
1032 : :
8034 1033 : 23280284 : return false;
1034 : : }
1035 : :
1036 : : /*
1037 : : * TransactionStartedDuringRecovery
1038 : : *
1039 : : * Returns true if the current transaction started while recovery was still
1040 : : * in progress. Recovery might have ended since so RecoveryInProgress() might
1041 : : * return false already.
1042 : : */
1043 : : bool
6037 simon@2ndQuadrant.co 1044 : 9222387 : TransactionStartedDuringRecovery(void)
1045 : : {
1046 : 9222387 : return CurrentTransactionState->startedInRecovery;
1047 : : }
1048 : :
1049 : : /*
1050 : : * GetTopReadOnlyTransactionNestLevel
1051 : : *
1052 : : * Note: this will return zero when not inside any transaction or when neither
1053 : : * a top-level transaction nor subtransactions are read-only, one when the
1054 : : * top-level transaction is read-only, two when one level of subtransaction is
1055 : : * read-only, etc.
1056 : : *
1057 : : * Note: subtransactions of the topmost read-only transaction are also
1058 : : * read-only, because they inherit read-only mode from the transaction, and
1059 : : * thus can't change to read-write mode (see check_transaction_read_only).
1060 : : */
1061 : : int
86 efujita@postgresql.o 1062 :GNC 9 : GetTopReadOnlyTransactionNestLevel(void)
1063 : : {
1064 : 9 : TransactionState s = CurrentTransactionState;
1065 : :
1066 [ - + ]: 9 : if (!XactReadOnly)
86 efujita@postgresql.o 1067 :UNC 0 : return 0;
86 efujita@postgresql.o 1068 [ + + ]:GNC 10 : while (s->nestingLevel > 1)
1069 : : {
1070 [ + + ]: 4 : if (!s->prevXactReadOnly)
1071 : 3 : return s->nestingLevel;
1072 : 1 : s = s->parent;
1073 : : }
1074 : 6 : return s->nestingLevel;
1075 : : }
1076 : :
1077 : : /*
1078 : : * EnterParallelMode
1079 : : */
1080 : : void
4079 rhaas@postgresql.org 1081 :CBC 4678 : EnterParallelMode(void)
1082 : : {
1083 : 4678 : TransactionState s = CurrentTransactionState;
1084 : :
1085 [ - + ]: 4678 : Assert(s->parallelModeLevel >= 0);
1086 : :
1087 : 4678 : ++s->parallelModeLevel;
1088 : 4678 : }
1089 : :
1090 : : /*
1091 : : * ExitParallelMode
1092 : : */
1093 : : void
1094 : 2657 : ExitParallelMode(void)
1095 : : {
1096 : 2657 : TransactionState s = CurrentTransactionState;
1097 : :
1098 [ - + ]: 2657 : Assert(s->parallelModeLevel > 0);
824 tgl@sss.pgh.pa.us 1099 [ + - + - : 2657 : Assert(s->parallelModeLevel > 1 || s->parallelChildXact ||
- + ]
1100 : : !ParallelContextActive());
1101 : :
4079 rhaas@postgresql.org 1102 : 2657 : --s->parallelModeLevel;
1103 : 2657 : }
1104 : :
1105 : : /*
1106 : : * IsInParallelMode
1107 : : *
1108 : : * Are we in a parallel operation, as either the leader or a worker? Check
1109 : : * this to prohibit operations that change backend-local state expected to
1110 : : * match across all workers. Mere caches usually don't require such a
1111 : : * restriction. State modified in a strict push/pop fashion, such as the
1112 : : * active snapshot stack, is often fine.
1113 : : *
1114 : : * We say we are in parallel mode if we are in a subxact of a transaction
1115 : : * that's initiated a parallel operation; for most purposes that context
1116 : : * has all the same restrictions.
1117 : : */
1118 : : bool
1119 : 7401749 : IsInParallelMode(void)
1120 : : {
824 tgl@sss.pgh.pa.us 1121 : 7401749 : TransactionState s = CurrentTransactionState;
1122 : :
1123 [ + + + + ]: 7401749 : return s->parallelModeLevel != 0 || s->parallelChildXact;
1124 : : }
1125 : :
1126 : : /*
1127 : : * CommandCounterIncrement
1128 : : */
1129 : : void
9380 1130 : 1342776 : CommandCounterIncrement(void)
1131 : : {
1132 : : /*
1133 : : * If the current value of the command counter hasn't been "used" to mark
1134 : : * tuples, we need not increment it, since there's no need to distinguish
1135 : : * a read-only command from others. This helps postpone command counter
1136 : : * overflow, and keeps no-op CommandCounterIncrement operations cheap.
1137 : : */
6787 1138 [ + + ]: 1342776 : if (currentCommandIdUsed)
1139 : : {
1140 : : /*
1141 : : * Workers synchronize transaction state at the beginning of each
1142 : : * parallel operation, so we can't account for new commands after that
1143 : : * point.
1144 : : */
3910 rhaas@postgresql.org 1145 [ + - - + ]: 723085 : if (IsInParallelMode() || IsParallelWorker())
824 tgl@sss.pgh.pa.us 1146 [ # # ]:UBC 0 : ereport(ERROR,
1147 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
1148 : : errmsg("cannot start commands during a parallel operation")));
1149 : :
6787 tgl@sss.pgh.pa.us 1150 :CBC 723085 : currentCommandId += 1;
4677 rhaas@postgresql.org 1151 [ - + ]: 723085 : if (currentCommandId == InvalidCommandId)
1152 : : {
6787 tgl@sss.pgh.pa.us 1153 :UBC 0 : currentCommandId -= 1;
1154 [ # # ]: 0 : ereport(ERROR,
1155 : : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
1156 : : errmsg("cannot have more than 2^32-2 commands in a transaction")));
1157 : : }
6787 tgl@sss.pgh.pa.us 1158 :CBC 723085 : currentCommandIdUsed = false;
1159 : :
1160 : : /* Propagate new command ID into static snapshots */
6623 alvherre@alvh.no-ip. 1161 : 723085 : SnapshotSetCommandId(currentCommandId);
1162 : :
1163 : : /*
1164 : : * Make any catalog changes done by the just-completed command visible
1165 : : * in the local syscache. We obviously don't need to do this after a
1166 : : * read-only command. (But see hacks in inval.c to make real sure we
1167 : : * don't think a command that queued inval messages was read-only.)
1168 : : */
5987 tgl@sss.pgh.pa.us 1169 : 723085 : AtCCI_LocalCache();
1170 : : }
10948 scrappy@hub.org 1171 : 1342772 : }
1172 : :
1173 : : /*
1174 : : * ForceSyncCommit
1175 : : *
1176 : : * Interface routine to allow commands to force a synchronous commit of the
1177 : : * current top-level transaction. Currently, two-phase commit does not
1178 : : * persist and restore this variable. So long as all callers use
1179 : : * PreventInTransactionBlock(), that omission has no consequences.
1180 : : */
1181 : : void
6908 tgl@sss.pgh.pa.us 1182 : 556 : ForceSyncCommit(void)
1183 : : {
1184 : 556 : forceSyncCommit = true;
1185 : 556 : }
1186 : :
1187 : :
1188 : : /* ----------------------------------------------------------------
1189 : : * StartTransaction stuff
1190 : : * ----------------------------------------------------------------
1191 : : */
1192 : :
1193 : : /*
1194 : : * AtStart_Cache
1195 : : */
1196 : : static void
9380 1197 : 425456 : AtStart_Cache(void)
1198 : : {
9142 1199 : 425456 : AcceptInvalidationMessages();
10948 scrappy@hub.org 1200 : 425456 : }
1201 : :
1202 : : /*
1203 : : * AtStart_Memory
1204 : : */
1205 : : static void
9380 tgl@sss.pgh.pa.us 1206 : 425456 : AtStart_Memory(void)
1207 : : {
8034 1208 : 425456 : TransactionState s = CurrentTransactionState;
1209 : :
1210 : : /*
1211 : : * Remember the memory context that was active prior to transaction start.
1212 : : */
729 1213 : 425456 : s->priorContext = CurrentMemoryContext;
1214 : :
1215 : : /*
1216 : : * If this is the first time through, create a private context for
1217 : : * AbortTransaction to work in. By reserving some space now, we can
1218 : : * insulate AbortTransaction from out-of-memory scenarios. Like
1219 : : * ErrorContext, we set it up with slow growth rate and a nonzero minimum
1220 : : * size, so that space will be reserved immediately.
1221 : : */
7159 1222 [ + + ]: 425456 : if (TransactionAbortContext == NULL)
1223 : 18233 : TransactionAbortContext =
2818 1224 : 18233 : AllocSetContextCreate(TopMemoryContext,
1225 : : "TransactionAbortContext",
1226 : : 32 * 1024,
1227 : : 32 * 1024,
1228 : : 32 * 1024);
1229 : :
1230 : : /*
1231 : : * Likewise, if this is the first time through, create a top-level context
1232 : : * for transaction-local data. This context will be reset at transaction
1233 : : * end, and then re-used in later transactions.
1234 : : */
729 1235 [ + + ]: 425456 : if (TopTransactionContext == NULL)
1236 : 18233 : TopTransactionContext =
1237 : 18233 : AllocSetContextCreate(TopMemoryContext,
1238 : : "TopTransactionContext",
1239 : : ALLOCSET_DEFAULT_SIZES);
1240 : :
1241 : : /*
1242 : : * In a top-level transaction, CurTransactionContext is the same as
1243 : : * TopTransactionContext.
1244 : : */
8034 1245 : 425456 : CurTransactionContext = TopTransactionContext;
1246 : 425456 : s->curTransactionContext = CurTransactionContext;
1247 : :
1248 : : /* Make the CurTransactionContext active. */
1249 : 425456 : MemoryContextSwitchTo(CurTransactionContext);
10948 scrappy@hub.org 1250 : 425456 : }
1251 : :
1252 : : /*
1253 : : * AtStart_ResourceOwner
1254 : : */
1255 : : static void
8018 tgl@sss.pgh.pa.us 1256 : 425456 : AtStart_ResourceOwner(void)
1257 : : {
1258 : 425456 : TransactionState s = CurrentTransactionState;
1259 : :
1260 : : /*
1261 : : * We shouldn't have a transaction resource owner already.
1262 : : */
1263 [ - + ]: 425456 : Assert(TopTransactionResourceOwner == NULL);
1264 : :
1265 : : /*
1266 : : * Create a toplevel resource owner for the transaction.
1267 : : */
1268 : 425456 : s->curTransactionOwner = ResourceOwnerCreate(NULL, "TopTransaction");
1269 : :
1270 : 425456 : TopTransactionResourceOwner = s->curTransactionOwner;
1271 : 425456 : CurTransactionResourceOwner = s->curTransactionOwner;
1272 : 425456 : CurrentResourceOwner = s->curTransactionOwner;
1273 : 425456 : }
1274 : :
1275 : : /* ----------------------------------------------------------------
1276 : : * StartSubTransaction stuff
1277 : : * ----------------------------------------------------------------
1278 : : */
1279 : :
1280 : : /*
1281 : : * AtSubStart_Memory
1282 : : */
1283 : : static void
8034 1284 : 12682 : AtSubStart_Memory(void)
1285 : : {
1286 : 12682 : TransactionState s = CurrentTransactionState;
1287 : :
1288 [ - + ]: 12682 : Assert(CurTransactionContext != NULL);
1289 : :
1290 : : /*
1291 : : * Remember the context that was active prior to subtransaction start.
1292 : : */
729 1293 : 12682 : s->priorContext = CurrentMemoryContext;
1294 : :
1295 : : /*
1296 : : * Create a CurTransactionContext, which will be used to hold data that
1297 : : * survives subtransaction commit but disappears on subtransaction abort.
1298 : : * We make it a child of the immediate parent's CurTransactionContext.
1299 : : */
8034 1300 : 12682 : CurTransactionContext = AllocSetContextCreate(CurTransactionContext,
1301 : : "CurTransactionContext",
1302 : : ALLOCSET_DEFAULT_SIZES);
1303 : 12682 : s->curTransactionContext = CurTransactionContext;
1304 : :
1305 : : /* Make the CurTransactionContext active. */
1306 : 12682 : MemoryContextSwitchTo(CurTransactionContext);
1307 : 12682 : }
1308 : :
1309 : : /*
1310 : : * AtSubStart_ResourceOwner
1311 : : */
1312 : : static void
8018 1313 : 12682 : AtSubStart_ResourceOwner(void)
1314 : : {
1315 : 12682 : TransactionState s = CurrentTransactionState;
1316 : :
1317 [ - + ]: 12682 : Assert(s->parent != NULL);
1318 : :
1319 : : /*
1320 : : * Create a resource owner for the subtransaction. We make it a child of
1321 : : * the immediate parent's resource owner.
1322 : : */
1323 : 12682 : s->curTransactionOwner =
1324 : 12682 : ResourceOwnerCreate(s->parent->curTransactionOwner,
1325 : : "SubTransaction");
1326 : :
1327 : 12682 : CurTransactionResourceOwner = s->curTransactionOwner;
1328 : 12682 : CurrentResourceOwner = s->curTransactionOwner;
1329 : 12682 : }
1330 : :
1331 : : /* ----------------------------------------------------------------
1332 : : * CommitTransaction stuff
1333 : : * ----------------------------------------------------------------
1334 : : */
1335 : :
1336 : : /*
1337 : : * RecordTransactionCommit
1338 : : *
1339 : : * Returns latest XID among xact and its children, or InvalidTransactionId
1340 : : * if the xact has no XID. (We compute that here just because it's easier.)
1341 : : *
1342 : : * If you change this function, see RecordTransactionCommitPrepared also.
1343 : : */
1344 : : static TransactionId
5986 1345 : 387843 : RecordTransactionCommit(void)
1346 : : {
6873 1347 : 387843 : TransactionId xid = GetTopTransactionIdIfAny();
6802 bruce@momjian.us 1348 : 387843 : bool markXidCommitted = TransactionIdIsValid(xid);
6870 tgl@sss.pgh.pa.us 1349 : 387843 : TransactionId latestXid = InvalidTransactionId;
1350 : : int nrels;
1351 : : RelFileLocator *rels;
1352 : : int nchildren;
1353 : : TransactionId *children;
1546 andres@anarazel.de 1354 : 387843 : int ndroppedstats = 0;
1355 : 387843 : xl_xact_stats_item *droppedstats = NULL;
5800 rhaas@postgresql.org 1356 : 387843 : int nmsgs = 0;
6037 simon@2ndQuadrant.co 1357 : 387843 : SharedInvalidationMessage *invalMessages = NULL;
5800 rhaas@postgresql.org 1358 : 387843 : bool RelcacheInitFileInval = false;
1359 : : bool wrote_xlog;
1360 : :
1361 : : /*
1362 : : * Log pending invalidations for logical decoding of in-progress
1363 : : * transactions. Normally for DDLs, we log this at each command end,
1364 : : * however, for certain cases where we directly update the system table
1365 : : * without a transaction block, the invalidations are not logged till this
1366 : : * time.
1367 : : */
2168 akapila@postgresql.o 1368 [ + + + + ]: 387843 : if (XLogLogicalInfoActive())
1369 : 14565 : LogLogicalInvalidations();
1370 : :
1371 : : /* Get data needed for commit record */
5800 rhaas@postgresql.org 1372 : 387843 : nrels = smgrGetPendingDeletes(true, &rels);
8018 tgl@sss.pgh.pa.us 1373 : 387843 : nchildren = xactGetCommittedChildren(&children);
1546 andres@anarazel.de 1374 : 387843 : ndroppedstats = pgstat_get_transactional_drops(true, &droppedstats);
5800 rhaas@postgresql.org 1375 [ + + ]: 387843 : if (XLogStandbyInfoActive())
1376 : 329180 : nmsgs = xactGetCommittedInvalidationMessages(&invalMessages,
1377 : : &RelcacheInitFileInval);
5119 heikki.linnakangas@i 1378 : 387843 : wrote_xlog = (XactLastRecEnd != 0);
1379 : :
1380 : : /*
1381 : : * If we haven't been assigned an XID yet, we neither can, nor do we want
1382 : : * to write a COMMIT record.
1383 : : */
6873 tgl@sss.pgh.pa.us 1384 [ + + ]: 387843 : if (!markXidCommitted)
1385 : : {
1386 : : /*
1387 : : * We expect that every RelationDropStorage is followed by a catalog
1388 : : * update, and hence XID assignment, so we shouldn't get here with any
1389 : : * pending deletes. Same is true for dropping stats.
1390 : : *
1391 : : * Use a real test not just an Assert to check this, since it's a bit
1392 : : * fragile.
1393 : : */
1546 andres@anarazel.de 1394 [ + - - + ]: 230440 : if (nrels != 0 || ndroppedstats != 0)
6873 tgl@sss.pgh.pa.us 1395 [ # # ]:UBC 0 : elog(ERROR, "cannot commit a transaction that deleted files but has no xid");
1396 : :
1397 : : /* Can't have child XIDs either; AssignTransactionId enforces this */
6873 tgl@sss.pgh.pa.us 1398 [ - + ]:CBC 230440 : Assert(nchildren == 0);
1399 : :
1400 : : /*
1401 : : * Transactions without an assigned xid can contain invalidation
1402 : : * messages. While inplace updates do this, this is not known to be
1403 : : * necessary; see comment at inplace CacheInvalidateHeapTuple().
1404 : : * Extensions might still rely on this capability, and standbys may
1405 : : * need to process those invals. We can't emit a commit record
1406 : : * without an xid, and we don't want to force assigning an xid,
1407 : : * because that'd be problematic for e.g. vacuum. Hence we emit a
1408 : : * bespoke record for the invalidations. We don't want to use that in
1409 : : * case a commit record is emitted, so they happen synchronously with
1410 : : * commits (besides not wanting to emit more WAL records).
1411 : : *
1412 : : * XXX Every known use of this capability is a defect. Since an XID
1413 : : * isn't controlling visibility of the change that prompted invals,
1414 : : * other sessions need the inval even if this transactions aborts.
1415 : : *
1416 : : * ON COMMIT DELETE ROWS does a nontransactional index_build(), which
1417 : : * queues a relcache inval, including in transactions without an xid
1418 : : * that had read the (empty) table. Standbys don't need any ON COMMIT
1419 : : * DELETE ROWS invals, but we've not done the work to withhold them.
1420 : : */
3720 andres@anarazel.de 1421 [ + + ]: 230440 : if (nmsgs != 0)
1422 : : {
1423 : 11919 : LogStandbyInvalidations(nmsgs, invalMessages,
1424 : : RelcacheInitFileInval);
3673 rhaas@postgresql.org 1425 : 11919 : wrote_xlog = true; /* not strictly necessary */
1426 : : }
1427 : :
1428 : : /*
1429 : : * If we didn't create XLOG entries, we're done here; otherwise we
1430 : : * should trigger flushing those entries the same as a commit record
1431 : : * would. This will primarily happen for HOT pruning and the like; we
1432 : : * want these to be flushed to disk in due time.
1433 : : */
5671 1434 [ + + ]: 230440 : if (!wrote_xlog)
6873 tgl@sss.pgh.pa.us 1435 : 204394 : goto cleanup;
1436 : : }
1437 : : else
1438 : : {
1439 : : bool replorigin;
1440 : :
1441 : : /*
1442 : : * Are we using the replication origins feature? Or, in other words,
1443 : : * are we replaying remote actions?
1444 : : */
153 msawada@postgresql.o 1445 [ + + ]:GNC 158465 : replorigin = (replorigin_xact_state.origin != InvalidReplOriginId &&
1446 [ + - ]: 1062 : replorigin_xact_state.origin != DoNotReplicateId);
1447 : :
1448 : : /*
1449 : : * Mark ourselves as within our "commit critical section". This
1450 : : * forces any concurrent checkpoint to wait until we've updated
1451 : : * pg_xact. Without this, it is possible for the checkpoint to set
1452 : : * REDO after the XLOG record but fail to flush the pg_xact update to
1453 : : * disk, leading to loss of the transaction commit if the system
1454 : : * crashes a little later.
1455 : : *
1456 : : * Note: we could, but don't bother to, set this flag in
1457 : : * RecordTransactionAbort. That's because loss of a transaction abort
1458 : : * is noncritical; the presumption would be that it aborted, anyway.
1459 : : *
1460 : : * It's safe to change the delayChkptFlags flag of our own backend
1461 : : * without holding the ProcArrayLock, since we're the only one
1462 : : * modifying it. This makes checkpoint's determination of which xacts
1463 : : * are delaying the checkpoint a bit fuzzy, but it doesn't matter.
1464 : : *
1465 : : * Note, it is important to get the commit timestamp after marking the
1466 : : * transaction in the commit critical section. See
1467 : : * RecordTransactionCommitPrepared.
1468 : : */
342 akapila@postgresql.o 1469 [ - + ]: 157403 : Assert((MyProc->delayChkptFlags & DELAY_CHKPT_IN_COMMIT) == 0);
6873 tgl@sss.pgh.pa.us 1470 :CBC 157403 : START_CRIT_SECTION();
342 akapila@postgresql.o 1471 :GNC 157403 : MyProc->delayChkptFlags |= DELAY_CHKPT_IN_COMMIT;
1472 : :
1473 [ - + ]: 157403 : Assert(xactStopTimestamp == 0);
1474 : :
1475 : : /*
1476 : : * Ensures the DELAY_CHKPT_IN_COMMIT flag write is globally visible
1477 : : * before commit time is written.
1478 : : */
1479 : 157403 : pg_write_barrier();
1480 : :
1481 : : /*
1482 : : * Insert the commit XLOG record.
1483 : : */
1355 andres@anarazel.de 1484 :CBC 157403 : XactLogCommitRecord(GetCurrentTransactionStopTimestamp(),
1485 : : nchildren, children, nrels, rels,
1486 : : ndroppedstats, droppedstats,
1487 : : nmsgs, invalMessages,
1488 : : RelcacheInitFileInval,
1489 : : MyXactFlags,
1490 : : InvalidTransactionId, NULL /* plain commit */ );
1491 : :
3927 alvherre@alvh.no-ip. 1492 [ + + ]: 157403 : if (replorigin)
1493 : : /* Move LSNs forward for this replication origin */
153 msawada@postgresql.o 1494 :GNC 1062 : replorigin_session_advance(replorigin_xact_state.origin_lsn,
1495 : : XactLastRecEnd);
1496 : :
1497 : : /*
1498 : : * Record commit timestamp. The value comes from plain commit
1499 : : * timestamp if there's no replication origin; otherwise, the
1500 : : * timestamp was already set in replorigin_xact_state.origin_timestamp
1501 : : * by replication.
1502 : : *
1503 : : * We don't need to WAL-log anything here, as the commit record
1504 : : * written above already contains the data.
1505 : : */
1506 : :
1507 [ + + + + ]: 157403 : if (!replorigin || replorigin_xact_state.origin_timestamp == 0)
1508 : 156442 : replorigin_xact_state.origin_timestamp = GetCurrentTransactionStopTimestamp();
1509 : :
4227 alvherre@alvh.no-ip. 1510 :CBC 157403 : TransactionTreeSetCommitTsData(xid, nchildren, children,
1511 : : replorigin_xact_state.origin_timestamp,
153 msawada@postgresql.o 1512 :GNC 157403 : replorigin_xact_state.origin);
1513 : : }
1514 : :
1515 : : /*
1516 : : * Check if we want to commit asynchronously. We can allow the XLOG flush
1517 : : * to happen asynchronously if synchronous_commit=off, or if the current
1518 : : * transaction has not performed any WAL-logged operation or didn't assign
1519 : : * an xid. The transaction can end up not writing any WAL, even if it has
1520 : : * an xid, if it only wrote to temporary and/or unlogged tables. It can
1521 : : * end up having written WAL without an xid if it did HOT pruning. In
1522 : : * case of a crash, the loss of such a transaction will be irrelevant;
1523 : : * temp tables will be lost anyway, unlogged tables will be truncated and
1524 : : * HOT pruning will be done again later. (Given the foregoing, you might
1525 : : * think that it would be unnecessary to emit the XLOG record at all in
1526 : : * this case, but we don't currently try to do that. It would certainly
1527 : : * cause problems at least in Hot Standby mode, where the
1528 : : * KnownAssignedXids machinery requires tracking every XID assignment. It
1529 : : * might be OK to skip it only when wal_level < replica, but for now we
1530 : : * don't.)
1531 : : *
1532 : : * However, if we're doing cleanup of any non-temp rels or committing any
1533 : : * command that wanted to force sync commit, then we must flush XLOG
1534 : : * immediately. (We must not allow asynchronous commit if there are any
1535 : : * non-temp tables to be deleted, because we might delete the files before
1536 : : * the COMMIT record is flushed to disk. We do allow asynchronous commit
1537 : : * if all to-be-deleted tables are temporary though, since they are lost
1538 : : * anyway if we crash.)
1539 : : */
4142 andres@anarazel.de 1540 [ + + + + ]:CBC 183449 : if ((wrote_xlog && markXidCommitted &&
1541 [ + + + + ]: 183449 : synchronous_commit > SYNCHRONOUS_COMMIT_OFF) ||
5566 rhaas@postgresql.org 1542 [ + + ]: 33247 : forceSyncCommit || nrels > 0)
1543 : : {
6873 tgl@sss.pgh.pa.us 1544 : 150216 : XLogFlush(XactLastRecEnd);
1545 : :
1546 : : /*
1547 : : * Now we may update the CLOG, if we wrote a COMMIT record above
1548 : : */
1549 [ + - ]: 150216 : if (markXidCommitted)
6462 alvherre@alvh.no-ip. 1550 : 150216 : TransactionIdCommitTree(xid, nchildren, children);
1551 : : }
1552 : : else
1553 : : {
1554 : : /*
1555 : : * Asynchronous commit case:
1556 : : *
1557 : : * This enables possible committed transaction loss in the case of a
1558 : : * postmaster crash because WAL buffers are left unwritten. Ideally we
1559 : : * could issue the WAL write without the fsync, but some
1560 : : * wal_sync_methods do not allow separate write/fsync.
1561 : : *
1562 : : * Report the latest async commit LSN, so that the WAL writer knows to
1563 : : * flush this commit.
1564 : : */
5815 simon@2ndQuadrant.co 1565 : 33233 : XLogSetAsyncXactLSN(XactLastRecEnd);
1566 : :
1567 : : /*
1568 : : * We must not immediately update the CLOG, since we didn't flush the
1569 : : * XLOG. Instead, we store the LSN up to which the XLOG must be
1570 : : * flushed before the CLOG may be updated.
1571 : : */
6873 tgl@sss.pgh.pa.us 1572 [ + + ]: 33233 : if (markXidCommitted)
6462 alvherre@alvh.no-ip. 1573 : 7187 : TransactionIdAsyncCommitTree(xid, nchildren, children, XactLastRecEnd);
1574 : : }
1575 : :
1576 : : /*
1577 : : * If we entered a commit critical section, leave it now, and let
1578 : : * checkpoints proceed.
1579 : : */
6873 tgl@sss.pgh.pa.us 1580 [ + + ]: 183449 : if (markXidCommitted)
1581 : : {
342 akapila@postgresql.o 1582 :GNC 157403 : MyProc->delayChkptFlags &= ~DELAY_CHKPT_IN_COMMIT;
9300 tgl@sss.pgh.pa.us 1583 [ - + ]:CBC 157403 : END_CRIT_SECTION();
1584 : : }
1585 : :
1586 : : /* Compute latestXid while we have the child XIDs handy */
6870 1587 : 183449 : latestXid = TransactionIdLatest(xid, nchildren, children);
1588 : :
1589 : : /*
1590 : : * Wait for synchronous replication, if required. Similar to the decision
1591 : : * above about using committing asynchronously we only want to wait if
1592 : : * this backend assigned an xid and wrote WAL. No need to wait if an xid
1593 : : * was assigned due to temporary/unlogged tables or due to HOT pruning.
1594 : : *
1595 : : * Note that at this stage we have marked clog, but still show as running
1596 : : * in the procarray and continue to hold locks.
1597 : : */
4142 andres@anarazel.de 1598 [ + + + + ]: 183449 : if (wrote_xlog && markXidCommitted)
3745 rhaas@postgresql.org 1599 : 152677 : SyncRepWaitForLSN(XactLastRecEnd, true);
1600 : :
1601 : : /* remember end of last commit record */
4080 andres@anarazel.de 1602 : 183449 : XactLastCommitEnd = XactLastRecEnd;
1603 : :
1604 : : /* Reset XactLastRecEnd until the next transaction writes something */
5119 heikki.linnakangas@i 1605 : 183449 : XactLastRecEnd = 0;
6873 tgl@sss.pgh.pa.us 1606 : 387843 : cleanup:
1607 : : /* Clean up local data */
7683 1608 [ + + ]: 387843 : if (rels)
1609 : 12719 : pfree(rels);
1546 andres@anarazel.de 1610 [ + + ]: 387843 : if (ndroppedstats)
1611 : 15123 : pfree(droppedstats);
1612 : :
6870 tgl@sss.pgh.pa.us 1613 : 387843 : return latestXid;
1614 : : }
1615 : :
1616 : :
1617 : : /*
1618 : : * AtCCI_LocalCache
1619 : : */
1620 : : static void
5987 1621 : 723085 : AtCCI_LocalCache(void)
1622 : : {
1623 : : /*
1624 : : * Make any pending relation map changes visible. We must do this before
1625 : : * processing local sinval messages, so that the map changes will get
1626 : : * reflected into the relcache when relcache invals are processed.
1627 : : */
1628 : 723085 : AtCCI_RelationMap();
1629 : :
1630 : : /*
1631 : : * Make catalog changes visible to me for the next command.
1632 : : */
8034 1633 : 723085 : CommandEndInvalidationMessages();
9668 inoue@tpf.co.jp 1634 : 723081 : }
1635 : :
1636 : : /*
1637 : : * AtCommit_Memory
1638 : : */
1639 : : static void
9380 tgl@sss.pgh.pa.us 1640 : 390151 : AtCommit_Memory(void)
1641 : : {
729 1642 : 390151 : TransactionState s = CurrentTransactionState;
1643 : :
1644 : : /*
1645 : : * Return to the memory context that was current before we started the
1646 : : * transaction. (In principle, this could not be any of the contexts we
1647 : : * are about to delete. If it somehow is, assertions in mcxt.c will
1648 : : * complain.)
1649 : : */
1650 : 390151 : MemoryContextSwitchTo(s->priorContext);
1651 : :
1652 : : /*
1653 : : * Release all transaction-local memory. TopTransactionContext survives
1654 : : * but becomes empty; any sub-contexts go away.
1655 : : */
9494 1656 [ - + ]: 390151 : Assert(TopTransactionContext != NULL);
729 1657 : 390151 : MemoryContextReset(TopTransactionContext);
1658 : :
1659 : : /*
1660 : : * Clear these pointers as a pro-forma matter. (Notionally, while
1661 : : * TopTransactionContext still exists, it's currently not associated with
1662 : : * this TransactionState struct.)
1663 : : */
8034 1664 : 390151 : CurTransactionContext = NULL;
729 1665 : 390151 : s->curTransactionContext = NULL;
8034 1666 : 390151 : }
1667 : :
1668 : : /* ----------------------------------------------------------------
1669 : : * CommitSubTransaction stuff
1670 : : * ----------------------------------------------------------------
1671 : : */
1672 : :
1673 : : /*
1674 : : * AtSubCommit_Memory
1675 : : */
1676 : : static void
1677 : 7262 : AtSubCommit_Memory(void)
1678 : : {
1679 : 7262 : TransactionState s = CurrentTransactionState;
1680 : :
1681 [ - + ]: 7262 : Assert(s->parent != NULL);
1682 : :
1683 : : /* Return to parent transaction level's memory context. */
1684 : 7262 : CurTransactionContext = s->parent->curTransactionContext;
1685 : 7262 : MemoryContextSwitchTo(CurTransactionContext);
1686 : :
1687 : : /*
1688 : : * Ordinarily we cannot throw away the child's CurTransactionContext,
1689 : : * since the data it contains will be needed at upper commit. However, if
1690 : : * there isn't actually anything in it, we can throw it away. This avoids
1691 : : * a small memory leak in the common case of "trivial" subxacts.
1692 : : */
7957 1693 [ + + ]: 7262 : if (MemoryContextIsEmpty(s->curTransactionContext))
1694 : : {
1695 : 7245 : MemoryContextDelete(s->curTransactionContext);
1696 : 7245 : s->curTransactionContext = NULL;
1697 : : }
8034 1698 : 7262 : }
1699 : :
1700 : : /*
1701 : : * AtSubCommit_childXids
1702 : : *
1703 : : * Pass my own XID and my child XIDs up to my parent as committed children.
1704 : : */
1705 : : static void
1706 : 5353 : AtSubCommit_childXids(void)
1707 : : {
1708 : 5353 : TransactionState s = CurrentTransactionState;
1709 : : int new_nChildXids;
1710 : :
1711 [ - + ]: 5353 : Assert(s->parent != NULL);
1712 : :
1713 : : /*
1714 : : * The parent childXids array will need to hold my XID and all my
1715 : : * childXids, in addition to the XIDs already there.
1716 : : */
6679 1717 : 5353 : new_nChildXids = s->parent->nChildXids + s->nChildXids + 1;
1718 : :
1719 : : /* Allocate or enlarge the parent array if necessary */
1720 [ + + ]: 5353 : if (s->parent->maxChildXids < new_nChildXids)
1721 : : {
1722 : : int new_maxChildXids;
1723 : : TransactionId *new_childXids;
1724 : :
1725 : : /*
1726 : : * Make it 2x what's needed right now, to avoid having to enlarge it
1727 : : * repeatedly. But we can't go above MaxAllocSize. (The latter limit
1728 : : * is what ensures that we don't need to worry about integer overflow
1729 : : * here or in the calculation of new_nChildXids.)
1730 : : */
1731 : 1714 : new_maxChildXids = Min(new_nChildXids * 2,
1732 : : (int) (MaxAllocSize / sizeof(TransactionId)));
1733 : :
1734 [ - + ]: 1714 : if (new_maxChildXids < new_nChildXids)
6679 tgl@sss.pgh.pa.us 1735 [ # # ]:UBC 0 : ereport(ERROR,
1736 : : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
1737 : : errmsg("maximum number of committed subtransactions (%d) exceeded",
1738 : : (int) (MaxAllocSize / sizeof(TransactionId)))));
1739 : :
1740 : : /*
1741 : : * We keep the child-XID arrays in TopTransactionContext; this avoids
1742 : : * setting up child-transaction contexts for what might be just a few
1743 : : * bytes of grandchild XIDs.
1744 : : */
6679 tgl@sss.pgh.pa.us 1745 [ + + ]:CBC 1714 : if (s->parent->childXids == NULL)
1746 : : new_childXids =
6228 bruce@momjian.us 1747 : 1654 : MemoryContextAlloc(TopTransactionContext,
1748 : : new_maxChildXids * sizeof(TransactionId));
1749 : : else
1750 : 60 : new_childXids = repalloc(s->parent->childXids,
1751 : : new_maxChildXids * sizeof(TransactionId));
1752 : :
1753 : 1714 : s->parent->childXids = new_childXids;
6679 tgl@sss.pgh.pa.us 1754 : 1714 : s->parent->maxChildXids = new_maxChildXids;
1755 : : }
1756 : :
1757 : : /*
1758 : : * Copy all my XIDs to parent's array.
1759 : : *
1760 : : * Note: We rely on the fact that the XID of a child always follows that
1761 : : * of its parent. By copying the XID of this subtransaction before the
1762 : : * XIDs of its children, we ensure that the array stays ordered. Likewise,
1763 : : * all XIDs already in the array belong to subtransactions started and
1764 : : * subcommitted before us, so their XIDs must precede ours.
1765 : : */
2651 tmunro@postgresql.or 1766 : 5353 : s->parent->childXids[s->parent->nChildXids] = XidFromFullTransactionId(s->fullTransactionId);
1767 : :
6679 tgl@sss.pgh.pa.us 1768 [ + + ]: 5353 : if (s->nChildXids > 0)
1769 : 1018 : memcpy(&s->parent->childXids[s->parent->nChildXids + 1],
1770 : 1018 : s->childXids,
1771 : 1018 : s->nChildXids * sizeof(TransactionId));
1772 : :
1773 : 5353 : s->parent->nChildXids = new_nChildXids;
1774 : :
1775 : : /* Release child's array to avoid leakage */
1776 [ + + ]: 5353 : if (s->childXids != NULL)
1777 : 1018 : pfree(s->childXids);
1778 : : /* We must reset these to avoid double-free if fail later in commit */
1779 : 5353 : s->childXids = NULL;
1780 : 5353 : s->nChildXids = 0;
1781 : 5353 : s->maxChildXids = 0;
8034 1782 : 5353 : }
1783 : :
1784 : : /* ----------------------------------------------------------------
1785 : : * AbortTransaction stuff
1786 : : * ----------------------------------------------------------------
1787 : : */
1788 : :
1789 : : /*
1790 : : * RecordTransactionAbort
1791 : : *
1792 : : * Returns latest XID among xact and its children, or InvalidTransactionId
1793 : : * if the xact has no XID. (We compute that here just because it's easier.)
1794 : : */
1795 : : static TransactionId
6873 1796 : 40716 : RecordTransactionAbort(bool isSubXact)
1797 : : {
1798 : 40716 : TransactionId xid = GetCurrentTransactionIdIfAny();
1799 : : TransactionId latestXid;
1800 : : int nrels;
1801 : : RelFileLocator *rels;
1546 andres@anarazel.de 1802 : 40716 : int ndroppedstats = 0;
1803 : 40716 : xl_xact_stats_item *droppedstats = NULL;
1804 : : int nchildren;
1805 : : TransactionId *children;
1806 : : TimestampTz xact_time;
1807 : : bool replorigin;
1808 : :
1809 : : /*
1810 : : * If we haven't been assigned an XID, nobody will care whether we aborted
1811 : : * or not. Hence, we're done in that case. It does not matter if we have
1812 : : * rels to delete (note that this routine is not responsible for actually
1813 : : * deleting 'em). We cannot have any child XIDs, either.
1814 : : */
6873 tgl@sss.pgh.pa.us 1815 [ + + ]: 40716 : if (!TransactionIdIsValid(xid))
1816 : : {
1817 : : /* Reset XactLastRecEnd until the next transaction writes something */
1818 [ + + ]: 31301 : if (!isSubXact)
5119 heikki.linnakangas@i 1819 : 26740 : XactLastRecEnd = 0;
6870 tgl@sss.pgh.pa.us 1820 : 31301 : return InvalidTransactionId;
1821 : : }
1822 : :
1823 : : /*
1824 : : * We have a valid XID, so we should write an ABORT record for it.
1825 : : *
1826 : : * We do not flush XLOG to disk here, since the default assumption after a
1827 : : * crash would be that we aborted, anyway. For the same reason, we don't
1828 : : * need to worry about interlocking against checkpoint start.
1829 : : */
1830 : :
1831 : : /*
1832 : : * Check that we haven't aborted halfway through RecordTransactionCommit.
1833 : : */
6873 1834 [ - + ]: 9415 : if (TransactionIdDidCommit(xid))
6873 tgl@sss.pgh.pa.us 1835 [ # # ]:UBC 0 : elog(PANIC, "cannot abort transaction %u, it was already committed",
1836 : : xid);
1837 : :
1838 : : /*
1839 : : * Are we using the replication origins feature? Or, in other words, are
1840 : : * we replaying remote actions?
1841 : : */
153 msawada@postgresql.o 1842 [ + + ]:GNC 9436 : replorigin = (replorigin_xact_state.origin != InvalidReplOriginId &&
1843 [ + - ]: 21 : replorigin_xact_state.origin != DoNotReplicateId);
1844 : :
1845 : : /* Fetch the data we need for the abort record */
5800 rhaas@postgresql.org 1846 :CBC 9415 : nrels = smgrGetPendingDeletes(false, &rels);
6873 tgl@sss.pgh.pa.us 1847 : 9415 : nchildren = xactGetCommittedChildren(&children);
1546 andres@anarazel.de 1848 : 9415 : ndroppedstats = pgstat_get_transactional_drops(false, &droppedstats);
1849 : :
1850 : : /* XXX do we really need a critical section here? */
6873 tgl@sss.pgh.pa.us 1851 : 9415 : START_CRIT_SECTION();
1852 : :
1853 : : /* Write the ABORT record */
1854 [ + + ]: 9415 : if (isSubXact)
4125 andres@anarazel.de 1855 : 859 : xact_time = GetCurrentTimestamp();
1856 : : else
1857 : : {
1355 1858 : 8556 : xact_time = GetCurrentTransactionStopTimestamp();
1859 : : }
1860 : :
4125 1861 : 9415 : XactLogAbortRecord(xact_time,
1862 : : nchildren, children,
1863 : : nrels, rels,
1864 : : ndroppedstats, droppedstats,
1865 : : MyXactFlags, InvalidTransactionId,
1866 : : NULL);
1867 : :
1268 akapila@postgresql.o 1868 [ + + ]: 9415 : if (replorigin)
1869 : : /* Move LSNs forward for this replication origin */
153 msawada@postgresql.o 1870 :GNC 21 : replorigin_session_advance(replorigin_xact_state.origin_lsn,
1871 : : XactLastRecEnd);
1872 : :
1873 : : /*
1874 : : * Report the latest async abort LSN, so that the WAL writer knows to
1875 : : * flush this abort. There's nothing to be gained by delaying this, since
1876 : : * WALWriter may as well do this when it can. This is important with
1877 : : * streaming replication because if we don't flush WAL regularly we will
1878 : : * find that large aborts leave us with a long backlog for when commits
1879 : : * occur after the abort, increasing our window of data loss should
1880 : : * problems occur at that point.
1881 : : */
5892 simon@2ndQuadrant.co 1882 [ + + ]:CBC 9415 : if (!isSubXact)
5815 1883 : 8556 : XLogSetAsyncXactLSN(XactLastRecEnd);
1884 : :
1885 : : /*
1886 : : * Mark the transaction aborted in clog. This is not absolutely necessary
1887 : : * but we may as well do it while we are here; also, in the subxact case
1888 : : * it is helpful because XactLockTableWait makes use of it to avoid
1889 : : * waiting for already-aborted subtransactions. It is OK to do it without
1890 : : * having flushed the ABORT record to disk, because in event of a crash
1891 : : * we'd be assumed to have aborted anyway.
1892 : : */
6462 alvherre@alvh.no-ip. 1893 : 9415 : TransactionIdAbortTree(xid, nchildren, children);
1894 : :
6873 tgl@sss.pgh.pa.us 1895 [ - + ]: 9415 : END_CRIT_SECTION();
1896 : :
1897 : : /* Compute latestXid while we have the child XIDs handy */
6870 1898 : 9415 : latestXid = TransactionIdLatest(xid, nchildren, children);
1899 : :
1900 : : /*
1901 : : * If we're aborting a subtransaction, we can immediately remove failed
1902 : : * XIDs from PGPROC's cache of running child XIDs. We do that here for
1903 : : * subxacts, because we already have the child XID array at hand. For
1904 : : * main xacts, the equivalent happens just after this function returns.
1905 : : */
6873 1906 [ + + ]: 9415 : if (isSubXact)
6870 1907 : 859 : XidCacheRemoveRunningXids(xid, nchildren, children, latestXid);
1908 : :
1909 : : /* Reset XactLastRecEnd until the next transaction writes something */
6873 1910 [ + + ]: 9415 : if (!isSubXact)
5119 heikki.linnakangas@i 1911 : 8556 : XactLastRecEnd = 0;
1912 : :
1913 : : /* And clean up local data */
7683 tgl@sss.pgh.pa.us 1914 [ + + ]: 9415 : if (rels)
1915 : 1350 : pfree(rels);
1546 andres@anarazel.de 1916 [ + + ]: 9415 : if (ndroppedstats)
1917 : 1928 : pfree(droppedstats);
1918 : :
6870 tgl@sss.pgh.pa.us 1919 : 9415 : return latestXid;
1920 : : }
1921 : :
1922 : : /*
1923 : : * AtAbort_Memory
1924 : : */
1925 : : static void
9380 1926 : 54069 : AtAbort_Memory(void)
1927 : : {
1928 : : /*
1929 : : * Switch into TransactionAbortContext, which should have some free space
1930 : : * even if nothing else does. We'll work in this context until we've
1931 : : * finished cleaning up.
1932 : : *
1933 : : * It is barely possible to get here when we've not been able to create
1934 : : * TransactionAbortContext yet; if so use TopMemoryContext.
1935 : : */
7159 1936 [ + - ]: 54069 : if (TransactionAbortContext != NULL)
1937 : 54069 : MemoryContextSwitchTo(TransactionAbortContext);
1938 : : else
9494 tgl@sss.pgh.pa.us 1939 :UBC 0 : MemoryContextSwitchTo(TopMemoryContext);
9498 tgl@sss.pgh.pa.us 1940 :CBC 54069 : }
1941 : :
1942 : : /*
1943 : : * AtSubAbort_Memory
1944 : : */
1945 : : static void
8034 1946 : 5420 : AtSubAbort_Memory(void)
1947 : : {
7159 1948 [ - + ]: 5420 : Assert(TransactionAbortContext != NULL);
1949 : :
1950 : 5420 : MemoryContextSwitchTo(TransactionAbortContext);
8034 1951 : 5420 : }
1952 : :
1953 : :
1954 : : /*
1955 : : * AtAbort_ResourceOwner
1956 : : */
1957 : : static void
7939 1958 : 35305 : AtAbort_ResourceOwner(void)
1959 : : {
1960 : : /*
1961 : : * Make sure we have a valid ResourceOwner, if possible (else it will be
1962 : : * NULL, which is OK)
1963 : : */
1964 : 35305 : CurrentResourceOwner = TopTransactionResourceOwner;
1965 : 35305 : }
1966 : :
1967 : : /*
1968 : : * AtSubAbort_ResourceOwner
1969 : : */
1970 : : static void
1971 : 5420 : AtSubAbort_ResourceOwner(void)
1972 : : {
1973 : 5420 : TransactionState s = CurrentTransactionState;
1974 : :
1975 : : /* Make sure we have a valid ResourceOwner */
1976 : 5420 : CurrentResourceOwner = s->curTransactionOwner;
1977 : 5420 : }
1978 : :
1979 : :
1980 : : /*
1981 : : * AtSubAbort_childXids
1982 : : */
1983 : : static void
7957 1984 : 859 : AtSubAbort_childXids(void)
1985 : : {
1986 : 859 : TransactionState s = CurrentTransactionState;
1987 : :
1988 : : /*
1989 : : * We keep the child-XID arrays in TopTransactionContext (see
1990 : : * AtSubCommit_childXids). This means we'd better free the array
1991 : : * explicitly at abort to avoid leakage.
1992 : : */
6679 1993 [ + + ]: 859 : if (s->childXids != NULL)
1994 : 26 : pfree(s->childXids);
1995 : 859 : s->childXids = NULL;
1996 : 859 : s->nChildXids = 0;
1997 : 859 : s->maxChildXids = 0;
1998 : :
1999 : : /*
2000 : : * We could prune the unreportedXids array here. But we don't bother. That
2001 : : * would potentially reduce number of XLOG_XACT_ASSIGNMENT records but it
2002 : : * would likely introduce more CPU time into the more common paths, so we
2003 : : * choose not to do that.
2004 : : */
7957 2005 : 859 : }
2006 : :
2007 : : /* ----------------------------------------------------------------
2008 : : * CleanupTransaction stuff
2009 : : * ----------------------------------------------------------------
2010 : : */
2011 : :
2012 : : /*
2013 : : * AtCleanup_Memory
2014 : : */
2015 : : static void
9380 2016 : 35305 : AtCleanup_Memory(void)
2017 : : {
729 2018 : 35305 : TransactionState s = CurrentTransactionState;
2019 : :
2020 : : /* Should be at top level */
2021 [ - + ]: 35305 : Assert(s->parent == NULL);
2022 : :
2023 : : /*
2024 : : * Return to the memory context that was current before we started the
2025 : : * transaction. (In principle, this could not be any of the contexts we
2026 : : * are about to delete. If it somehow is, assertions in mcxt.c will
2027 : : * complain.)
2028 : : */
2029 : 35305 : MemoryContextSwitchTo(s->priorContext);
2030 : :
2031 : : /*
2032 : : * Clear the special abort context for next time.
2033 : : */
7159 2034 [ + - ]: 35305 : if (TransactionAbortContext != NULL)
958 nathan@postgresql.or 2035 : 35305 : MemoryContextReset(TransactionAbortContext);
2036 : :
2037 : : /*
2038 : : * Release all transaction-local memory, the same as in AtCommit_Memory,
2039 : : * except we must cope with the possibility that we didn't get as far as
2040 : : * creating TopTransactionContext.
2041 : : */
9494 tgl@sss.pgh.pa.us 2042 [ + - ]: 35305 : if (TopTransactionContext != NULL)
729 2043 : 35305 : MemoryContextReset(TopTransactionContext);
2044 : :
2045 : : /*
2046 : : * Clear these pointers as a pro-forma matter. (Notionally, while
2047 : : * TopTransactionContext still exists, it's currently not associated with
2048 : : * this TransactionState struct.)
2049 : : */
8034 2050 : 35305 : CurTransactionContext = NULL;
729 2051 : 35305 : s->curTransactionContext = NULL;
10948 scrappy@hub.org 2052 : 35305 : }
2053 : :
2054 : :
2055 : : /* ----------------------------------------------------------------
2056 : : * CleanupSubTransaction stuff
2057 : : * ----------------------------------------------------------------
2058 : : */
2059 : :
2060 : : /*
2061 : : * AtSubCleanup_Memory
2062 : : */
2063 : : static void
8034 tgl@sss.pgh.pa.us 2064 : 5420 : AtSubCleanup_Memory(void)
2065 : : {
2066 : 5420 : TransactionState s = CurrentTransactionState;
2067 : :
2068 [ - + ]: 5420 : Assert(s->parent != NULL);
2069 : :
2070 : : /*
2071 : : * Return to the memory context that was current before we started the
2072 : : * subtransaction. (In principle, this could not be any of the contexts
2073 : : * we are about to delete. If it somehow is, assertions in mcxt.c will
2074 : : * complain.)
2075 : : */
729 2076 : 5420 : MemoryContextSwitchTo(s->priorContext);
2077 : :
2078 : : /* Update CurTransactionContext (might not be same as priorContext) */
8034 2079 : 5420 : CurTransactionContext = s->parent->curTransactionContext;
2080 : :
2081 : : /*
2082 : : * Clear the special abort context for next time.
2083 : : */
7159 2084 [ + - ]: 5420 : if (TransactionAbortContext != NULL)
958 nathan@postgresql.or 2085 : 5420 : MemoryContextReset(TransactionAbortContext);
2086 : :
2087 : : /*
2088 : : * Delete the subxact local memory contexts. Its CurTransactionContext can
2089 : : * go too (note this also kills CurTransactionContexts from any children
2090 : : * of the subxact).
2091 : : */
7957 tgl@sss.pgh.pa.us 2092 [ + - ]: 5420 : if (s->curTransactionContext)
2093 : 5420 : MemoryContextDelete(s->curTransactionContext);
2094 : 5420 : s->curTransactionContext = NULL;
8034 2095 : 5420 : }
2096 : :
2097 : : /* ----------------------------------------------------------------
2098 : : * interface routines
2099 : : * ----------------------------------------------------------------
2100 : : */
2101 : :
2102 : : /*
2103 : : * StartTransaction
2104 : : */
2105 : : static void
9380 2106 : 425456 : StartTransaction(void)
2107 : : {
2108 : : TransactionState s;
2109 : : VirtualTransactionId vxid;
2110 : :
2111 : : /*
2112 : : * Let's just make sure the state stack is empty
2113 : : */
7957 2114 : 425456 : s = &TopTransactionStateData;
2115 : 425456 : CurrentTransactionState = s;
2116 : :
2651 tmunro@postgresql.or 2117 [ - + ]: 425456 : Assert(!FullTransactionIdIsValid(XactTopFullTransactionId));
2118 : :
2119 : : /* check the current transaction state */
2785 michael@paquier.xyz 2120 [ - + ]: 425456 : Assert(s->state == TRANS_DEFAULT);
2121 : :
2122 : : /*
2123 : : * Set the current transaction state information appropriately during
2124 : : * start processing. Note that once the transaction status is switched
2125 : : * this process cannot fail until the user ID and the security context
2126 : : * flags are fetched below.
2127 : : */
10523 bruce@momjian.us 2128 : 425456 : s->state = TRANS_START;
2651 tmunro@postgresql.or 2129 : 425456 : s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
2130 : :
2131 : : /* Determine if statements are logged in this transaction */
2645 alvherre@alvh.no-ip. 2132 [ - + ]: 425456 : xact_is_sampled = log_xact_sample_rate != 0 &&
2645 alvherre@alvh.no-ip. 2133 [ # # ]:UBC 0 : (log_xact_sample_rate == 1 ||
1675 tgl@sss.pgh.pa.us 2134 [ # # ]: 0 : pg_prng_double(&pg_global_prng_state) <= log_xact_sample_rate);
2135 : :
2136 : : /*
2137 : : * initialize current transaction state fields
2138 : : *
2139 : : * note: prevXactReadOnly is not used at the outermost level
2140 : : */
2785 michael@paquier.xyz 2141 :CBC 425456 : s->nestingLevel = 1;
2142 : 425456 : s->gucNestLevel = 1;
2143 : 425456 : s->childXids = NULL;
2144 : 425456 : s->nChildXids = 0;
2145 : 425456 : s->maxChildXids = 0;
2146 : :
2147 : : /*
2148 : : * Once the current user ID and the security context flags are fetched,
2149 : : * both will be properly reset even if transaction startup fails.
2150 : : */
2151 : 425456 : GetUserIdAndSecContext(&s->prevUser, &s->prevSecContext);
2152 : :
2153 : : /* SecurityRestrictionContext should never be set outside a transaction */
2154 [ - + ]: 425456 : Assert(s->prevSecContext == 0);
2155 : :
2156 : : /*
2157 : : * Make sure we've reset xact state variables
2158 : : *
2159 : : * If recovery is still in progress, mark this transaction as read-only.
2160 : : * We have lower level defences in XLogInsert and elsewhere to stop us
2161 : : * from modifying data during recovery, but this gives the normal
2162 : : * indication to the user that the transaction is read-only.
2163 : : */
6037 simon@2ndQuadrant.co 2164 [ + + ]: 425456 : if (RecoveryInProgress())
2165 : : {
2166 : 2362 : s->startedInRecovery = true;
2167 : 2362 : XactReadOnly = true;
2168 : : }
2169 : : else
2170 : : {
2171 : 423094 : s->startedInRecovery = false;
2172 : 423094 : XactReadOnly = DefaultXactReadOnly;
2173 : : }
5622 heikki.linnakangas@i 2174 : 425456 : XactDeferrable = DefaultXactDeferrable;
8293 tgl@sss.pgh.pa.us 2175 : 425456 : XactIsoLevel = DefaultXactIsoLevel;
6908 2176 : 425456 : forceSyncCommit = false;
3387 simon@2ndQuadrant.co 2177 : 425456 : MyXactFlags = 0;
2178 : :
2179 : : /*
2180 : : * reinitialize within-transaction counters
2181 : : */
7957 tgl@sss.pgh.pa.us 2182 : 425456 : s->subTransactionId = TopSubTransactionId;
2183 : 425456 : currentSubTransactionId = TopSubTransactionId;
2184 : 425456 : currentCommandId = FirstCommandId;
6787 2185 : 425456 : currentCommandIdUsed = false;
2186 : :
2187 : : /*
2188 : : * initialize reported xid accounting
2189 : : */
6037 simon@2ndQuadrant.co 2190 : 425456 : nUnreportedXids = 0;
4585 rhaas@postgresql.org 2191 : 425456 : s->didLogXid = false;
2192 : :
2193 : : /*
2194 : : * must initialize resource-management stuff first
2195 : : */
8018 tgl@sss.pgh.pa.us 2196 : 425456 : AtStart_Memory();
2197 : 425456 : AtStart_ResourceOwner();
2198 : :
2199 : : /*
2200 : : * Assign a new LocalTransactionId, and combine it with the proc number to
2201 : : * form a virtual transaction id.
2202 : : */
849 heikki.linnakangas@i 2203 : 425456 : vxid.procNumber = MyProcNumber;
6873 tgl@sss.pgh.pa.us 2204 : 425456 : vxid.localTransactionId = GetNextLocalTransactionId();
2205 : :
2206 : : /*
2207 : : * Lock the virtual transaction id before we announce it in the proc array
2208 : : */
2209 : 425456 : VirtualXactLockTableInsert(vxid);
2210 : :
2211 : : /*
2212 : : * Advertise it in the proc array. We assume assignment of
2213 : : * localTransactionId is atomic, and the proc number should be set
2214 : : * already.
2215 : : */
849 heikki.linnakangas@i 2216 [ - + ]: 425456 : Assert(MyProc->vxid.procNumber == vxid.procNumber);
2217 : 425456 : MyProc->vxid.lxid = vxid.localTransactionId;
2218 : :
2219 : : TRACE_POSTGRESQL_TRANSACTION_START(vxid.localTransactionId);
2220 : :
2221 : : /*
2222 : : * set transaction_timestamp() (a/k/a now()). Normally, we want this to
2223 : : * be the same as the first command's statement_timestamp(), so don't do a
2224 : : * fresh GetCurrentTimestamp() call (which'd be expensive anyway). But
2225 : : * for transactions started inside procedures (i.e., nonatomic SPI
2226 : : * contexts), we do need to advance the timestamp. Also, in a parallel
2227 : : * worker, the timestamp should already have been provided by a call to
2228 : : * SetParallelStartTimestamps().
2229 : : */
2824 tgl@sss.pgh.pa.us 2230 [ + + ]: 425456 : if (!IsParallelWorker())
2231 : : {
2822 2232 [ + + ]: 419420 : if (!SPI_inside_nonatomic_context())
2233 : 417201 : xactStartTimestamp = stmtStartTimestamp;
2234 : : else
2235 : 2219 : xactStartTimestamp = GetCurrentTimestamp();
2236 : : }
2237 : : else
2824 2238 [ - + ]: 6036 : Assert(xactStartTimestamp != 0);
6867 2239 : 425456 : pgstat_report_xact_timestamp(xactStartTimestamp);
2240 : : /* Mark xactStopTimestamp as unset. */
2822 2241 : 425456 : xactStopTimestamp = 0;
2242 : :
2243 : : /*
2244 : : * initialize other subsystems for new transaction
2245 : : */
6875 2246 : 425456 : AtStart_GUC();
10523 bruce@momjian.us 2247 : 425456 : AtStart_Cache();
7963 tgl@sss.pgh.pa.us 2248 : 425456 : AfterTriggerBeginXact();
2249 : :
2250 : : /*
2251 : : * done with start processing, set current transaction state to "in
2252 : : * progress"
2253 : : */
10523 bruce@momjian.us 2254 : 425456 : s->state = TRANS_INPROGRESS;
2255 : :
2256 : : /* Schedule transaction timeout */
866 akorotkov@postgresql 2257 [ + + ]: 425456 : if (TransactionTimeout > 0)
2258 : 1 : enable_timeout_after(TRANSACTION_TIMEOUT, TransactionTimeout);
2259 : :
8034 tgl@sss.pgh.pa.us 2260 : 425456 : ShowTransactionState("StartTransaction");
10948 scrappy@hub.org 2261 : 425456 : }
2262 : :
2263 : :
2264 : : /*
2265 : : * CommitTransaction
2266 : : *
2267 : : * NB: if you change this routine, better look at PrepareTransaction too!
2268 : : */
2269 : : static void
9380 tgl@sss.pgh.pa.us 2270 : 390145 : CommitTransaction(void)
2271 : : {
10523 bruce@momjian.us 2272 : 390145 : TransactionState s = CurrentTransactionState;
2273 : : TransactionId latestXid;
2274 : : bool is_parallel_worker;
2275 : :
4079 rhaas@postgresql.org 2276 : 390145 : is_parallel_worker = (s->blockState == TBLOCK_PARALLEL_INPROGRESS);
2277 : :
2278 : : /* Enforce parallel mode restrictions during parallel worker commit. */
3910 2279 [ + + ]: 390145 : if (is_parallel_worker)
2280 : 2003 : EnterParallelMode();
2281 : :
8034 tgl@sss.pgh.pa.us 2282 : 390145 : ShowTransactionState("CommitTransaction");
2283 : :
2284 : : /*
2285 : : * check the current transaction state
2286 : : */
10523 bruce@momjian.us 2287 [ - + ]: 390145 : if (s->state != TRANS_INPROGRESS)
8007 tgl@sss.pgh.pa.us 2288 [ # # ]:UBC 0 : elog(WARNING, "CommitTransaction while in %s state",
2289 : : TransStateAsString(s->state));
8034 tgl@sss.pgh.pa.us 2290 [ + - ]:CBC 390145 : Assert(s->parent == NULL);
2291 : :
2292 : : /*
2293 : : * Do pre-commit processing that involves calling user-defined code, such
2294 : : * as triggers. SECURITY_RESTRICTED_OPERATION contexts must not queue an
2295 : : * action that would run here, because that would bypass the sandbox.
2296 : : * Since closing cursors could queue trigger actions, triggers could open
2297 : : * cursors, etc, we have to keep looping until there's nothing left to do.
2298 : : */
2299 : : for (;;)
2300 : : {
2301 : : /*
2302 : : * Fire all currently pending deferred triggers.
2303 : : */
7750 2304 : 397920 : AfterTriggerFireDeferred();
2305 : :
2306 : : /*
2307 : : * Close open portals (converting holdable ones into static portals).
2308 : : * If there weren't any, we are done ... otherwise loop back to check
2309 : : * if they queued deferred triggers. Lather, rinse, repeat.
2310 : : */
5602 2311 [ + + ]: 397780 : if (!PreCommit_Portals(false))
7750 2312 : 390005 : break;
2313 : : }
2314 : :
2315 : : /*
2316 : : * The remaining actions cannot call any user-defined code, so it's safe
2317 : : * to start shutting down within-transaction services. But note that most
2318 : : * of this stuff could still throw an error, which would switch us into
2319 : : * the transaction-abort path.
2320 : : */
2321 : :
2059 noah@leadboat.com 2322 [ + + ]: 390005 : CallXactCallbacks(is_parallel_worker ? XACT_EVENT_PARALLEL_PRE_COMMIT
2323 : : : XACT_EVENT_PRE_COMMIT);
2324 : :
2325 : : /*
2326 : : * If this xact has started any unfinished parallel operation, clean up
2327 : : * its workers, warning about leaked resources. (But we don't actually
2328 : : * reset parallelModeLevel till entering TRANS_COMMIT, a bit below. This
2329 : : * keeps parallel mode restrictions active as long as possible in a
2330 : : * parallel worker.)
2331 : : */
824 tgl@sss.pgh.pa.us 2332 : 390005 : AtEOXact_Parallel(true);
2333 [ + + ]: 390005 : if (is_parallel_worker)
2334 : : {
2335 [ - + ]: 2003 : if (s->parallelModeLevel != 1)
824 tgl@sss.pgh.pa.us 2336 [ # # ]:UBC 0 : elog(WARNING, "parallelModeLevel is %d not 1 at end of parallel worker transaction",
2337 : : s->parallelModeLevel);
2338 : : }
2339 : : else
2340 : : {
824 tgl@sss.pgh.pa.us 2341 [ - + ]:CBC 388002 : if (s->parallelModeLevel != 0)
824 tgl@sss.pgh.pa.us 2342 [ # # ]:UBC 0 : elog(WARNING, "parallelModeLevel is %d not 0 at end of transaction",
2343 : : s->parallelModeLevel);
2344 : : }
2345 : :
2346 : : /* Shut down the deferred-trigger manager */
5602 tgl@sss.pgh.pa.us 2347 :CBC 390005 : AfterTriggerEndXact(true);
2348 : :
2349 : : /*
2350 : : * Let ON COMMIT management do its thing (must happen after closing
2351 : : * cursors, to avoid dangling-reference problems)
2352 : : */
7914 2353 : 390005 : PreCommit_on_commit_actions();
2354 : :
2355 : : /*
2356 : : * Synchronize files that are created and not WAL-logged during this
2357 : : * transaction. This must happen before AtEOXact_RelationMap(), so that we
2358 : : * don't see committed-but-broken files after a crash.
2359 : : */
2278 noah@leadboat.com 2360 : 390001 : smgrDoPendingSyncs(true, is_parallel_worker);
2361 : :
2362 : : /* close large objects before lower-level cleanup */
8007 tgl@sss.pgh.pa.us 2363 : 390001 : AtEOXact_LargeObject(true);
2364 : :
2365 : : /*
2366 : : * Insert notifications sent by NOTIFY commands into the queue. This
2367 : : * should be late in the pre-commit sequence to minimize time spent
2368 : : * holding the notify-insertion lock. However, this could result in
2369 : : * creating a snapshot, so we must do it before serializable cleanup.
2370 : : */
2410 2371 : 390001 : PreCommit_Notify();
2372 : :
2373 : : /*
2374 : : * Mark serializable transaction as complete for predicate locking
2375 : : * purposes. This should be done as late as we can put it and still allow
2376 : : * errors to be raised for failure patterns found at commit. This is not
2377 : : * appropriate in a parallel worker however, because we aren't committing
2378 : : * the leader's transaction and its serializable state will live on.
2379 : : */
2664 tmunro@postgresql.or 2380 [ + + ]: 390001 : if (!is_parallel_worker)
2381 : 387998 : PreCommit_CheckForSerializationFailure();
2382 : :
2383 : : /* Prevent cancel/die interrupt while cleaning up */
7914 tgl@sss.pgh.pa.us 2384 : 389846 : HOLD_INTERRUPTS();
2385 : :
2386 : : /* Commit updates to the relation map --- do this as late as possible */
2881 pg@bowt.ie 2387 : 389846 : AtEOXact_RelationMap(true, is_parallel_worker);
2388 : :
2389 : : /*
2390 : : * set the current transaction state information appropriately during
2391 : : * commit processing
2392 : : */
7914 tgl@sss.pgh.pa.us 2393 : 389846 : s->state = TRANS_COMMIT;
3910 rhaas@postgresql.org 2394 : 389846 : s->parallelModeLevel = 0;
824 tgl@sss.pgh.pa.us 2395 : 389846 : s->parallelChildXact = false; /* should be false already */
2396 : :
2397 : : /* Disable transaction timeout */
865 akorotkov@postgresql 2398 [ + + ]: 389846 : if (TransactionTimeout > 0)
2399 : 1 : disable_timeout(TRANSACTION_TIMEOUT, false);
2400 : :
4079 rhaas@postgresql.org 2401 [ + + ]: 389846 : if (!is_parallel_worker)
2402 : : {
2403 : : /*
2404 : : * We need to mark our XIDs as committed in pg_xact. This is where we
2405 : : * durably commit.
2406 : : */
2407 : 387843 : latestXid = RecordTransactionCommit();
2408 : : }
2409 : : else
2410 : : {
2411 : : /*
2412 : : * We must not mark our XID committed; the parallel leader is
2413 : : * responsible for that.
2414 : : */
2415 : 2003 : latestXid = InvalidTransactionId;
2416 : :
2417 : : /*
2418 : : * Make sure the leader will know about any WAL we wrote before it
2419 : : * commits.
2420 : : */
2421 : 2003 : ParallelWorkerReportLastRecEnd(XactLastRecEnd);
2422 : : }
2423 : :
2424 : : TRACE_POSTGRESQL_TRANSACTION_COMMIT(MyProc->vxid.lxid);
2425 : :
2426 : : /*
2427 : : * Let others know about no transaction in progress by me. Note that this
2428 : : * must be done _before_ releasing locks we hold and _after_
2429 : : * RecordTransactionCommit.
2430 : : */
6870 tgl@sss.pgh.pa.us 2431 : 389846 : ProcArrayEndTransaction(MyProc, latestXid);
2432 : :
2433 : : /*
2434 : : * This is all post-commit cleanup. Note that if an error is raised here,
2435 : : * it's too late to abort the transaction. This should be just
2436 : : * noncritical resource releasing.
2437 : : *
2438 : : * The ordering of operations is not entirely random. The idea is:
2439 : : * release resources visible to other backends (eg, files, buffer pins);
2440 : : * then release locks; then release backend-local resources. We want to
2441 : : * release locks at the point where any backend waiting for us will see
2442 : : * our transaction as being fully cleaned up.
2443 : : *
2444 : : * Resources that can be associated with individual queries are handled by
2445 : : * the ResourceOwner mechanism. The other calls here are for backend-wide
2446 : : * state.
2447 : : */
2448 : :
4079 rhaas@postgresql.org 2449 : 389846 : CallXactCallbacks(is_parallel_worker ? XACT_EVENT_PARALLEL_COMMIT
2450 : : : XACT_EVENT_COMMIT);
2451 : :
958 heikki.linnakangas@i 2452 : 389846 : CurrentResourceOwner = NULL;
8018 tgl@sss.pgh.pa.us 2453 : 389846 : ResourceOwnerRelease(TopTransactionResourceOwner,
2454 : : RESOURCE_RELEASE_BEFORE_LOCKS,
2455 : : true, true);
2456 : :
470 andres@anarazel.de 2457 : 389846 : AtEOXact_Aio(true);
2458 : :
2459 : : /* Check we've released all buffer pins */
7927 tgl@sss.pgh.pa.us 2460 : 389846 : AtEOXact_Buffers(true);
2461 : :
2462 : : /* Clean up the relation cache */
7631 2463 : 389846 : AtEOXact_RelationCache(true);
2464 : :
2465 : : /* Clean up the type cache */
614 akorotkov@postgresql 2466 : 389846 : AtEOXact_TypeCache();
2467 : :
2468 : : /*
2469 : : * Make catalog changes visible to all backends. This has to happen after
2470 : : * relcache references are dropped (see comments for
2471 : : * AtEOXact_RelationCache), but before locks are released (if anyone is
2472 : : * waiting for lock on a relation we've modified, we want them to know
2473 : : * about the catalog change before they start using the relation).
2474 : : */
8018 tgl@sss.pgh.pa.us 2475 : 389846 : AtEOXact_Inval(true);
2476 : :
7733 2477 : 389846 : AtEOXact_MultiXact();
2478 : :
8018 2479 : 389846 : ResourceOwnerRelease(TopTransactionResourceOwner,
2480 : : RESOURCE_RELEASE_LOCKS,
2481 : : true, true);
2482 : 389846 : ResourceOwnerRelease(TopTransactionResourceOwner,
2483 : : RESOURCE_RELEASE_AFTER_LOCKS,
2484 : : true, true);
2485 : :
2486 : : /*
2487 : : * Likewise, dropping of files deleted during the transaction is best done
2488 : : * after releasing relcache and buffer pins. (This is not strictly
2489 : : * necessary during commit, since such pins should have been released
2490 : : * already, but this ordering is definitely critical during abort.) Since
2491 : : * this may take many seconds, also delay until after releasing locks.
2492 : : * Other backends will observe the attendant catalog changes and not
2493 : : * attempt to access affected files.
2494 : : */
5129 rhaas@postgresql.org 2495 : 389846 : smgrDoPendingDeletes(true);
2496 : :
2497 : : /*
2498 : : * Send out notification signals to other backends (and do other
2499 : : * post-commit NOTIFY cleanup). This must not happen until after our
2500 : : * transaction is fully done from the viewpoint of other backends.
2501 : : */
5978 tgl@sss.pgh.pa.us 2502 : 389846 : AtCommit_Notify();
2503 : :
2504 : : /*
2505 : : * Everything after this should be purely internal-to-this-backend
2506 : : * cleanup.
2507 : : */
6875 2508 : 389846 : AtEOXact_GUC(true, 1);
8246 mail@joeconway.com 2509 : 389846 : AtEOXact_SPI(true);
2821 tmunro@postgresql.or 2510 : 389846 : AtEOXact_Enum();
7957 tgl@sss.pgh.pa.us 2511 : 389846 : AtEOXact_on_commit_actions(true);
4079 rhaas@postgresql.org 2512 : 389846 : AtEOXact_Namespace(true, is_parallel_worker);
5004 tgl@sss.pgh.pa.us 2513 : 389846 : AtEOXact_SMgr();
2985 2514 : 389846 : AtEOXact_Files(true);
7081 2515 : 389846 : AtEOXact_ComboCid();
7005 2516 : 389846 : AtEOXact_HashTables(true);
1 amitlan@postgresql.o 2517 :GNC 389846 : AtEOXact_RI(true);
2638 akapila@postgresql.o 2518 :CBC 389846 : AtEOXact_PgStat(true, is_parallel_worker);
3372 simon@2ndQuadrant.co 2519 : 389846 : AtEOXact_Snapshot(true, false);
3347 peter_e@gmx.net 2520 : 389846 : AtEOXact_ApplyLauncher(true);
1271 tgl@sss.pgh.pa.us 2521 : 389846 : AtEOXact_LogicalRepWorkers(true);
189 msawada@postgresql.o 2522 :GNC 389846 : AtEOXact_LogicalCtl();
6867 tgl@sss.pgh.pa.us 2523 :CBC 389846 : pgstat_report_xact_timestamp(0);
2524 : :
8018 2525 : 389846 : ResourceOwnerDelete(TopTransactionResourceOwner);
2526 : 389846 : s->curTransactionOwner = NULL;
2527 : 389846 : CurTransactionResourceOwner = NULL;
2528 : 389846 : TopTransactionResourceOwner = NULL;
2529 : :
8652 2530 : 389846 : AtCommit_Memory();
2531 : :
2651 tmunro@postgresql.or 2532 : 389846 : s->fullTransactionId = InvalidFullTransactionId;
7957 tgl@sss.pgh.pa.us 2533 : 389846 : s->subTransactionId = InvalidSubTransactionId;
8034 2534 : 389846 : s->nestingLevel = 0;
6875 2535 : 389846 : s->gucNestLevel = 0;
6679 2536 : 389846 : s->childXids = NULL;
2537 : 389846 : s->nChildXids = 0;
2538 : 389846 : s->maxChildXids = 0;
2539 : :
2651 tmunro@postgresql.or 2540 : 389846 : XactTopFullTransactionId = InvalidFullTransactionId;
4079 rhaas@postgresql.org 2541 : 389846 : nParallelCurrentXids = 0;
2542 : :
2543 : : /*
2544 : : * done with commit processing, set current transaction state back to
2545 : : * default
2546 : : */
10523 bruce@momjian.us 2547 : 389846 : s->state = TRANS_DEFAULT;
2548 : :
9293 tgl@sss.pgh.pa.us 2549 [ - + ]: 389846 : RESUME_INTERRUPTS();
10948 scrappy@hub.org 2550 : 389846 : }
2551 : :
2552 : :
2553 : : /*
2554 : : * PrepareTransaction
2555 : : *
2556 : : * NB: if you change this routine, better look at CommitTransaction too!
2557 : : */
2558 : : static void
7683 tgl@sss.pgh.pa.us 2559 : 371 : PrepareTransaction(void)
2560 : : {
7563 bruce@momjian.us 2561 : 371 : TransactionState s = CurrentTransactionState;
358 michael@paquier.xyz 2562 :GNC 371 : FullTransactionId fxid = GetCurrentFullTransactionId();
2563 : : GlobalTransaction gxact;
2564 : : TimestampTz prepared_at;
2565 : :
4079 rhaas@postgresql.org 2566 [ - + ]:CBC 371 : Assert(!IsInParallelMode());
2567 : :
7683 tgl@sss.pgh.pa.us 2568 : 371 : ShowTransactionState("PrepareTransaction");
2569 : :
2570 : : /*
2571 : : * check the current transaction state
2572 : : */
2573 [ - + ]: 371 : if (s->state != TRANS_INPROGRESS)
7683 tgl@sss.pgh.pa.us 2574 [ # # ]:UBC 0 : elog(WARNING, "PrepareTransaction while in %s state",
2575 : : TransStateAsString(s->state));
7683 tgl@sss.pgh.pa.us 2576 [ + - ]:CBC 371 : Assert(s->parent == NULL);
2577 : :
2578 : : /*
2579 : : * Do pre-commit processing that involves calling user-defined code, such
2580 : : * as triggers. Since closing cursors could queue trigger actions,
2581 : : * triggers could open cursors, etc, we have to keep looping until there's
2582 : : * nothing left to do.
2583 : : */
2584 : : for (;;)
2585 : : {
2586 : : /*
2587 : : * Fire all currently pending deferred triggers.
2588 : : */
2589 : 373 : AfterTriggerFireDeferred();
2590 : :
2591 : : /*
2592 : : * Close open portals (converting holdable ones into static portals).
2593 : : * If there weren't any, we are done ... otherwise loop back to check
2594 : : * if they queued deferred triggers. Lather, rinse, repeat.
2595 : : */
5602 2596 [ + + ]: 373 : if (!PreCommit_Portals(true))
7683 2597 : 371 : break;
2598 : : }
2599 : :
4884 2600 : 371 : CallXactCallbacks(XACT_EVENT_PRE_PREPARE);
2601 : :
2602 : : /*
2603 : : * The remaining actions cannot call any user-defined code, so it's safe
2604 : : * to start shutting down within-transaction services. But note that most
2605 : : * of this stuff could still throw an error, which would switch us into
2606 : : * the transaction-abort path.
2607 : : */
2608 : :
2609 : : /* Shut down the deferred-trigger manager */
5602 2610 : 370 : AfterTriggerEndXact(true);
2611 : :
2612 : : /*
2613 : : * Let ON COMMIT management do its thing (must happen after closing
2614 : : * cursors, to avoid dangling-reference problems)
2615 : : */
7683 2616 : 370 : PreCommit_on_commit_actions();
2617 : :
2618 : : /*
2619 : : * Synchronize files that are created and not WAL-logged during this
2620 : : * transaction. This must happen before EndPrepare(), so that we don't see
2621 : : * committed-but-broken files after a crash and COMMIT PREPARED.
2622 : : */
2278 noah@leadboat.com 2623 : 370 : smgrDoPendingSyncs(true, false);
2624 : :
2625 : : /* close large objects before lower-level cleanup */
7683 tgl@sss.pgh.pa.us 2626 : 370 : AtEOXact_LargeObject(true);
2627 : :
2628 : : /* NOTIFY requires no work at this point */
2629 : :
2630 : : /*
2631 : : * Mark serializable transaction as complete for predicate locking
2632 : : * purposes. This should be done as late as we can put it and still allow
2633 : : * errors to be raised for failure patterns found at commit.
2634 : : */
5622 heikki.linnakangas@i 2635 : 370 : PreCommit_CheckForSerializationFailure();
2636 : :
2637 : : /*
2638 : : * Don't allow PREPARE TRANSACTION if we've accessed a temporary table in
2639 : : * this transaction. Having the prepared xact hold locks on another
2640 : : * backend's temp table seems a bad idea --- for instance it would prevent
2641 : : * the backend from exiting. There are other problems too, such as how to
2642 : : * clean up the source backend's local buffers and ON COMMIT state if the
2643 : : * prepared xact includes a DROP of a temp table.
2644 : : *
2645 : : * Other objects types, like functions, operators or extensions, share the
2646 : : * same restriction as they should not be created, locked or dropped as
2647 : : * this can mess up with this session or even a follow-up session trying
2648 : : * to use the same temporary namespace.
2649 : : *
2650 : : * We must check this after executing any ON COMMIT actions, because they
2651 : : * might still access a temp relation.
2652 : : *
2653 : : * XXX In principle this could be relaxed to allow some useful special
2654 : : * cases, such as a temp table created and dropped all within the
2655 : : * transaction. That seems to require much more bookkeeping though.
2656 : : */
2720 michael@paquier.xyz 2657 [ + + ]: 370 : if ((MyXactFlags & XACT_FLAGS_ACCESSEDTEMPNAMESPACE))
2658 [ + - ]: 45 : ereport(ERROR,
2659 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
2660 : : errmsg("cannot PREPARE a transaction that has operated on temporary objects")));
2661 : :
2662 : : /*
2663 : : * Likewise, don't allow PREPARE after pg_export_snapshot. This could be
2664 : : * supported if we added cleanup logic to twophase.c, but for now it
2665 : : * doesn't seem worth the trouble.
2666 : : */
5365 tgl@sss.pgh.pa.us 2667 [ - + ]: 325 : if (XactHasExportedSnapshots())
5365 tgl@sss.pgh.pa.us 2668 [ # # ]:UBC 0 : ereport(ERROR,
2669 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
2670 : : errmsg("cannot PREPARE a transaction that has exported snapshots")));
2671 : :
2672 : : /* Prevent cancel/die interrupt while cleaning up */
7683 tgl@sss.pgh.pa.us 2673 :CBC 325 : HOLD_INTERRUPTS();
2674 : :
2675 : : /*
2676 : : * set the current transaction state information appropriately during
2677 : : * prepare processing
2678 : : */
2679 : 325 : s->state = TRANS_PREPARE;
2680 : :
2681 : : /* Disable transaction timeout */
865 akorotkov@postgresql 2682 [ - + ]: 325 : if (TransactionTimeout > 0)
865 akorotkov@postgresql 2683 :UBC 0 : disable_timeout(TRANSACTION_TIMEOUT, false);
2684 : :
7671 tgl@sss.pgh.pa.us 2685 :CBC 325 : prepared_at = GetCurrentTimestamp();
2686 : :
2687 : : /*
2688 : : * Reserve the GID for this transaction. This could fail if the requested
2689 : : * GID is invalid or already in use.
2690 : : */
358 michael@paquier.xyz 2691 :GNC 325 : gxact = MarkAsPreparing(fxid, prepareGID, prepared_at,
2692 : : GetUserId(), MyDatabaseId);
7683 tgl@sss.pgh.pa.us 2693 :CBC 307 : prepareGID = NULL;
2694 : :
2695 : : /*
2696 : : * Collect data for the 2PC state file. Note that in general, no actual
2697 : : * state change should happen in the called modules during this step,
2698 : : * since it's still possible to fail before commit, and in that case we
2699 : : * want transaction abort to be able to clean up. (In particular, the
2700 : : * AtPrepare routines may error out if they find cases they cannot
2701 : : * handle.) State cleanup should happen in the PostPrepare routines
2702 : : * below. However, some modules can go ahead and clear state here because
2703 : : * they wouldn't do anything with it during abort anyway.
2704 : : *
2705 : : * Note: because the 2PC state file records will be replayed in the same
2706 : : * order they are made, the order of these calls has to match the order in
2707 : : * which we want things to happen during COMMIT PREPARED or ROLLBACK
2708 : : * PREPARED; in particular, pay attention to whether things should happen
2709 : : * before or after releasing the transaction's locks.
2710 : : */
2711 : 307 : StartPrepare(gxact);
2712 : :
2713 : 307 : AtPrepare_Notify();
2714 : 307 : AtPrepare_Locks();
5622 heikki.linnakangas@i 2715 : 305 : AtPrepare_PredicateLocks();
6974 tgl@sss.pgh.pa.us 2716 : 305 : AtPrepare_PgStat();
6063 heikki.linnakangas@i 2717 : 305 : AtPrepare_MultiXact();
5987 tgl@sss.pgh.pa.us 2718 : 305 : AtPrepare_RelationMap();
2719 : :
2720 : : /*
2721 : : * Here is where we really truly prepare.
2722 : : *
2723 : : * We have to record transaction prepares even if we didn't make any
2724 : : * updates, because the transaction manager might get confused if we lose
2725 : : * a global transaction.
2726 : : */
7683 2727 : 305 : EndPrepare(gxact);
2728 : :
2729 : : /*
2730 : : * Now we clean up backend-internal state and release internal resources.
2731 : : */
2732 : :
2733 : : /* Reset XactLastRecEnd until the next transaction writes something */
5119 heikki.linnakangas@i 2734 : 305 : XactLastRecEnd = 0;
2735 : :
2736 : : /*
2737 : : * Transfer our locks to a dummy PGPROC. This has to be done before
2738 : : * ProcArrayClearTransaction(). Otherwise, a GetLockConflicts() would
2739 : : * conclude "xact already committed or aborted" for our locks.
2740 : : */
358 michael@paquier.xyz 2741 :GNC 305 : PostPrepare_Locks(fxid);
2742 : :
2743 : : /*
2744 : : * Let others know about no transaction in progress by me. This has to be
2745 : : * done *after* the prepared transaction has been marked valid, else
2746 : : * someone may think it is unlocked and recyclable.
2747 : : */
6870 tgl@sss.pgh.pa.us 2748 :CBC 305 : ProcArrayClearTransaction(MyProc);
2749 : :
2750 : : /*
2751 : : * In normal commit-processing, this is all non-critical post-transaction
2752 : : * cleanup. When the transaction is prepared, however, it's important
2753 : : * that the locks and other per-backend resources are transferred to the
2754 : : * prepared transaction's PGPROC entry. Note that if an error is raised
2755 : : * here, it's too late to abort the transaction. XXX: This probably should
2756 : : * be in a critical section, to force a PANIC if any of this fails, but
2757 : : * that cure could be worse than the disease.
2758 : : */
2759 : :
7683 2760 : 305 : CallXactCallbacks(XACT_EVENT_PREPARE);
2761 : :
2762 : 305 : ResourceOwnerRelease(TopTransactionResourceOwner,
2763 : : RESOURCE_RELEASE_BEFORE_LOCKS,
2764 : : true, true);
2765 : :
470 andres@anarazel.de 2766 : 305 : AtEOXact_Aio(true);
2767 : :
2768 : : /* Check we've released all buffer pins */
7683 tgl@sss.pgh.pa.us 2769 : 305 : AtEOXact_Buffers(true);
2770 : :
2771 : : /* Clean up the relation cache */
7631 2772 : 305 : AtEOXact_RelationCache(true);
2773 : :
2774 : : /* Clean up the type cache */
614 akorotkov@postgresql 2775 : 305 : AtEOXact_TypeCache();
2776 : :
2777 : : /* notify doesn't need a postprepare call */
2778 : :
6974 tgl@sss.pgh.pa.us 2779 : 305 : PostPrepare_PgStat();
2780 : :
7683 2781 : 305 : PostPrepare_Inval();
2782 : :
2783 : 305 : PostPrepare_smgr();
2784 : :
358 michael@paquier.xyz 2785 :GNC 305 : PostPrepare_MultiXact(fxid);
2786 : :
2787 : 305 : PostPrepare_PredicateLocks(fxid);
2788 : :
7683 tgl@sss.pgh.pa.us 2789 :CBC 305 : ResourceOwnerRelease(TopTransactionResourceOwner,
2790 : : RESOURCE_RELEASE_LOCKS,
2791 : : true, true);
2792 : 305 : ResourceOwnerRelease(TopTransactionResourceOwner,
2793 : : RESOURCE_RELEASE_AFTER_LOCKS,
2794 : : true, true);
2795 : :
2796 : : /*
2797 : : * Allow another backend to finish the transaction. After
2798 : : * PostPrepare_Twophase(), the transaction is completely detached from our
2799 : : * backend. The rest is just non-critical cleanup of backend-local state.
2800 : : */
4429 heikki.linnakangas@i 2801 : 305 : PostPrepare_Twophase();
2802 : :
2803 : : /* PREPARE acts the same as COMMIT as far as GUC is concerned */
6875 tgl@sss.pgh.pa.us 2804 : 305 : AtEOXact_GUC(true, 1);
7683 2805 : 305 : AtEOXact_SPI(true);
2821 tmunro@postgresql.or 2806 : 305 : AtEOXact_Enum();
7683 tgl@sss.pgh.pa.us 2807 : 305 : AtEOXact_on_commit_actions(true);
4079 rhaas@postgresql.org 2808 : 305 : AtEOXact_Namespace(true, false);
5004 tgl@sss.pgh.pa.us 2809 : 305 : AtEOXact_SMgr();
2985 2810 : 305 : AtEOXact_Files(true);
7081 2811 : 305 : AtEOXact_ComboCid();
7005 2812 : 305 : AtEOXact_HashTables(true);
1 amitlan@postgresql.o 2813 :GNC 305 : AtEOXact_RI(true);
2814 : : /* don't call AtEOXact_PgStat here; we fixed pgstat state above */
3372 simon@2ndQuadrant.co 2815 :CBC 305 : AtEOXact_Snapshot(true, true);
2816 : : /* we treat PREPARE as ROLLBACK so far as waking workers goes */
1271 tgl@sss.pgh.pa.us 2817 : 305 : AtEOXact_ApplyLauncher(false);
2818 : 305 : AtEOXact_LogicalRepWorkers(false);
189 msawada@postgresql.o 2819 :GNC 305 : AtEOXact_LogicalCtl();
4450 tgl@sss.pgh.pa.us 2820 :CBC 305 : pgstat_report_xact_timestamp(0);
2821 : :
7683 2822 : 305 : CurrentResourceOwner = NULL;
2823 : 305 : ResourceOwnerDelete(TopTransactionResourceOwner);
2824 : 305 : s->curTransactionOwner = NULL;
2825 : 305 : CurTransactionResourceOwner = NULL;
2826 : 305 : TopTransactionResourceOwner = NULL;
2827 : :
2828 : 305 : AtCommit_Memory();
2829 : :
2651 tmunro@postgresql.or 2830 : 305 : s->fullTransactionId = InvalidFullTransactionId;
7683 tgl@sss.pgh.pa.us 2831 : 305 : s->subTransactionId = InvalidSubTransactionId;
2832 : 305 : s->nestingLevel = 0;
6875 2833 : 305 : s->gucNestLevel = 0;
6679 2834 : 305 : s->childXids = NULL;
2835 : 305 : s->nChildXids = 0;
2836 : 305 : s->maxChildXids = 0;
2837 : :
2651 tmunro@postgresql.or 2838 : 305 : XactTopFullTransactionId = InvalidFullTransactionId;
4079 rhaas@postgresql.org 2839 : 305 : nParallelCurrentXids = 0;
2840 : :
2841 : : /*
2842 : : * done with 1st phase commit processing, set current transaction state
2843 : : * back to default
2844 : : */
7683 tgl@sss.pgh.pa.us 2845 : 305 : s->state = TRANS_DEFAULT;
2846 : :
2847 [ - + ]: 305 : RESUME_INTERRUPTS();
2848 : 305 : }
2849 : :
2850 : :
2851 : : /*
2852 : : * AbortTransaction
2853 : : */
2854 : : static void
9380 2855 : 35305 : AbortTransaction(void)
2856 : : {
10523 bruce@momjian.us 2857 : 35305 : TransactionState s = CurrentTransactionState;
2858 : : TransactionId latestXid;
2859 : : bool is_parallel_worker;
2860 : :
2861 : : /* Prevent cancel/die interrupt while cleaning up */
9293 tgl@sss.pgh.pa.us 2862 : 35305 : HOLD_INTERRUPTS();
2863 : :
2864 : : /* Disable transaction timeout */
865 akorotkov@postgresql 2865 [ + + ]: 35305 : if (TransactionTimeout > 0)
2866 : 1 : disable_timeout(TRANSACTION_TIMEOUT, false);
2867 : :
2868 : : /* Make sure we have a valid memory context and resource owner */
7159 tgl@sss.pgh.pa.us 2869 : 35305 : AtAbort_Memory();
2870 : 35305 : AtAbort_ResourceOwner();
2871 : :
2872 : : /*
2873 : : * Release any LW locks we might be holding as quickly as possible.
2874 : : * (Regular locks, however, must be held till we finish aborting.)
2875 : : * Releasing LW locks is critical since we might try to grab them again
2876 : : * while cleaning up!
2877 : : */
9040 2878 : 35305 : LWLockReleaseAll();
2879 : :
2880 : : /*
2881 : : * Cleanup waiting for LSN if any.
2882 : : */
237 akorotkov@postgresql 2883 :GNC 35305 : WaitLSNCleanup();
2884 : :
2885 : : /* Clear wait information and command progress indicator */
3764 rhaas@postgresql.org 2886 :CBC 35305 : pgstat_report_wait_end();
2887 : 35305 : pgstat_progress_end_command();
2888 : :
470 andres@anarazel.de 2889 : 35305 : pgaio_error_cleanup();
2890 : :
2891 : : /* Clean up buffer content locks, too */
9325 tgl@sss.pgh.pa.us 2892 : 35305 : UnlockBuffers();
2893 : :
2894 : : /* Reset WAL record construction state */
4240 heikki.linnakangas@i 2895 : 35305 : XLogResetInsertion();
2896 : :
2897 : : /* Cancel condition variable sleep */
3507 rhaas@postgresql.org 2898 : 35305 : ConditionVariableCancelSleep();
2899 : :
2900 : : /*
2901 : : * Also clean up any open wait for lock, since the lock manager will choke
2902 : : * if we try to wait for another lock before doing this.
2903 : : */
5186 2904 : 35305 : LockErrorCleanup();
2905 : :
2906 : : /*
2907 : : * If any timeout events are still active, make sure the timeout interrupt
2908 : : * is scheduled. This covers possible loss of a timeout interrupt due to
2909 : : * longjmp'ing out of the SIGINT handler (see notes in handle_sig_alarm).
2910 : : * We delay this till after LockErrorCleanup so that we don't uselessly
2911 : : * reschedule lock or deadlock check timeouts.
2912 : : */
4596 tgl@sss.pgh.pa.us 2913 : 35305 : reschedule_timeouts();
2914 : :
2915 : : /*
2916 : : * Re-enable signals, in case we got here by longjmp'ing out of a signal
2917 : : * handler. We do this fairly early in the sequence so that the timeout
2918 : : * infrastructure will be functional if needed while aborting.
2919 : : */
1243 tmunro@postgresql.or 2920 : 35305 : sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
2921 : :
2922 : : /*
2923 : : * check the current transaction state
2924 : : */
4079 rhaas@postgresql.org 2925 : 35305 : is_parallel_worker = (s->blockState == TBLOCK_PARALLEL_INPROGRESS);
7683 tgl@sss.pgh.pa.us 2926 [ + + - + ]: 35305 : if (s->state != TRANS_INPROGRESS && s->state != TRANS_PREPARE)
8007 tgl@sss.pgh.pa.us 2927 [ # # ]:UBC 0 : elog(WARNING, "AbortTransaction while in %s state",
2928 : : TransStateAsString(s->state));
8034 tgl@sss.pgh.pa.us 2929 [ - + ]:CBC 35305 : Assert(s->parent == NULL);
2930 : :
2931 : : /*
2932 : : * set the current transaction state information appropriately during the
2933 : : * abort processing
2934 : : */
10523 bruce@momjian.us 2935 : 35305 : s->state = TRANS_ABORT;
2936 : :
2937 : : /*
2938 : : * Reset user ID which might have been changed transiently. We need this
2939 : : * to clean up in case control escaped out of a SECURITY DEFINER function
2940 : : * or other local change of CurrentUserId; therefore, the prior value of
2941 : : * SecurityRestrictionContext also needs to be restored.
2942 : : *
2943 : : * (Note: it is not necessary to restore session authorization or role
2944 : : * settings here because those can only be changed via GUC, and GUC will
2945 : : * take care of rolling them back if need be.)
2946 : : */
6047 tgl@sss.pgh.pa.us 2947 : 35305 : SetUserIdAndSecContext(s->prevUser, s->prevSecContext);
2948 : :
2949 : : /* Forget about any active REINDEX. */
2261 2950 : 35305 : ResetReindexState(s->nestingLevel);
2951 : :
2952 : : /* Reset logical streaming state. */
2152 akapila@postgresql.o 2953 : 35305 : ResetLogicalStreamingState();
2954 : :
2955 : : /* Reset snapshot export state. */
1716 michael@paquier.xyz 2956 : 35305 : SnapBuildResetExportedSnapshotState();
2957 : :
2958 : : /*
2959 : : * If this xact has started any unfinished parallel operation, clean up
2960 : : * its workers and exit parallel mode. Don't warn about leaked resources.
2961 : : */
824 tgl@sss.pgh.pa.us 2962 : 35305 : AtEOXact_Parallel(false);
2963 : 35305 : s->parallelModeLevel = 0;
2964 : 35305 : s->parallelChildXact = false; /* should be false already */
2965 : :
2966 : : /*
2967 : : * do abort processing
2968 : : */
5968 bruce@momjian.us 2969 : 35305 : AfterTriggerEndXact(false); /* 'false' means it's abort */
8460 tgl@sss.pgh.pa.us 2970 : 35305 : AtAbort_Portals();
2278 noah@leadboat.com 2971 : 35305 : smgrDoPendingSyncs(false, is_parallel_worker);
5987 tgl@sss.pgh.pa.us 2972 : 35305 : AtEOXact_LargeObject(false);
10129 2973 : 35305 : AtAbort_Notify();
2881 pg@bowt.ie 2974 : 35305 : AtEOXact_RelationMap(false, is_parallel_worker);
4429 heikki.linnakangas@i 2975 : 35305 : AtAbort_Twophase();
2976 : :
2977 : : /*
2978 : : * Advertise the fact that we aborted in pg_xact (assuming that we got as
2979 : : * far as assigning an XID to advertise). But if we're inside a parallel
2980 : : * worker, skip this; the user backend must be the one to write the abort
2981 : : * record.
2982 : : */
4079 rhaas@postgresql.org 2983 [ + + ]: 35305 : if (!is_parallel_worker)
2984 : 35296 : latestXid = RecordTransactionAbort(false);
2985 : : else
2986 : : {
2987 : 9 : latestXid = InvalidTransactionId;
2988 : :
2989 : : /*
2990 : : * Since the parallel leader won't get our value of XactLastRecEnd in
2991 : : * this case, we nudge WAL-writer ourselves in this case. See related
2992 : : * comments in RecordTransactionAbort for why this matters.
2993 : : */
2994 : 9 : XLogSetAsyncXactLSN(XactLastRecEnd);
2995 : : }
2996 : :
2997 : : TRACE_POSTGRESQL_TRANSACTION_ABORT(MyProc->vxid.lxid);
2998 : :
2999 : : /*
3000 : : * Let others know about no transaction in progress by me. Note that this
3001 : : * must be done _before_ releasing locks we hold and _after_
3002 : : * RecordTransactionAbort.
3003 : : */
6870 tgl@sss.pgh.pa.us 3004 : 35305 : ProcArrayEndTransaction(MyProc, latestXid);
3005 : :
3006 : : /*
3007 : : * Post-abort cleanup. See notes in CommitTransaction() concerning
3008 : : * ordering. We can skip all of it if the transaction failed before
3009 : : * creating a resource owner.
3010 : : */
6001 3011 [ + - ]: 35305 : if (TopTransactionResourceOwner != NULL)
3012 : : {
4079 rhaas@postgresql.org 3013 [ + + ]: 35305 : if (is_parallel_worker)
3014 : 9 : CallXactCallbacks(XACT_EVENT_PARALLEL_ABORT);
3015 : : else
3016 : 35296 : CallXactCallbacks(XACT_EVENT_ABORT);
3017 : :
6001 tgl@sss.pgh.pa.us 3018 : 35305 : ResourceOwnerRelease(TopTransactionResourceOwner,
3019 : : RESOURCE_RELEASE_BEFORE_LOCKS,
3020 : : false, true);
470 andres@anarazel.de 3021 : 35305 : AtEOXact_Aio(false);
6001 tgl@sss.pgh.pa.us 3022 : 35305 : AtEOXact_Buffers(false);
3023 : 35305 : AtEOXact_RelationCache(false);
614 akorotkov@postgresql 3024 : 35305 : AtEOXact_TypeCache();
6001 tgl@sss.pgh.pa.us 3025 : 35305 : AtEOXact_Inval(false);
3026 : 35305 : AtEOXact_MultiXact();
3027 : 35305 : ResourceOwnerRelease(TopTransactionResourceOwner,
3028 : : RESOURCE_RELEASE_LOCKS,
3029 : : false, true);
3030 : 35305 : ResourceOwnerRelease(TopTransactionResourceOwner,
3031 : : RESOURCE_RELEASE_AFTER_LOCKS,
3032 : : false, true);
5129 rhaas@postgresql.org 3033 : 35305 : smgrDoPendingDeletes(false);
3034 : :
6001 tgl@sss.pgh.pa.us 3035 : 35305 : AtEOXact_GUC(false, 1);
3036 : 35305 : AtEOXact_SPI(false);
2821 tmunro@postgresql.or 3037 : 35305 : AtEOXact_Enum();
6001 tgl@sss.pgh.pa.us 3038 : 35305 : AtEOXact_on_commit_actions(false);
4079 rhaas@postgresql.org 3039 : 35305 : AtEOXact_Namespace(false, is_parallel_worker);
5004 tgl@sss.pgh.pa.us 3040 : 35305 : AtEOXact_SMgr();
2985 3041 : 35305 : AtEOXact_Files(false);
6001 3042 : 35305 : AtEOXact_ComboCid();
3043 : 35305 : AtEOXact_HashTables(false);
1 amitlan@postgresql.o 3044 :GNC 35305 : AtEOXact_RI(false);
2638 akapila@postgresql.o 3045 :CBC 35305 : AtEOXact_PgStat(false, is_parallel_worker);
3347 peter_e@gmx.net 3046 : 35305 : AtEOXact_ApplyLauncher(false);
1271 tgl@sss.pgh.pa.us 3047 : 35305 : AtEOXact_LogicalRepWorkers(false);
189 msawada@postgresql.o 3048 :GNC 35305 : AtEOXact_LogicalCtl();
6001 tgl@sss.pgh.pa.us 3049 :CBC 35305 : pgstat_report_xact_timestamp(0);
3050 : : }
3051 : :
3052 : : /*
3053 : : * State remains TRANS_ABORT until CleanupTransaction().
3054 : : */
9293 3055 [ - + ]: 35305 : RESUME_INTERRUPTS();
9498 3056 : 35305 : }
3057 : :
3058 : : /*
3059 : : * CleanupTransaction
3060 : : */
3061 : : static void
9380 3062 : 35305 : CleanupTransaction(void)
3063 : : {
9498 3064 : 35305 : TransactionState s = CurrentTransactionState;
3065 : :
3066 : : /*
3067 : : * State should still be TRANS_ABORT from AbortTransaction().
3068 : : */
3069 [ - + ]: 35305 : if (s->state != TRANS_ABORT)
8004 tgl@sss.pgh.pa.us 3070 [ # # ]:UBC 0 : elog(FATAL, "CleanupTransaction: unexpected state %s",
3071 : : TransStateAsString(s->state));
3072 : :
3073 : : /*
3074 : : * do abort cleanup processing
3075 : : */
8460 tgl@sss.pgh.pa.us 3076 :CBC 35305 : AtCleanup_Portals(); /* now safe to release portal memory */
3296 3077 : 35305 : AtEOXact_Snapshot(false, true); /* and release the transaction's snapshots */
3078 : :
7975 bruce@momjian.us 3079 : 35305 : CurrentResourceOwner = NULL; /* and resource owner */
7957 tgl@sss.pgh.pa.us 3080 [ + - ]: 35305 : if (TopTransactionResourceOwner)
3081 : 35305 : ResourceOwnerDelete(TopTransactionResourceOwner);
8018 3082 : 35305 : s->curTransactionOwner = NULL;
3083 : 35305 : CurTransactionResourceOwner = NULL;
3084 : 35305 : TopTransactionResourceOwner = NULL;
3085 : :
8460 3086 : 35305 : AtCleanup_Memory(); /* and transaction memory */
3087 : :
2651 tmunro@postgresql.or 3088 : 35305 : s->fullTransactionId = InvalidFullTransactionId;
7957 tgl@sss.pgh.pa.us 3089 : 35305 : s->subTransactionId = InvalidSubTransactionId;
8034 3090 : 35305 : s->nestingLevel = 0;
6875 3091 : 35305 : s->gucNestLevel = 0;
6679 3092 : 35305 : s->childXids = NULL;
3093 : 35305 : s->nChildXids = 0;
3094 : 35305 : s->maxChildXids = 0;
4079 rhaas@postgresql.org 3095 : 35305 : s->parallelModeLevel = 0;
824 tgl@sss.pgh.pa.us 3096 : 35305 : s->parallelChildXact = false;
3097 : :
2651 tmunro@postgresql.or 3098 : 35305 : XactTopFullTransactionId = InvalidFullTransactionId;
4079 rhaas@postgresql.org 3099 : 35305 : nParallelCurrentXids = 0;
3100 : :
3101 : : /*
3102 : : * done with abort processing, set current transaction state back to
3103 : : * default
3104 : : */
10523 bruce@momjian.us 3105 : 35305 : s->state = TRANS_DEFAULT;
3106 : 35305 : }
3107 : :
3108 : : /*
3109 : : * StartTransactionCommand
3110 : : */
3111 : : void
8448 tgl@sss.pgh.pa.us 3112 : 524553 : StartTransactionCommand(void)
3113 : : {
10523 bruce@momjian.us 3114 : 524553 : TransactionState s = CurrentTransactionState;
3115 : :
3116 [ + + + - : 524553 : switch (s->blockState)
- ]
3117 : : {
3118 : : /*
3119 : : * if we aren't in a transaction block, we just do our usual start
3120 : : * transaction.
3121 : : */
10522 3122 : 423404 : case TBLOCK_DEFAULT:
3123 : 423404 : StartTransaction();
8121 3124 : 423404 : s->blockState = TBLOCK_STARTED;
3125 : 423404 : break;
3126 : :
3127 : : /*
3128 : : * We are somewhere in a transaction block or subtransaction and
3129 : : * about to start a new command. For now we do nothing, but
3130 : : * someday we may do command-local resource initialization. (Note
3131 : : * that any needed CommandCounterIncrement was done by the
3132 : : * previous CommitTransactionCommand.)
3133 : : */
10522 3134 : 99955 : case TBLOCK_INPROGRESS:
3135 : : case TBLOCK_IMPLICIT_INPROGRESS:
3136 : : case TBLOCK_SUBINPROGRESS:
3137 : 99955 : break;
3138 : :
3139 : : /*
3140 : : * Here we are in a failed transaction block (one of the commands
3141 : : * caused an abort) so we do nothing but remain in the abort
3142 : : * state. Eventually we will get a ROLLBACK command which will
3143 : : * get us out of this state. (It is up to other code to ensure
3144 : : * that no commands other than ROLLBACK will be processed in these
3145 : : * states.)
3146 : : */
3147 : 1194 : case TBLOCK_ABORT:
3148 : : case TBLOCK_SUBABORT:
3149 : 1194 : break;
3150 : :
3151 : : /* These cases are invalid. */
8034 tgl@sss.pgh.pa.us 3152 :UBC 0 : case TBLOCK_STARTED:
3153 : : case TBLOCK_BEGIN:
3154 : : case TBLOCK_PARALLEL_INPROGRESS:
3155 : : case TBLOCK_SUBBEGIN:
3156 : : case TBLOCK_END:
3157 : : case TBLOCK_SUBRELEASE:
3158 : : case TBLOCK_SUBCOMMIT:
3159 : : case TBLOCK_ABORT_END:
3160 : : case TBLOCK_SUBABORT_END:
3161 : : case TBLOCK_ABORT_PENDING:
3162 : : case TBLOCK_SUBABORT_PENDING:
3163 : : case TBLOCK_SUBRESTART:
3164 : : case TBLOCK_SUBABORT_RESTART:
3165 : : case TBLOCK_PREPARE:
7957 3166 [ # # ]: 0 : elog(ERROR, "StartTransactionCommand: unexpected state %s",
3167 : : BlockStateAsString(s->blockState));
3168 : : break;
3169 : : }
3170 : :
3171 : : /*
3172 : : * We must switch to CurTransactionContext before returning. This is
3173 : : * already done if we called StartTransaction, otherwise not.
3174 : : */
8034 tgl@sss.pgh.pa.us 3175 [ - + ]:CBC 524553 : Assert(CurTransactionContext != NULL);
3176 : 524553 : MemoryContextSwitchTo(CurTransactionContext);
10523 bruce@momjian.us 3177 : 524553 : }
3178 : :
3179 : :
3180 : : /*
3181 : : * Simple system for saving and restoring transaction characteristics
3182 : : * (isolation level, read only, deferrable). We need this for transaction
3183 : : * chaining, so that we can set the characteristics of the new transaction to
3184 : : * be the same as the previous one. (We need something like this because the
3185 : : * GUC system resets the characteristics at transaction end, so for example
3186 : : * just skipping the reset in StartTransaction() won't work.)
3187 : : */
3188 : : void
1583 tgl@sss.pgh.pa.us 3189 : 491792 : SaveTransactionCharacteristics(SavedTransactionCharacteristics *s)
3190 : : {
3191 : 491792 : s->save_XactIsoLevel = XactIsoLevel;
3192 : 491792 : s->save_XactReadOnly = XactReadOnly;
3193 : 491792 : s->save_XactDeferrable = XactDeferrable;
2655 peter@eisentraut.org 3194 : 491792 : }
3195 : :
3196 : : void
1583 tgl@sss.pgh.pa.us 3197 : 44 : RestoreTransactionCharacteristics(const SavedTransactionCharacteristics *s)
3198 : : {
3199 : 44 : XactIsoLevel = s->save_XactIsoLevel;
3200 : 44 : XactReadOnly = s->save_XactReadOnly;
3201 : 44 : XactDeferrable = s->save_XactDeferrable;
2655 peter@eisentraut.org 3202 : 44 : }
3203 : :
3204 : : /*
3205 : : * CommitTransactionCommand -- a wrapper function handling the
3206 : : * loop over subtransactions to avoid a potentially dangerous recursion
3207 : : * in CommitTransactionCommandInternal().
3208 : : */
3209 : : void
8448 tgl@sss.pgh.pa.us 3210 : 491497 : CommitTransactionCommand(void)
3211 : : {
3212 : : /*
3213 : : * Repeatedly call CommitTransactionCommandInternal() until all the work
3214 : : * is done.
3215 : : */
803 akorotkov@postgresql 3216 [ + + ]: 491788 : while (!CommitTransactionCommandInternal())
3217 : : {
3218 : : }
844 3219 : 491132 : }
3220 : :
3221 : : /*
3222 : : * CommitTransactionCommandInternal - a function doing an iteration of work
3223 : : * regarding handling the commit transaction command. In the case of
3224 : : * subtransactions more than one iterations could be required. Returns
3225 : : * true when no more iterations required, false otherwise.
3226 : : */
3227 : : static bool
3228 : 491788 : CommitTransactionCommandInternal(void)
3229 : : {
10523 bruce@momjian.us 3230 : 491788 : TransactionState s = CurrentTransactionState;
3231 : : SavedTransactionCharacteristics savetc;
3232 : :
3233 : : /* Must save in case we need to restore below */
1583 tgl@sss.pgh.pa.us 3234 : 491788 : SaveTransactionCharacteristics(&savetc);
3235 : :
10523 bruce@momjian.us 3236 [ - + + + : 491788 : switch (s->blockState)
+ + + + +
+ + + + +
+ + - ]
3237 : : {
3238 : : /*
3239 : : * These shouldn't happen. TBLOCK_DEFAULT means the previous
3240 : : * StartTransactionCommand didn't set the STARTED state
3241 : : * appropriately, while TBLOCK_PARALLEL_INPROGRESS should be ended
3242 : : * by EndParallelWorkerTransaction(), not this function.
3243 : : */
8121 bruce@momjian.us 3244 :UBC 0 : case TBLOCK_DEFAULT:
3245 : : case TBLOCK_PARALLEL_INPROGRESS:
8008 tgl@sss.pgh.pa.us 3246 [ # # ]: 0 : elog(FATAL, "CommitTransactionCommand: unexpected state %s",
3247 : : BlockStateAsString(s->blockState));
3248 : : break;
3249 : :
3250 : : /*
3251 : : * If we aren't in a transaction block, just do our usual
3252 : : * transaction commit, and return to the idle state.
3253 : : */
8121 bruce@momjian.us 3254 :CBC 378890 : case TBLOCK_STARTED:
8448 tgl@sss.pgh.pa.us 3255 : 378890 : CommitTransaction();
8121 bruce@momjian.us 3256 : 378847 : s->blockState = TBLOCK_DEFAULT;
10522 3257 : 378847 : break;
3258 : :
3259 : : /*
3260 : : * We are completing a "BEGIN TRANSACTION" command, so we change
3261 : : * to the "transaction block in progress" state and return. (We
3262 : : * assume the BEGIN did nothing to the database, so we need no
3263 : : * CommandCounterIncrement.)
3264 : : */
3265 : 12546 : case TBLOCK_BEGIN:
3266 : 12546 : s->blockState = TBLOCK_INPROGRESS;
3267 : 12546 : break;
3268 : :
3269 : : /*
3270 : : * This is the case when we have finished executing a command
3271 : : * someplace within a transaction block. We increment the command
3272 : : * counter and return.
3273 : : */
3274 : 74860 : case TBLOCK_INPROGRESS:
3275 : : case TBLOCK_IMPLICIT_INPROGRESS:
3276 : : case TBLOCK_SUBINPROGRESS:
3277 : 74860 : CommandCounterIncrement();
3278 : 74860 : break;
3279 : :
3280 : : /*
3281 : : * We are completing a "COMMIT" command. Do it and return to the
3282 : : * idle state.
3283 : : */
3284 : 8850 : case TBLOCK_END:
3285 : 8850 : CommitTransaction();
9498 tgl@sss.pgh.pa.us 3286 : 8611 : s->blockState = TBLOCK_DEFAULT;
2655 peter@eisentraut.org 3287 [ + + ]: 8611 : if (s->chain)
3288 : : {
3289 : 8 : StartTransaction();
3290 : 8 : s->blockState = TBLOCK_INPROGRESS;
3291 : 8 : s->chain = false;
1583 tgl@sss.pgh.pa.us 3292 : 8 : RestoreTransactionCharacteristics(&savetc);
3293 : : }
10522 bruce@momjian.us 3294 : 8611 : break;
3295 : :
3296 : : /*
3297 : : * Here we are in the middle of a transaction block but one of the
3298 : : * commands caused an abort so we do nothing but remain in the
3299 : : * abort state. Eventually we will get a ROLLBACK command.
3300 : : */
3301 : 12 : case TBLOCK_ABORT:
3302 : : case TBLOCK_SUBABORT:
3303 : 12 : break;
3304 : :
3305 : : /*
3306 : : * Here we were in an aborted transaction block and we just got
3307 : : * the ROLLBACK command from the user, so clean up the
3308 : : * already-aborted transaction and return to the idle state.
3309 : : */
7957 tgl@sss.pgh.pa.us 3310 : 919 : case TBLOCK_ABORT_END:
9498 3311 : 919 : CleanupTransaction();
10522 bruce@momjian.us 3312 : 919 : s->blockState = TBLOCK_DEFAULT;
2655 peter@eisentraut.org 3313 [ + + ]: 919 : if (s->chain)
3314 : : {
3315 : 8 : StartTransaction();
3316 : 8 : s->blockState = TBLOCK_INPROGRESS;
3317 : 8 : s->chain = false;
1583 tgl@sss.pgh.pa.us 3318 : 8 : RestoreTransactionCharacteristics(&savetc);
3319 : : }
10522 bruce@momjian.us 3320 : 919 : break;
3321 : :
3322 : : /*
3323 : : * Here we were in a perfectly good transaction block but the user
3324 : : * told us to ROLLBACK anyway. We have to abort the transaction
3325 : : * and then clean up.
3326 : : */
7957 tgl@sss.pgh.pa.us 3327 : 1786 : case TBLOCK_ABORT_PENDING:
3328 : 1786 : AbortTransaction();
3329 : 1786 : CleanupTransaction();
3330 : 1786 : s->blockState = TBLOCK_DEFAULT;
2655 peter@eisentraut.org 3331 [ + + ]: 1786 : if (s->chain)
3332 : : {
3333 : 12 : StartTransaction();
3334 : 12 : s->blockState = TBLOCK_INPROGRESS;
3335 : 12 : s->chain = false;
1583 tgl@sss.pgh.pa.us 3336 : 12 : RestoreTransactionCharacteristics(&savetc);
3337 : : }
8008 3338 : 1786 : break;
3339 : :
3340 : : /*
3341 : : * We are completing a "PREPARE TRANSACTION" command. Do it and
3342 : : * return to the idle state.
3343 : : */
7683 3344 : 259 : case TBLOCK_PREPARE:
3345 : 259 : PrepareTransaction();
3346 : 195 : s->blockState = TBLOCK_DEFAULT;
3347 : 195 : break;
3348 : :
3349 : : /*
3350 : : * The user issued a SAVEPOINT inside a transaction block. Start a
3351 : : * subtransaction. (DefineSavepoint already did PushTransaction,
3352 : : * so as to have someplace to put the SUBBEGIN state.)
3353 : : */
8034 3354 : 12200 : case TBLOCK_SUBBEGIN:
3355 : 12200 : StartSubTransaction();
3356 : 12200 : s->blockState = TBLOCK_SUBINPROGRESS;
3357 : 12200 : break;
3358 : :
3359 : : /*
3360 : : * The user issued a RELEASE command, so we end the current
3361 : : * subtransaction and return to the parent transaction. The parent
3362 : : * might be ended too, so repeat till we find an INPROGRESS
3363 : : * transaction or subtransaction.
3364 : : */
5460 simon@2ndQuadrant.co 3365 : 266 : case TBLOCK_SUBRELEASE:
3366 : : do
3367 : : {
5410 3368 : 266 : CommitSubTransaction();
7975 bruce@momjian.us 3369 : 266 : s = CurrentTransactionState; /* changed by pop */
5460 simon@2ndQuadrant.co 3370 [ + + ]: 266 : } while (s->blockState == TBLOCK_SUBRELEASE);
3371 : :
3372 [ + + - + ]: 179 : Assert(s->blockState == TBLOCK_INPROGRESS ||
3373 : : s->blockState == TBLOCK_SUBINPROGRESS);
3374 : 179 : break;
3375 : :
3376 : : /*
3377 : : * The user issued a COMMIT, so we end the current subtransaction
3378 : : * hierarchy and perform final commit. We do this by rolling up
3379 : : * any subtransactions into their parent, which leads to O(N^2)
3380 : : * operations with respect to resource owners - this isn't that
3381 : : * bad until we approach a thousands of savepoints but is
3382 : : * necessary for correctness should after triggers create new
3383 : : * resource owners.
3384 : : */
3385 : 603 : case TBLOCK_SUBCOMMIT:
3386 : : do
3387 : : {
5410 3388 : 603 : CommitSubTransaction();
5460 3389 : 603 : s = CurrentTransactionState; /* changed by pop */
3390 [ + + ]: 603 : } while (s->blockState == TBLOCK_SUBCOMMIT);
3391 : : /* If we had a COMMIT command, finish off the main xact too */
7963 tgl@sss.pgh.pa.us 3392 [ + + ]: 514 : if (s->blockState == TBLOCK_END)
3393 : : {
3394 [ - + ]: 402 : Assert(s->parent == NULL);
3395 : 402 : CommitTransaction();
3396 : 385 : s->blockState = TBLOCK_DEFAULT;
1957 fujii@postgresql.org 3397 [ + + ]: 385 : if (s->chain)
3398 : : {
3399 : 12 : StartTransaction();
3400 : 12 : s->blockState = TBLOCK_INPROGRESS;
3401 : 12 : s->chain = false;
1583 tgl@sss.pgh.pa.us 3402 : 12 : RestoreTransactionCharacteristics(&savetc);
3403 : : }
3404 : : }
7683 3405 [ + - ]: 112 : else if (s->blockState == TBLOCK_PREPARE)
3406 : : {
3407 [ - + ]: 112 : Assert(s->parent == NULL);
3408 : 112 : PrepareTransaction();
3409 : 110 : s->blockState = TBLOCK_DEFAULT;
3410 : : }
3411 : : else
5460 simon@2ndQuadrant.co 3412 [ # # ]:UBC 0 : elog(ERROR, "CommitTransactionCommand: unexpected state %s",
3413 : : BlockStateAsString(s->blockState));
8034 tgl@sss.pgh.pa.us 3414 :CBC 495 : break;
3415 : :
3416 : : /*
3417 : : * The current already-failed subtransaction is ending due to a
3418 : : * ROLLBACK or ROLLBACK TO command, so pop it and recursively
3419 : : * examine the parent (which could be in any of several states).
3420 : : * As we need to examine the parent, return false to request the
3421 : : * caller to do the next iteration.
3422 : : */
803 akorotkov@postgresql 3423 : 55 : case TBLOCK_SUBABORT_END:
3424 : 55 : CleanupSubTransaction();
3425 : 55 : return false;
3426 : :
3427 : : /*
3428 : : * As above, but it's not dead yet, so abort first.
3429 : : */
3430 : 236 : case TBLOCK_SUBABORT_PENDING:
3431 : 236 : AbortSubTransaction();
3432 : 236 : CleanupSubTransaction();
3433 : 236 : return false;
3434 : :
3435 : : /*
3436 : : * The current subtransaction is the target of a ROLLBACK TO
3437 : : * command. Abort and pop it, then start a new subtransaction
3438 : : * with the same name.
3439 : : */
7957 tgl@sss.pgh.pa.us 3440 : 337 : case TBLOCK_SUBRESTART:
3441 : : {
3442 : : char *name;
3443 : : int savepointLevel;
3444 : :
3445 : : /* save name and keep Cleanup from freeing it */
3446 : 337 : name = s->name;
3447 : 337 : s->name = NULL;
3448 : 337 : savepointLevel = s->savepointLevel;
3449 : :
3450 : 337 : AbortSubTransaction();
3451 : 337 : CleanupSubTransaction();
3452 : :
3453 : 337 : DefineSavepoint(NULL);
3454 : 337 : s = CurrentTransactionState; /* changed by push */
3455 : 337 : s->name = name;
3456 : 337 : s->savepointLevel = savepointLevel;
3457 : :
3458 : : /* This is the same as TBLOCK_SUBBEGIN case */
1341 peter@eisentraut.org 3459 [ - + ]: 337 : Assert(s->blockState == TBLOCK_SUBBEGIN);
8008 tgl@sss.pgh.pa.us 3460 : 337 : StartSubTransaction();
3461 : 337 : s->blockState = TBLOCK_SUBINPROGRESS;
3462 : : }
8034 3463 : 337 : break;
3464 : :
3465 : : /*
3466 : : * Same as above, but the subtransaction had already failed, so we
3467 : : * don't need AbortSubTransaction.
3468 : : */
7957 3469 : 145 : case TBLOCK_SUBABORT_RESTART:
3470 : : {
3471 : : char *name;
3472 : : int savepointLevel;
3473 : :
3474 : : /* save name and keep Cleanup from freeing it */
3475 : 145 : name = s->name;
3476 : 145 : s->name = NULL;
3477 : 145 : savepointLevel = s->savepointLevel;
3478 : :
3479 : 145 : CleanupSubTransaction();
3480 : :
3481 : 145 : DefineSavepoint(NULL);
3482 : 145 : s = CurrentTransactionState; /* changed by push */
3483 : 145 : s->name = name;
3484 : 145 : s->savepointLevel = savepointLevel;
3485 : :
3486 : : /* This is the same as TBLOCK_SUBBEGIN case */
1341 peter@eisentraut.org 3487 [ - + ]: 145 : Assert(s->blockState == TBLOCK_SUBBEGIN);
7957 tgl@sss.pgh.pa.us 3488 : 145 : StartSubTransaction();
3489 : 145 : s->blockState = TBLOCK_SUBINPROGRESS;
3490 : : }
3491 : 145 : break;
3492 : : }
3493 : :
3494 : : /* Done, no more iterations required */
803 akorotkov@postgresql 3495 : 491132 : return true;
3496 : : }
3497 : :
3498 : : /*
3499 : : * AbortCurrentTransaction -- a wrapper function handling the
3500 : : * loop over subtransactions to avoid potentially dangerous recursion in
3501 : : * AbortCurrentTransactionInternal().
3502 : : */
3503 : : void
8034 tgl@sss.pgh.pa.us 3504 : 34219 : AbortCurrentTransaction(void)
3505 : : {
3506 : : /*
3507 : : * Repeatedly call AbortCurrentTransactionInternal() until all the work is
3508 : : * done.
3509 : : */
803 akorotkov@postgresql 3510 [ - + ]: 34219 : while (!AbortCurrentTransactionInternal())
3511 : : {
3512 : : }
844 3513 : 34219 : }
3514 : :
3515 : : /*
3516 : : * AbortCurrentTransactionInternal - a function doing an iteration of work
3517 : : * regarding handling the current transaction abort. In the case of
3518 : : * subtransactions more than one iterations could be required. Returns
3519 : : * true when no more iterations required, false otherwise.
3520 : : */
3521 : : static bool
3522 : 34219 : AbortCurrentTransactionInternal(void)
3523 : : {
8034 tgl@sss.pgh.pa.us 3524 : 34219 : TransactionState s = CurrentTransactionState;
3525 : :
3526 [ + + - + : 34219 : switch (s->blockState)
+ + - - +
+ - - - ]
3527 : : {
8121 bruce@momjian.us 3528 : 51 : case TBLOCK_DEFAULT:
7800 tgl@sss.pgh.pa.us 3529 [ - + ]: 51 : if (s->state == TRANS_DEFAULT)
3530 : : {
3531 : : /* we are idle, so nothing to do */
3532 : : }
3533 : : else
3534 : : {
3535 : : /*
3536 : : * We can get here after an error during transaction start
3537 : : * (state will be TRANS_START). Need to clean up the
3538 : : * incompletely started transaction. First, adjust the
3539 : : * low-level state to suppress warning message from
3540 : : * AbortTransaction.
3541 : : */
7800 tgl@sss.pgh.pa.us 3542 [ # # ]:UBC 0 : if (s->state == TRANS_START)
3543 : 0 : s->state = TRANS_INPROGRESS;
3544 : 0 : AbortTransaction();
3545 : 0 : CleanupTransaction();
3546 : : }
8121 bruce@momjian.us 3547 :CBC 51 : break;
3548 : :
3549 : : /*
3550 : : * If we aren't in a transaction block, we just do the basic abort
3551 : : * & cleanup transaction. For this purpose, we treat an implicit
3552 : : * transaction block as if it were a simple statement.
3553 : : */
3554 : 31666 : case TBLOCK_STARTED:
3555 : : case TBLOCK_IMPLICIT_INPROGRESS:
10522 3556 : 31666 : AbortTransaction();
8448 tgl@sss.pgh.pa.us 3557 : 31666 : CleanupTransaction();
8121 bruce@momjian.us 3558 : 31666 : s->blockState = TBLOCK_DEFAULT;
10522 3559 : 31666 : break;
3560 : :
3561 : : /*
3562 : : * If we are in TBLOCK_BEGIN it means something screwed up right
3563 : : * after reading "BEGIN TRANSACTION". We assume that the user
3564 : : * will interpret the error as meaning the BEGIN failed to get him
3565 : : * into a transaction block, so we should abort and return to idle
3566 : : * state.
3567 : : */
10522 bruce@momjian.us 3568 :UBC 0 : case TBLOCK_BEGIN:
3569 : 0 : AbortTransaction();
7957 tgl@sss.pgh.pa.us 3570 : 0 : CleanupTransaction();
3571 : 0 : s->blockState = TBLOCK_DEFAULT;
10522 bruce@momjian.us 3572 : 0 : break;
3573 : :
3574 : : /*
3575 : : * We are somewhere in a transaction block and we've gotten a
3576 : : * failure, so we abort the transaction and set up the persistent
3577 : : * ABORT state. We will stay in ABORT until we get a ROLLBACK.
3578 : : */
10522 bruce@momjian.us 3579 :CBC 933 : case TBLOCK_INPROGRESS:
3580 : : case TBLOCK_PARALLEL_INPROGRESS:
3581 : 933 : AbortTransaction();
8121 3582 : 933 : s->blockState = TBLOCK_ABORT;
3583 : : /* CleanupTransaction happens when we exit TBLOCK_ABORT_END */
10522 3584 : 933 : break;
3585 : :
3586 : : /*
3587 : : * Here, we failed while trying to COMMIT. Clean up the
3588 : : * transaction and return to idle state (we do not want to stay in
3589 : : * the transaction).
3590 : : */
3591 : 256 : case TBLOCK_END:
3592 : 256 : AbortTransaction();
9498 tgl@sss.pgh.pa.us 3593 : 256 : CleanupTransaction();
8121 bruce@momjian.us 3594 : 256 : s->blockState = TBLOCK_DEFAULT;
10522 3595 : 256 : break;
3596 : :
3597 : : /*
3598 : : * Here, we are already in an aborted transaction state and are
3599 : : * waiting for a ROLLBACK, but for some reason we failed again! So
3600 : : * we just remain in the abort state.
3601 : : */
3602 : 64 : case TBLOCK_ABORT:
3603 : : case TBLOCK_SUBABORT:
3604 : 64 : break;
3605 : :
3606 : : /*
3607 : : * We are in a failed transaction and we got the ROLLBACK command.
3608 : : * We have already aborted, we just need to cleanup and go to idle
3609 : : * state.
3610 : : */
7957 tgl@sss.pgh.pa.us 3611 :UBC 0 : case TBLOCK_ABORT_END:
9498 3612 : 0 : CleanupTransaction();
10522 bruce@momjian.us 3613 : 0 : s->blockState = TBLOCK_DEFAULT;
3614 : 0 : break;
3615 : :
3616 : : /*
3617 : : * We are in a live transaction and we got a ROLLBACK command.
3618 : : * Abort, cleanup, go to idle state.
3619 : : */
7957 tgl@sss.pgh.pa.us 3620 : 0 : case TBLOCK_ABORT_PENDING:
3621 : 0 : AbortTransaction();
3622 : 0 : CleanupTransaction();
3623 : 0 : s->blockState = TBLOCK_DEFAULT;
8034 3624 : 0 : break;
3625 : :
3626 : : /*
3627 : : * Here, we failed while trying to PREPARE. Clean up the
3628 : : * transaction and return to idle state (we do not want to stay in
3629 : : * the transaction).
3630 : : */
7683 tgl@sss.pgh.pa.us 3631 :CBC 64 : case TBLOCK_PREPARE:
3632 : 64 : AbortTransaction();
3633 : 64 : CleanupTransaction();
3634 : 64 : s->blockState = TBLOCK_DEFAULT;
3635 : 64 : break;
3636 : :
3637 : : /*
3638 : : * We got an error inside a subtransaction. Abort just the
3639 : : * subtransaction, and go to the persistent SUBABORT state until
3640 : : * we get ROLLBACK.
3641 : : */
8034 3642 : 1185 : case TBLOCK_SUBINPROGRESS:
3643 : 1185 : AbortSubTransaction();
3644 : 1185 : s->blockState = TBLOCK_SUBABORT;
3645 : 1185 : break;
3646 : :
3647 : : /*
3648 : : * If we failed while trying to create a subtransaction, clean up
3649 : : * the broken subtransaction and abort the parent. The same
3650 : : * applies if we get a failure while ending a subtransaction. As
3651 : : * we need to abort the parent, return false to request the caller
3652 : : * to do the next iteration.
3653 : : */
803 akorotkov@postgresql 3654 :UBC 0 : case TBLOCK_SUBBEGIN:
3655 : : case TBLOCK_SUBRELEASE:
3656 : : case TBLOCK_SUBCOMMIT:
3657 : : case TBLOCK_SUBABORT_PENDING:
3658 : : case TBLOCK_SUBRESTART:
3659 : 0 : AbortSubTransaction();
3660 : 0 : CleanupSubTransaction();
3661 : 0 : return false;
3662 : :
3663 : : /*
3664 : : * Same as above, except the Abort() was already done.
3665 : : */
3666 : 0 : case TBLOCK_SUBABORT_END:
3667 : : case TBLOCK_SUBABORT_RESTART:
3668 : 0 : CleanupSubTransaction();
3669 : 0 : return false;
3670 : : }
3671 : :
3672 : : /* Done, no more iterations required */
803 akorotkov@postgresql 3673 :CBC 34219 : return true;
3674 : : }
3675 : :
3676 : : /*
3677 : : * PreventInTransactionBlock
3678 : : *
3679 : : * This routine is to be called by statements that must not run inside
3680 : : * a transaction block, typically because they have non-rollback-able
3681 : : * side effects or do internal commits.
3682 : : *
3683 : : * If this routine completes successfully, then the calling statement is
3684 : : * guaranteed that if it completes without error, its results will be
3685 : : * committed immediately.
3686 : : *
3687 : : * If we have already started a transaction block, issue an error; also issue
3688 : : * an error if we appear to be running inside a user-defined function (which
3689 : : * could issue more commands and possibly cause a failure after the statement
3690 : : * completes). Subtransactions are verboten too.
3691 : : *
3692 : : * We must also set XACT_FLAGS_NEEDIMMEDIATECOMMIT in MyXactFlags, to ensure
3693 : : * that postgres.c follows through by committing after the statement is done.
3694 : : *
3695 : : * isTopLevel: passed down from ProcessUtility to determine whether we are
3696 : : * inside a function. (We will always fail if this is false, but it's
3697 : : * convenient to centralize the check here instead of making callers do it.)
3698 : : * stmtType: statement type name, for error messages.
3699 : : */
3700 : : void
3056 peter_e@gmx.net 3701 : 9327 : PreventInTransactionBlock(bool isTopLevel, const char *stmtType)
3702 : : {
3703 : : /*
3704 : : * xact block already started?
3705 : : */
8653 tgl@sss.pgh.pa.us 3706 [ + + ]: 9327 : if (IsTransactionBlock())
8380 3707 [ + - ]: 78 : ereport(ERROR,
3708 : : (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
3709 : : /* translator: %s represents an SQL statement name */
3710 : : errmsg("%s cannot run inside a transaction block",
3711 : : stmtType)));
3712 : :
3713 : : /*
3714 : : * subtransaction?
3715 : : */
8034 3716 [ - + ]: 9249 : if (IsSubTransaction())
8034 tgl@sss.pgh.pa.us 3717 [ # # ]:UBC 0 : ereport(ERROR,
3718 : : (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
3719 : : /* translator: %s represents an SQL statement name */
3720 : : errmsg("%s cannot run inside a subtransaction",
3721 : : stmtType)));
3722 : :
3723 : : /*
3724 : : * inside a function call?
3725 : : */
7049 tgl@sss.pgh.pa.us 3726 [ + + ]:CBC 9249 : if (!isTopLevel)
8380 3727 [ + - ]: 4 : ereport(ERROR,
3728 : : (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
3729 : : /* translator: %s represents an SQL statement name */
3730 : : errmsg("%s cannot be executed from a function or procedure",
3731 : : stmtType)));
3732 : :
3733 : : /* If we got past IsTransactionBlock test, should be in default state */
8121 bruce@momjian.us 3734 [ + + ]: 9245 : if (CurrentTransactionState->blockState != TBLOCK_DEFAULT &&
8074 tgl@sss.pgh.pa.us 3735 [ - + ]: 8269 : CurrentTransactionState->blockState != TBLOCK_STARTED)
8034 tgl@sss.pgh.pa.us 3736 [ # # ]:UBC 0 : elog(FATAL, "cannot prevent transaction chain");
3737 : :
3738 : : /* All okay. Set the flag to make sure the right thing happens later. */
1435 tgl@sss.pgh.pa.us 3739 :CBC 9245 : MyXactFlags |= XACT_FLAGS_NEEDIMMEDIATECOMMIT;
8653 3740 : 9245 : }
3741 : :
3742 : : /*
3743 : : * WarnNoTransactionBlock
3744 : : * RequireTransactionBlock
3745 : : *
3746 : : * These two functions allow for warnings or errors if a command is executed
3747 : : * outside of a transaction block. This is useful for commands that have no
3748 : : * effects that persist past transaction end (and so calling them outside a
3749 : : * transaction block is presumably an error). DECLARE CURSOR is an example.
3750 : : * While top-level transaction control commands (BEGIN/COMMIT/ABORT) and SET
3751 : : * that have no effect issue warnings, all other no-effect commands generate
3752 : : * errors.
3753 : : *
3754 : : * If we appear to be running inside a user-defined function, we do not
3755 : : * issue anything, since the function could issue more commands that make
3756 : : * use of the current statement's results. Likewise subtransactions.
3757 : : * Thus these are inverses for PreventInTransactionBlock.
3758 : : *
3759 : : * isTopLevel: passed down from ProcessUtility to determine whether we are
3760 : : * inside a function.
3761 : : * stmtType: statement type name, for warning or error messages.
3762 : : */
3763 : : void
3056 peter_e@gmx.net 3764 : 1401 : WarnNoTransactionBlock(bool isTopLevel, const char *stmtType)
3765 : : {
3766 : 1401 : CheckTransactionBlock(isTopLevel, false, stmtType);
4600 bruce@momjian.us 3767 : 1401 : }
3768 : :
3769 : : void
3056 peter_e@gmx.net 3770 : 5025 : RequireTransactionBlock(bool isTopLevel, const char *stmtType)
3771 : : {
3772 : 5025 : CheckTransactionBlock(isTopLevel, true, stmtType);
4600 bruce@momjian.us 3773 : 5002 : }
3774 : :
3775 : : /*
3776 : : * This is the implementation of the above two.
3777 : : */
3778 : : static void
3056 peter_e@gmx.net 3779 : 6426 : CheckTransactionBlock(bool isTopLevel, bool throwError, const char *stmtType)
3780 : : {
3781 : : /*
3782 : : * xact block already started?
3783 : : */
8625 tgl@sss.pgh.pa.us 3784 [ + + ]: 6426 : if (IsTransactionBlock())
3785 : 6307 : return;
3786 : :
3787 : : /*
3788 : : * subtransaction?
3789 : : */
8034 3790 [ - + ]: 119 : if (IsSubTransaction())
8034 tgl@sss.pgh.pa.us 3791 :UBC 0 : return;
3792 : :
3793 : : /*
3794 : : * inside a function call?
3795 : : */
7049 tgl@sss.pgh.pa.us 3796 [ + + ]:CBC 119 : if (!isTopLevel)
8625 3797 : 82 : return;
3798 : :
4600 bruce@momjian.us 3799 [ + + + - ]: 37 : ereport(throwError ? ERROR : WARNING,
3800 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
3801 : : /* translator: %s represents an SQL statement name */
3802 : : errmsg("%s can only be used in transaction blocks",
3803 : : stmtType)));
3804 : : }
3805 : :
3806 : : /*
3807 : : * IsInTransactionBlock
3808 : : *
3809 : : * This routine is for statements that need to behave differently inside
3810 : : * a transaction block than when running as single commands. ANALYZE is
3811 : : * currently the only example.
3812 : : *
3813 : : * If this routine returns "false", then the calling statement is allowed
3814 : : * to perform internal transaction-commit-and-start cycles; there is not a
3815 : : * risk of messing up any transaction already in progress. (Note that this
3816 : : * is not the identical guarantee provided by PreventInTransactionBlock,
3817 : : * since we will not force a post-statement commit.)
3818 : : *
3819 : : * isTopLevel: passed down from ProcessUtility to determine whether we are
3820 : : * inside a function.
3821 : : */
3822 : : bool
3056 peter_e@gmx.net 3823 : 3565 : IsInTransactionBlock(bool isTopLevel)
3824 : : {
3825 : : /*
3826 : : * Return true on same conditions that would make
3827 : : * PreventInTransactionBlock error out
3828 : : */
8074 tgl@sss.pgh.pa.us 3829 [ + + ]: 3565 : if (IsTransactionBlock())
3830 : 98 : return true;
3831 : :
8034 3832 [ - + ]: 3467 : if (IsSubTransaction())
8034 tgl@sss.pgh.pa.us 3833 :UBC 0 : return true;
3834 : :
7049 tgl@sss.pgh.pa.us 3835 [ + + ]:CBC 3467 : if (!isTopLevel)
8074 3836 : 74 : return true;
3837 : :
3838 [ + - ]: 3393 : if (CurrentTransactionState->blockState != TBLOCK_DEFAULT &&
3839 [ - + ]: 3393 : CurrentTransactionState->blockState != TBLOCK_STARTED)
8074 tgl@sss.pgh.pa.us 3840 :UBC 0 : return true;
3841 : :
8074 tgl@sss.pgh.pa.us 3842 :CBC 3393 : return false;
3843 : : }
3844 : :
3845 : :
3846 : : /*
3847 : : * Register or deregister callback functions for start- and end-of-xact
3848 : : * operations.
3849 : : *
3850 : : * These functions are intended for use by dynamically loaded modules.
3851 : : * For built-in modules we generally just hardwire the appropriate calls
3852 : : * (mainly because it's easier to control the order that way, where needed).
3853 : : *
3854 : : * At transaction end, the callback occurs post-commit or post-abort, so the
3855 : : * callback functions can only do noncritical cleanup.
3856 : : */
3857 : : void
8003 3858 : 2324 : RegisterXactCallback(XactCallback callback, void *arg)
3859 : : {
3860 : : XactCallbackItem *item;
3861 : :
3862 : : item = (XactCallbackItem *)
3863 : 2324 : MemoryContextAlloc(TopMemoryContext, sizeof(XactCallbackItem));
8311 3864 : 2324 : item->callback = callback;
3865 : 2324 : item->arg = arg;
8003 3866 : 2324 : item->next = Xact_callbacks;
3867 : 2324 : Xact_callbacks = item;
8311 3868 : 2324 : }
3869 : :
3870 : : void
8003 tgl@sss.pgh.pa.us 3871 :UBC 0 : UnregisterXactCallback(XactCallback callback, void *arg)
3872 : : {
3873 : : XactCallbackItem *item;
3874 : : XactCallbackItem *prev;
3875 : :
8311 3876 : 0 : prev = NULL;
8003 3877 [ # # ]: 0 : for (item = Xact_callbacks; item; prev = item, item = item->next)
3878 : : {
8311 3879 [ # # # # ]: 0 : if (item->callback == callback && item->arg == arg)
3880 : : {
3881 [ # # ]: 0 : if (prev)
3882 : 0 : prev->next = item->next;
3883 : : else
8003 3884 : 0 : Xact_callbacks = item->next;
8311 3885 : 0 : pfree(item);
3886 : 0 : break;
3887 : : }
3888 : : }
3889 : 0 : }
3890 : :
3891 : : static void
7957 tgl@sss.pgh.pa.us 3892 :CBC 815832 : CallXactCallbacks(XactEvent event)
3893 : : {
3894 : : XactCallbackItem *item;
3895 : : XactCallbackItem *next;
3896 : :
1371 3897 [ + + ]: 1029328 : for (item = Xact_callbacks; item; item = next)
3898 : : {
3899 : : /* allow callbacks to unregister themselves when called */
3900 : 213497 : next = item->next;
3218 peter_e@gmx.net 3901 : 213497 : item->callback(event, item->arg);
3902 : : }
7957 tgl@sss.pgh.pa.us 3903 : 815831 : }
3904 : :
3905 : :
3906 : : /*
3907 : : * Register or deregister callback functions for start- and end-of-subxact
3908 : : * operations.
3909 : : *
3910 : : * Pretty much same as above, but for subtransaction events.
3911 : : *
3912 : : * At subtransaction end, the callback occurs post-subcommit or post-subabort,
3913 : : * so the callback functions can only do noncritical cleanup. At
3914 : : * subtransaction start, the callback is called when the subtransaction has
3915 : : * finished initializing.
3916 : : */
3917 : : void
3918 : 2324 : RegisterSubXactCallback(SubXactCallback callback, void *arg)
3919 : : {
3920 : : SubXactCallbackItem *item;
3921 : :
3922 : : item = (SubXactCallbackItem *)
3923 : 2324 : MemoryContextAlloc(TopMemoryContext, sizeof(SubXactCallbackItem));
3924 : 2324 : item->callback = callback;
3925 : 2324 : item->arg = arg;
3926 : 2324 : item->next = SubXact_callbacks;
3927 : 2324 : SubXact_callbacks = item;
3928 : 2324 : }
3929 : :
3930 : : void
7957 tgl@sss.pgh.pa.us 3931 :UBC 0 : UnregisterSubXactCallback(SubXactCallback callback, void *arg)
3932 : : {
3933 : : SubXactCallbackItem *item;
3934 : : SubXactCallbackItem *prev;
3935 : :
3936 : 0 : prev = NULL;
3937 [ # # ]: 0 : for (item = SubXact_callbacks; item; prev = item, item = item->next)
3938 : : {
3939 [ # # # # ]: 0 : if (item->callback == callback && item->arg == arg)
3940 : : {
3941 [ # # ]: 0 : if (prev)
3942 : 0 : prev->next = item->next;
3943 : : else
3944 : 0 : SubXact_callbacks = item->next;
3945 : 0 : pfree(item);
3946 : 0 : break;
3947 : : }
3948 : : }
3949 : 0 : }
3950 : :
3951 : : static void
7957 tgl@sss.pgh.pa.us 3952 :CBC 32626 : CallSubXactCallbacks(SubXactEvent event,
3953 : : SubTransactionId mySubid,
3954 : : SubTransactionId parentSubid)
3955 : : {
3956 : : SubXactCallbackItem *item;
3957 : : SubXactCallbackItem *next;
3958 : :
1371 3959 [ + + ]: 59312 : for (item = SubXact_callbacks; item; item = next)
3960 : : {
3961 : : /* allow callbacks to unregister themselves when called */
3962 : 26686 : next = item->next;
3218 peter_e@gmx.net 3963 : 26686 : item->callback(event, mySubid, parentSubid, item->arg);
3964 : : }
8311 tgl@sss.pgh.pa.us 3965 : 32626 : }
3966 : :
3967 : :
3968 : : /* ----------------------------------------------------------------
3969 : : * transaction block support
3970 : : * ----------------------------------------------------------------
3971 : : */
3972 : :
3973 : : /*
3974 : : * BeginTransactionBlock
3975 : : * This executes a BEGIN command.
3976 : : */
3977 : : void
10523 bruce@momjian.us 3978 : 12546 : BeginTransactionBlock(void)
3979 : : {
3980 : 12546 : TransactionState s = CurrentTransactionState;
3981 : :
8008 tgl@sss.pgh.pa.us 3982 [ + + - - : 12546 : switch (s->blockState)
- ]
3983 : : {
3984 : : /*
3985 : : * We are not inside a transaction block, so allow one to begin.
3986 : : */
8121 bruce@momjian.us 3987 : 12045 : case TBLOCK_STARTED:
3988 : 12045 : s->blockState = TBLOCK_BEGIN;
3989 : 12045 : break;
3990 : :
3991 : : /*
3992 : : * BEGIN converts an implicit transaction block to a regular one.
3993 : : * (Note that we allow this even if we've already done some
3994 : : * commands, which is a bit odd but matches historical practice.)
3995 : : */
3218 tgl@sss.pgh.pa.us 3996 : 501 : case TBLOCK_IMPLICIT_INPROGRESS:
3997 : 501 : s->blockState = TBLOCK_BEGIN;
3998 : 501 : break;
3999 : :
4000 : : /*
4001 : : * Already a transaction block in progress.
4002 : : */
8121 bruce@momjian.us 4003 :UBC 0 : case TBLOCK_INPROGRESS:
4004 : : case TBLOCK_PARALLEL_INPROGRESS:
4005 : : case TBLOCK_SUBINPROGRESS:
4006 : : case TBLOCK_ABORT:
4007 : : case TBLOCK_SUBABORT:
8008 tgl@sss.pgh.pa.us 4008 [ # # ]: 0 : ereport(WARNING,
4009 : : (errcode(ERRCODE_ACTIVE_SQL_TRANSACTION),
4010 : : errmsg("there is already a transaction in progress")));
8121 bruce@momjian.us 4011 : 0 : break;
4012 : :
4013 : : /* These cases are invalid. */
4014 : 0 : case TBLOCK_DEFAULT:
4015 : : case TBLOCK_BEGIN:
4016 : : case TBLOCK_SUBBEGIN:
4017 : : case TBLOCK_END:
4018 : : case TBLOCK_SUBRELEASE:
4019 : : case TBLOCK_SUBCOMMIT:
4020 : : case TBLOCK_ABORT_END:
4021 : : case TBLOCK_SUBABORT_END:
4022 : : case TBLOCK_ABORT_PENDING:
4023 : : case TBLOCK_SUBABORT_PENDING:
4024 : : case TBLOCK_SUBRESTART:
4025 : : case TBLOCK_SUBABORT_RESTART:
4026 : : case TBLOCK_PREPARE:
8034 tgl@sss.pgh.pa.us 4027 [ # # ]: 0 : elog(FATAL, "BeginTransactionBlock: unexpected state %s",
4028 : : BlockStateAsString(s->blockState));
4029 : : break;
4030 : : }
10948 scrappy@hub.org 4031 :CBC 12546 : }
4032 : :
4033 : : /*
4034 : : * PrepareTransactionBlock
4035 : : * This executes a PREPARE command.
4036 : : *
4037 : : * Since PREPARE may actually do a ROLLBACK, the result indicates what
4038 : : * happened: true for PREPARE, false for ROLLBACK.
4039 : : *
4040 : : * Note that we don't actually do anything here except change blockState.
4041 : : * The real work will be done in the upcoming PrepareTransaction().
4042 : : * We do it this way because it's not convenient to change memory context,
4043 : : * resource owner, etc while executing inside a Portal.
4044 : : */
4045 : : bool
3164 peter_e@gmx.net 4046 : 373 : PrepareTransactionBlock(const char *gid)
4047 : : {
4048 : : TransactionState s;
4049 : : bool result;
4050 : :
4051 : : /* Set up to commit the current transaction */
2655 peter@eisentraut.org 4052 : 373 : result = EndTransactionBlock(false);
4053 : :
4054 : : /* If successful, change outer tblock state to PREPARE */
7683 tgl@sss.pgh.pa.us 4055 [ + + ]: 373 : if (result)
4056 : : {
4057 : 371 : s = CurrentTransactionState;
4058 : :
4059 [ + + ]: 493 : while (s->parent != NULL)
4060 : 122 : s = s->parent;
4061 : :
4062 [ + - ]: 371 : if (s->blockState == TBLOCK_END)
4063 : : {
4064 : : /* Save GID where PrepareTransaction can find it again */
4065 : 371 : prepareGID = MemoryContextStrdup(TopTransactionContext, gid);
4066 : :
4067 : 371 : s->blockState = TBLOCK_PREPARE;
4068 : : }
4069 : : else
4070 : : {
4071 : : /*
4072 : : * ignore case where we are not in a transaction;
4073 : : * EndTransactionBlock already issued a warning.
4074 : : */
3218 tgl@sss.pgh.pa.us 4075 [ # # # # ]:UBC 0 : Assert(s->blockState == TBLOCK_STARTED ||
4076 : : s->blockState == TBLOCK_IMPLICIT_INPROGRESS);
4077 : : /* Don't send back a PREPARE result tag... */
7683 4078 : 0 : result = false;
4079 : : }
4080 : : }
4081 : :
7683 tgl@sss.pgh.pa.us 4082 :CBC 373 : return result;
4083 : : }
4084 : :
4085 : : /*
4086 : : * EndTransactionBlock
4087 : : * This executes a COMMIT command.
4088 : : *
4089 : : * Since COMMIT may actually do a ROLLBACK, the result indicates what
4090 : : * happened: true for COMMIT, false for ROLLBACK.
4091 : : *
4092 : : * Note that we don't actually do anything here except change blockState.
4093 : : * The real work will be done in the upcoming CommitTransactionCommand().
4094 : : * We do it this way because it's not convenient to change memory context,
4095 : : * resource owner, etc while executing inside a Portal.
4096 : : */
4097 : : bool
2655 peter@eisentraut.org 4098 : 10165 : EndTransactionBlock(bool chain)
4099 : : {
10523 bruce@momjian.us 4100 : 10165 : TransactionState s = CurrentTransactionState;
8008 tgl@sss.pgh.pa.us 4101 : 10165 : bool result = false;
4102 : :
4103 [ + + + + : 10165 : switch (s->blockState)
+ + - -
- ]
4104 : : {
4105 : : /*
4106 : : * We are in a transaction block, so tell CommitTransactionCommand
4107 : : * to COMMIT.
4108 : : */
8121 bruce@momjian.us 4109 : 9093 : case TBLOCK_INPROGRESS:
8008 tgl@sss.pgh.pa.us 4110 : 9093 : s->blockState = TBLOCK_END;
4111 : 9093 : result = true;
8034 4112 : 9093 : break;
4113 : :
4114 : : /*
4115 : : * We are in an implicit transaction block. If AND CHAIN was
4116 : : * specified, error. Otherwise commit, but issue a warning
4117 : : * because there was no explicit BEGIN before this.
4118 : : */
3218 4119 : 32 : case TBLOCK_IMPLICIT_INPROGRESS:
2487 peter@eisentraut.org 4120 [ + + ]: 32 : if (chain)
4121 [ + - ]: 16 : ereport(ERROR,
4122 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4123 : : /* translator: %s represents an SQL statement name */
4124 : : errmsg("%s can only be used in transaction blocks",
4125 : : "COMMIT AND CHAIN")));
4126 : : else
4127 [ + - ]: 16 : ereport(WARNING,
4128 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4129 : : errmsg("there is no transaction in progress")));
3218 tgl@sss.pgh.pa.us 4130 : 16 : s->blockState = TBLOCK_END;
4131 : 16 : result = true;
4132 : 16 : break;
4133 : :
4134 : : /*
4135 : : * We are in a failed transaction block. Tell
4136 : : * CommitTransactionCommand it's time to exit the block.
4137 : : */
8121 bruce@momjian.us 4138 : 472 : case TBLOCK_ABORT:
7957 tgl@sss.pgh.pa.us 4139 : 472 : s->blockState = TBLOCK_ABORT_END;
8121 bruce@momjian.us 4140 : 472 : break;
4141 : :
4142 : : /*
4143 : : * We are in a live subtransaction block. Set up to subcommit all
4144 : : * open subtransactions and then commit the main transaction.
4145 : : */
7963 tgl@sss.pgh.pa.us 4146 : 514 : case TBLOCK_SUBINPROGRESS:
4147 [ + + ]: 1117 : while (s->parent != NULL)
4148 : : {
7957 4149 [ + - ]: 603 : if (s->blockState == TBLOCK_SUBINPROGRESS)
5460 simon@2ndQuadrant.co 4150 : 603 : s->blockState = TBLOCK_SUBCOMMIT;
4151 : : else
7957 tgl@sss.pgh.pa.us 4152 [ # # ]:UBC 0 : elog(FATAL, "EndTransactionBlock: unexpected state %s",
4153 : : BlockStateAsString(s->blockState));
7963 tgl@sss.pgh.pa.us 4154 :CBC 603 : s = s->parent;
4155 : : }
7957 4156 [ + - ]: 514 : if (s->blockState == TBLOCK_INPROGRESS)
4157 : 514 : s->blockState = TBLOCK_END;
4158 : : else
7957 tgl@sss.pgh.pa.us 4159 [ # # ]:UBC 0 : elog(FATAL, "EndTransactionBlock: unexpected state %s",
4160 : : BlockStateAsString(s->blockState));
7963 tgl@sss.pgh.pa.us 4161 :CBC 514 : result = true;
4162 : 514 : break;
4163 : :
4164 : : /*
4165 : : * Here we are inside an aborted subtransaction. Treat the COMMIT
4166 : : * as ROLLBACK: set up to abort everything and exit the main
4167 : : * transaction.
4168 : : */
8034 4169 : 38 : case TBLOCK_SUBABORT:
7957 4170 [ + + ]: 76 : while (s->parent != NULL)
4171 : : {
4172 [ - + ]: 38 : if (s->blockState == TBLOCK_SUBINPROGRESS)
7957 tgl@sss.pgh.pa.us 4173 :UBC 0 : s->blockState = TBLOCK_SUBABORT_PENDING;
7957 tgl@sss.pgh.pa.us 4174 [ + - ]:CBC 38 : else if (s->blockState == TBLOCK_SUBABORT)
4175 : 38 : s->blockState = TBLOCK_SUBABORT_END;
4176 : : else
7957 tgl@sss.pgh.pa.us 4177 [ # # ]:UBC 0 : elog(FATAL, "EndTransactionBlock: unexpected state %s",
4178 : : BlockStateAsString(s->blockState));
7957 tgl@sss.pgh.pa.us 4179 :CBC 38 : s = s->parent;
4180 : : }
4181 [ + - ]: 38 : if (s->blockState == TBLOCK_INPROGRESS)
4182 : 38 : s->blockState = TBLOCK_ABORT_PENDING;
7957 tgl@sss.pgh.pa.us 4183 [ # # ]:UBC 0 : else if (s->blockState == TBLOCK_ABORT)
4184 : 0 : s->blockState = TBLOCK_ABORT_END;
4185 : : else
4186 [ # # ]: 0 : elog(FATAL, "EndTransactionBlock: unexpected state %s",
4187 : : BlockStateAsString(s->blockState));
8034 tgl@sss.pgh.pa.us 4188 :CBC 38 : break;
4189 : :
4190 : : /*
4191 : : * The user issued COMMIT when not inside a transaction. For
4192 : : * COMMIT without CHAIN, issue a WARNING, staying in
4193 : : * TBLOCK_STARTED state. The upcoming call to
4194 : : * CommitTransactionCommand() will then close the transaction and
4195 : : * put us back into the default state. For COMMIT AND CHAIN,
4196 : : * error.
4197 : : */
7957 4198 : 16 : case TBLOCK_STARTED:
2487 peter@eisentraut.org 4199 [ + + ]: 16 : if (chain)
4200 [ + - ]: 4 : ereport(ERROR,
4201 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4202 : : /* translator: %s represents an SQL statement name */
4203 : : errmsg("%s can only be used in transaction blocks",
4204 : : "COMMIT AND CHAIN")));
4205 : : else
4206 [ + - ]: 12 : ereport(WARNING,
4207 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4208 : : errmsg("there is no transaction in progress")));
7913 tgl@sss.pgh.pa.us 4209 : 12 : result = true;
8121 bruce@momjian.us 4210 : 12 : break;
4211 : :
4212 : : /*
4213 : : * The user issued a COMMIT that somehow ran inside a parallel
4214 : : * worker. We can't cope with that.
4215 : : */
4079 rhaas@postgresql.org 4216 :UBC 0 : case TBLOCK_PARALLEL_INPROGRESS:
4217 [ # # ]: 0 : ereport(FATAL,
4218 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
4219 : : errmsg("cannot commit during a parallel operation")));
4220 : : break;
4221 : :
4222 : : /* These cases are invalid. */
8121 bruce@momjian.us 4223 : 0 : case TBLOCK_DEFAULT:
4224 : : case TBLOCK_BEGIN:
4225 : : case TBLOCK_SUBBEGIN:
4226 : : case TBLOCK_END:
4227 : : case TBLOCK_SUBRELEASE:
4228 : : case TBLOCK_SUBCOMMIT:
4229 : : case TBLOCK_ABORT_END:
4230 : : case TBLOCK_SUBABORT_END:
4231 : : case TBLOCK_ABORT_PENDING:
4232 : : case TBLOCK_SUBABORT_PENDING:
4233 : : case TBLOCK_SUBRESTART:
4234 : : case TBLOCK_SUBABORT_RESTART:
4235 : : case TBLOCK_PREPARE:
8034 tgl@sss.pgh.pa.us 4236 [ # # ]: 0 : elog(FATAL, "EndTransactionBlock: unexpected state %s",
4237 : : BlockStateAsString(s->blockState));
4238 : : break;
4239 : : }
4240 : :
2655 peter@eisentraut.org 4241 [ + + + + :CBC 10145 : Assert(s->blockState == TBLOCK_STARTED ||
+ + - + ]
4242 : : s->blockState == TBLOCK_END ||
4243 : : s->blockState == TBLOCK_ABORT_END ||
4244 : : s->blockState == TBLOCK_ABORT_PENDING);
4245 : :
4246 : 10145 : s->chain = chain;
4247 : :
8008 tgl@sss.pgh.pa.us 4248 : 10145 : return result;
4249 : : }
4250 : :
4251 : : /*
4252 : : * UserAbortTransactionBlock
4253 : : * This executes a ROLLBACK command.
4254 : : *
4255 : : * As above, we don't actually do anything here except change blockState.
4256 : : */
4257 : : void
2655 peter@eisentraut.org 4258 : 2215 : UserAbortTransactionBlock(bool chain)
4259 : : {
10523 bruce@momjian.us 4260 : 2215 : TransactionState s = CurrentTransactionState;
4261 : :
8008 tgl@sss.pgh.pa.us 4262 [ + + + + : 2215 : switch (s->blockState)
- - - ]
4263 : : {
4264 : : /*
4265 : : * We are inside a transaction block and we got a ROLLBACK command
4266 : : * from the user, so tell CommitTransactionCommand to abort and
4267 : : * exit the transaction block.
4268 : : */
7957 4269 : 1642 : case TBLOCK_INPROGRESS:
4270 : 1642 : s->blockState = TBLOCK_ABORT_PENDING;
8034 4271 : 1642 : break;
4272 : :
4273 : : /*
4274 : : * We are inside a failed transaction block and we got a ROLLBACK
4275 : : * command from the user. Abort processing is already done, so
4276 : : * CommitTransactionCommand just has to cleanup and go back to
4277 : : * idle state.
4278 : : */
7957 4279 : 447 : case TBLOCK_ABORT:
4280 : 447 : s->blockState = TBLOCK_ABORT_END;
8034 4281 : 447 : break;
4282 : :
4283 : : /*
4284 : : * We are inside a subtransaction. Mark everything up to top
4285 : : * level as exitable.
4286 : : */
4287 : 75 : case TBLOCK_SUBINPROGRESS:
4288 : : case TBLOCK_SUBABORT:
7957 4289 [ + + ]: 290 : while (s->parent != NULL)
4290 : : {
4291 [ + + ]: 215 : if (s->blockState == TBLOCK_SUBINPROGRESS)
4292 : 198 : s->blockState = TBLOCK_SUBABORT_PENDING;
4293 [ + - ]: 17 : else if (s->blockState == TBLOCK_SUBABORT)
4294 : 17 : s->blockState = TBLOCK_SUBABORT_END;
4295 : : else
7957 tgl@sss.pgh.pa.us 4296 [ # # ]:UBC 0 : elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
4297 : : BlockStateAsString(s->blockState));
7957 tgl@sss.pgh.pa.us 4298 :CBC 215 : s = s->parent;
4299 : : }
4300 [ + - ]: 75 : if (s->blockState == TBLOCK_INPROGRESS)
4301 : 75 : s->blockState = TBLOCK_ABORT_PENDING;
7957 tgl@sss.pgh.pa.us 4302 [ # # ]:UBC 0 : else if (s->blockState == TBLOCK_ABORT)
4303 : 0 : s->blockState = TBLOCK_ABORT_END;
4304 : : else
4305 [ # # ]: 0 : elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
4306 : : BlockStateAsString(s->blockState));
8034 tgl@sss.pgh.pa.us 4307 :CBC 75 : break;
4308 : :
4309 : : /*
4310 : : * The user issued ABORT when not inside a transaction. For
4311 : : * ROLLBACK without CHAIN, issue a WARNING and go to abort state.
4312 : : * The upcoming call to CommitTransactionCommand() will then put
4313 : : * us back into the default state. For ROLLBACK AND CHAIN, error.
4314 : : *
4315 : : * We do the same thing with ABORT inside an implicit transaction,
4316 : : * although in this case we might be rolling back actual database
4317 : : * state changes. (It's debatable whether we should issue a
4318 : : * WARNING in this case, but we have done so historically.)
4319 : : */
4320 : 51 : case TBLOCK_STARTED:
4321 : : case TBLOCK_IMPLICIT_INPROGRESS:
2487 peter@eisentraut.org 4322 [ + + ]: 51 : if (chain)
4323 [ + - ]: 20 : ereport(ERROR,
4324 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4325 : : /* translator: %s represents an SQL statement name */
4326 : : errmsg("%s can only be used in transaction blocks",
4327 : : "ROLLBACK AND CHAIN")));
4328 : : else
4329 [ + - ]: 31 : ereport(WARNING,
4330 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4331 : : errmsg("there is no transaction in progress")));
7957 tgl@sss.pgh.pa.us 4332 : 31 : s->blockState = TBLOCK_ABORT_PENDING;
8034 4333 : 31 : break;
4334 : :
4335 : : /*
4336 : : * The user issued an ABORT that somehow ran inside a parallel
4337 : : * worker. We can't cope with that.
4338 : : */
4079 rhaas@postgresql.org 4339 :UBC 0 : case TBLOCK_PARALLEL_INPROGRESS:
4340 [ # # ]: 0 : ereport(FATAL,
4341 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
4342 : : errmsg("cannot abort during a parallel operation")));
4343 : : break;
4344 : :
4345 : : /* These cases are invalid. */
8034 tgl@sss.pgh.pa.us 4346 : 0 : case TBLOCK_DEFAULT:
4347 : : case TBLOCK_BEGIN:
4348 : : case TBLOCK_SUBBEGIN:
4349 : : case TBLOCK_END:
4350 : : case TBLOCK_SUBRELEASE:
4351 : : case TBLOCK_SUBCOMMIT:
4352 : : case TBLOCK_ABORT_END:
4353 : : case TBLOCK_SUBABORT_END:
4354 : : case TBLOCK_ABORT_PENDING:
4355 : : case TBLOCK_SUBABORT_PENDING:
4356 : : case TBLOCK_SUBRESTART:
4357 : : case TBLOCK_SUBABORT_RESTART:
4358 : : case TBLOCK_PREPARE:
4359 [ # # ]: 0 : elog(FATAL, "UserAbortTransactionBlock: unexpected state %s",
4360 : : BlockStateAsString(s->blockState));
4361 : : break;
4362 : : }
4363 : :
2655 peter@eisentraut.org 4364 [ + + - + ]:CBC 2195 : Assert(s->blockState == TBLOCK_ABORT_END ||
4365 : : s->blockState == TBLOCK_ABORT_PENDING);
4366 : :
4367 : 2195 : s->chain = chain;
8008 tgl@sss.pgh.pa.us 4368 : 2195 : }
4369 : :
4370 : : /*
4371 : : * BeginImplicitTransactionBlock
4372 : : * Start an implicit transaction block if we're not already in one.
4373 : : *
4374 : : * Unlike BeginTransactionBlock, this is called directly from the main loop
4375 : : * in postgres.c, not within a Portal. So we can just change blockState
4376 : : * without a lot of ceremony. We do not expect caller to do
4377 : : * CommitTransactionCommand/StartTransactionCommand.
4378 : : */
4379 : : void
3218 4380 : 47621 : BeginImplicitTransactionBlock(void)
4381 : : {
4382 : 47621 : TransactionState s = CurrentTransactionState;
4383 : :
4384 : : /*
4385 : : * If we are in STARTED state (that is, no transaction block is open),
4386 : : * switch to IMPLICIT_INPROGRESS state, creating an implicit transaction
4387 : : * block.
4388 : : *
4389 : : * For caller convenience, we consider all other transaction states as
4390 : : * legal here; otherwise the caller would need its own state check, which
4391 : : * seems rather pointless.
4392 : : */
4393 [ + + ]: 47621 : if (s->blockState == TBLOCK_STARTED)
4394 : 6018 : s->blockState = TBLOCK_IMPLICIT_INPROGRESS;
4395 : 47621 : }
4396 : :
4397 : : /*
4398 : : * EndImplicitTransactionBlock
4399 : : * End an implicit transaction block, if we're in one.
4400 : : *
4401 : : * Like EndTransactionBlock, we just make any needed blockState change here.
4402 : : * The real work will be done in the upcoming CommitTransactionCommand().
4403 : : */
4404 : : void
4405 : 19845 : EndImplicitTransactionBlock(void)
4406 : : {
4407 : 19845 : TransactionState s = CurrentTransactionState;
4408 : :
4409 : : /*
4410 : : * If we are in IMPLICIT_INPROGRESS state, switch back to STARTED state,
4411 : : * allowing CommitTransactionCommand to commit whatever happened during
4412 : : * the implicit transaction block as though it were a single statement.
4413 : : *
4414 : : * For caller convenience, we consider all other transaction states as
4415 : : * legal here; otherwise the caller would need its own state check, which
4416 : : * seems rather pointless.
4417 : : */
4418 [ + + ]: 19845 : if (s->blockState == TBLOCK_IMPLICIT_INPROGRESS)
4419 : 5405 : s->blockState = TBLOCK_STARTED;
4420 : 19845 : }
4421 : :
4422 : : /*
4423 : : * DefineSavepoint
4424 : : * This executes a SAVEPOINT command.
4425 : : */
4426 : : void
3164 peter_e@gmx.net 4427 : 1652 : DefineSavepoint(const char *name)
4428 : : {
7975 bruce@momjian.us 4429 : 1652 : TransactionState s = CurrentTransactionState;
4430 : :
4431 : : /*
4432 : : * Workers synchronize transaction state at the beginning of each parallel
4433 : : * operation, so we can't account for new subtransactions after that
4434 : : * point. (Note that this check will certainly error out if s->blockState
4435 : : * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
4436 : : * below.)
4437 : : */
824 tgl@sss.pgh.pa.us 4438 [ + - - + ]: 1652 : if (IsInParallelMode() || IsParallelWorker())
4079 rhaas@postgresql.org 4439 [ # # ]:UBC 0 : ereport(ERROR,
4440 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
4441 : : errmsg("cannot define savepoints during a parallel operation")));
4442 : :
8008 tgl@sss.pgh.pa.us 4443 [ + + - - ]:CBC 1652 : switch (s->blockState)
4444 : : {
4445 : 1644 : case TBLOCK_INPROGRESS:
4446 : : case TBLOCK_SUBINPROGRESS:
4447 : : /* Normal subtransaction start */
4448 : 1644 : PushTransaction();
3296 4449 : 1644 : s = CurrentTransactionState; /* changed by push */
4450 : :
4451 : : /*
4452 : : * Savepoint names, like the TransactionState block itself, live
4453 : : * in TopTransactionContext.
4454 : : */
7957 4455 [ + + ]: 1644 : if (name)
4456 : 1162 : s->name = MemoryContextStrdup(TopTransactionContext, name);
8008 4457 : 1644 : break;
4458 : :
4459 : : /*
4460 : : * We disallow savepoint commands in implicit transaction blocks.
4461 : : * There would be no great difficulty in allowing them so far as
4462 : : * this module is concerned, but a savepoint seems inconsistent
4463 : : * with exec_simple_query's behavior of abandoning the whole query
4464 : : * string upon error. Also, the point of an implicit transaction
4465 : : * block (as opposed to a regular one) is to automatically close
4466 : : * after an error, so it's hard to see how a savepoint would fit
4467 : : * into that.
4468 : : *
4469 : : * The error messages for this are phrased as if there were no
4470 : : * active transaction block at all, which is historical but
4471 : : * perhaps could be improved.
4472 : : */
3218 4473 : 8 : case TBLOCK_IMPLICIT_INPROGRESS:
4474 [ + - ]: 8 : ereport(ERROR,
4475 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4476 : : /* translator: %s represents an SQL statement name */
4477 : : errmsg("%s can only be used in transaction blocks",
4478 : : "SAVEPOINT")));
4479 : : break;
4480 : :
4481 : : /* These cases are invalid. */
8008 tgl@sss.pgh.pa.us 4482 :UBC 0 : case TBLOCK_DEFAULT:
4483 : : case TBLOCK_STARTED:
4484 : : case TBLOCK_BEGIN:
4485 : : case TBLOCK_PARALLEL_INPROGRESS:
4486 : : case TBLOCK_SUBBEGIN:
4487 : : case TBLOCK_END:
4488 : : case TBLOCK_SUBRELEASE:
4489 : : case TBLOCK_SUBCOMMIT:
4490 : : case TBLOCK_ABORT:
4491 : : case TBLOCK_SUBABORT:
4492 : : case TBLOCK_ABORT_END:
4493 : : case TBLOCK_SUBABORT_END:
4494 : : case TBLOCK_ABORT_PENDING:
4495 : : case TBLOCK_SUBABORT_PENDING:
4496 : : case TBLOCK_SUBRESTART:
4497 : : case TBLOCK_SUBABORT_RESTART:
4498 : : case TBLOCK_PREPARE:
8004 4499 [ # # ]: 0 : elog(FATAL, "DefineSavepoint: unexpected state %s",
4500 : : BlockStateAsString(s->blockState));
4501 : : break;
4502 : : }
8008 tgl@sss.pgh.pa.us 4503 :CBC 1644 : }
4504 : :
4505 : : /*
4506 : : * ReleaseSavepoint
4507 : : * This executes a RELEASE command.
4508 : : *
4509 : : * As above, we don't actually do anything here except change blockState.
4510 : : */
4511 : : void
3056 peter_e@gmx.net 4512 : 183 : ReleaseSavepoint(const char *name)
4513 : : {
7975 bruce@momjian.us 4514 : 183 : TransactionState s = CurrentTransactionState;
4515 : : TransactionState target,
4516 : : xact;
4517 : :
4518 : : /*
4519 : : * Workers synchronize transaction state at the beginning of each parallel
4520 : : * operation, so we can't account for transaction state change after that
4521 : : * point. (Note that this check will certainly error out if s->blockState
4522 : : * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
4523 : : * below.)
4524 : : */
824 tgl@sss.pgh.pa.us 4525 [ + - - + ]: 183 : if (IsInParallelMode() || IsParallelWorker())
4079 rhaas@postgresql.org 4526 [ # # ]:UBC 0 : ereport(ERROR,
4527 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
4528 : : errmsg("cannot release savepoints during a parallel operation")));
4529 : :
8008 tgl@sss.pgh.pa.us 4530 [ - + + - :CBC 183 : switch (s->blockState)
- ]
4531 : : {
4532 : : /*
4533 : : * We can't release a savepoint if there is no savepoint defined.
4534 : : */
8008 tgl@sss.pgh.pa.us 4535 :UBC 0 : case TBLOCK_INPROGRESS:
4536 [ # # ]: 0 : ereport(ERROR,
4537 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4538 : : errmsg("savepoint \"%s\" does not exist", name)));
4539 : : break;
4540 : :
3218 tgl@sss.pgh.pa.us 4541 :CBC 4 : case TBLOCK_IMPLICIT_INPROGRESS:
4542 : : /* See comment about implicit transactions in DefineSavepoint */
4543 [ + - ]: 4 : ereport(ERROR,
4544 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4545 : : /* translator: %s represents an SQL statement name */
4546 : : errmsg("%s can only be used in transaction blocks",
4547 : : "RELEASE SAVEPOINT")));
4548 : : break;
4549 : :
4550 : : /*
4551 : : * We are in a non-aborted subtransaction. This is the only valid
4552 : : * case.
4553 : : */
8008 4554 : 179 : case TBLOCK_SUBINPROGRESS:
4555 : 179 : break;
4556 : :
4557 : : /* These cases are invalid. */
8008 tgl@sss.pgh.pa.us 4558 :UBC 0 : case TBLOCK_DEFAULT:
4559 : : case TBLOCK_STARTED:
4560 : : case TBLOCK_BEGIN:
4561 : : case TBLOCK_PARALLEL_INPROGRESS:
4562 : : case TBLOCK_SUBBEGIN:
4563 : : case TBLOCK_END:
4564 : : case TBLOCK_SUBRELEASE:
4565 : : case TBLOCK_SUBCOMMIT:
4566 : : case TBLOCK_ABORT:
4567 : : case TBLOCK_SUBABORT:
4568 : : case TBLOCK_ABORT_END:
4569 : : case TBLOCK_SUBABORT_END:
4570 : : case TBLOCK_ABORT_PENDING:
4571 : : case TBLOCK_SUBABORT_PENDING:
4572 : : case TBLOCK_SUBRESTART:
4573 : : case TBLOCK_SUBABORT_RESTART:
4574 : : case TBLOCK_PREPARE:
4575 [ # # ]: 0 : elog(FATAL, "ReleaseSavepoint: unexpected state %s",
4576 : : BlockStateAsString(s->blockState));
4577 : : break;
4578 : : }
4579 : :
279 peter@eisentraut.org 4580 [ + - ]:GNC 266 : for (target = s; target; target = target->parent)
4581 : : {
4582 [ + - + + ]: 266 : if (target->name && strcmp(target->name, name) == 0)
8008 tgl@sss.pgh.pa.us 4583 :CBC 179 : break;
4584 : : }
4585 : :
279 peter@eisentraut.org 4586 [ - + ]:GNC 179 : if (!target)
8008 tgl@sss.pgh.pa.us 4587 [ # # ]:UBC 0 : ereport(ERROR,
4588 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4589 : : errmsg("savepoint \"%s\" does not exist", name)));
4590 : :
4591 : : /* disallow crossing savepoint level boundaries */
7979 tgl@sss.pgh.pa.us 4592 [ - + ]:CBC 179 : if (target->savepointLevel != s->savepointLevel)
7979 tgl@sss.pgh.pa.us 4593 [ # # ]:UBC 0 : ereport(ERROR,
4594 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4595 : : errmsg("savepoint \"%s\" does not exist within current savepoint level", name)));
4596 : :
4597 : : /*
4598 : : * Mark "commit pending" all subtransactions up to the target
4599 : : * subtransaction. The actual commits will happen when control gets to
4600 : : * CommitTransactionCommand.
4601 : : */
7979 tgl@sss.pgh.pa.us 4602 :CBC 179 : xact = CurrentTransactionState;
4603 : : for (;;)
4604 : : {
4605 [ - + ]: 266 : Assert(xact->blockState == TBLOCK_SUBINPROGRESS);
5460 simon@2ndQuadrant.co 4606 : 266 : xact->blockState = TBLOCK_SUBRELEASE;
7979 tgl@sss.pgh.pa.us 4607 [ + + ]: 266 : if (xact == target)
4608 : 179 : break;
4609 : 87 : xact = xact->parent;
279 peter@eisentraut.org 4610 [ - + ]:GNC 87 : Assert(xact);
4611 : : }
8008 tgl@sss.pgh.pa.us 4612 :CBC 179 : }
4613 : :
4614 : : /*
4615 : : * RollbackToSavepoint
4616 : : * This executes a ROLLBACK TO <savepoint> command.
4617 : : *
4618 : : * As above, we don't actually do anything here except change blockState.
4619 : : */
4620 : : void
3056 peter_e@gmx.net 4621 : 490 : RollbackToSavepoint(const char *name)
4622 : : {
8008 tgl@sss.pgh.pa.us 4623 : 490 : TransactionState s = CurrentTransactionState;
4624 : : TransactionState target,
4625 : : xact;
4626 : :
4627 : : /*
4628 : : * Workers synchronize transaction state at the beginning of each parallel
4629 : : * operation, so we can't account for transaction state change after that
4630 : : * point. (Note that this check will certainly error out if s->blockState
4631 : : * is TBLOCK_PARALLEL_INPROGRESS, so we can treat that as an invalid case
4632 : : * below.)
4633 : : */
824 4634 [ + - - + ]: 490 : if (IsInParallelMode() || IsParallelWorker())
4079 rhaas@postgresql.org 4635 [ # # ]:UBC 0 : ereport(ERROR,
4636 : : (errcode(ERRCODE_INVALID_TRANSACTION_STATE),
4637 : : errmsg("cannot rollback to savepoints during a parallel operation")));
4638 : :
8008 tgl@sss.pgh.pa.us 4639 [ + + + - :CBC 490 : switch (s->blockState)
- ]
4640 : : {
4641 : : /*
4642 : : * We can't rollback to a savepoint if there is no savepoint
4643 : : * defined.
4644 : : */
4645 : 4 : case TBLOCK_INPROGRESS:
4646 : : case TBLOCK_ABORT:
4647 [ + - ]: 4 : ereport(ERROR,
4648 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4649 : : errmsg("savepoint \"%s\" does not exist", name)));
4650 : : break;
4651 : :
3218 4652 : 4 : case TBLOCK_IMPLICIT_INPROGRESS:
4653 : : /* See comment about implicit transactions in DefineSavepoint */
4654 [ + - ]: 4 : ereport(ERROR,
4655 : : (errcode(ERRCODE_NO_ACTIVE_SQL_TRANSACTION),
4656 : : /* translator: %s represents an SQL statement name */
4657 : : errmsg("%s can only be used in transaction blocks",
4658 : : "ROLLBACK TO SAVEPOINT")));
4659 : : break;
4660 : :
4661 : : /*
4662 : : * There is at least one savepoint, so proceed.
4663 : : */
8008 4664 : 482 : case TBLOCK_SUBINPROGRESS:
4665 : : case TBLOCK_SUBABORT:
4666 : 482 : break;
4667 : :
4668 : : /* These cases are invalid. */
8008 tgl@sss.pgh.pa.us 4669 :UBC 0 : case TBLOCK_DEFAULT:
4670 : : case TBLOCK_STARTED:
4671 : : case TBLOCK_BEGIN:
4672 : : case TBLOCK_PARALLEL_INPROGRESS:
4673 : : case TBLOCK_SUBBEGIN:
4674 : : case TBLOCK_END:
4675 : : case TBLOCK_SUBRELEASE:
4676 : : case TBLOCK_SUBCOMMIT:
4677 : : case TBLOCK_ABORT_END:
4678 : : case TBLOCK_SUBABORT_END:
4679 : : case TBLOCK_ABORT_PENDING:
4680 : : case TBLOCK_SUBABORT_PENDING:
4681 : : case TBLOCK_SUBRESTART:
4682 : : case TBLOCK_SUBABORT_RESTART:
4683 : : case TBLOCK_PREPARE:
4684 [ # # ]: 0 : elog(FATAL, "RollbackToSavepoint: unexpected state %s",
4685 : : BlockStateAsString(s->blockState));
4686 : : break;
4687 : : }
4688 : :
279 peter@eisentraut.org 4689 [ + - ]:GNC 520 : for (target = s; target; target = target->parent)
4690 : : {
4691 [ + - + + ]: 520 : if (target->name && strcmp(target->name, name) == 0)
8008 tgl@sss.pgh.pa.us 4692 :CBC 482 : break;
4693 : : }
4694 : :
279 peter@eisentraut.org 4695 [ - + ]:GNC 482 : if (!target)
8008 tgl@sss.pgh.pa.us 4696 [ # # ]:UBC 0 : ereport(ERROR,
4697 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4698 : : errmsg("savepoint \"%s\" does not exist", name)));
4699 : :
4700 : : /* disallow crossing savepoint level boundaries */
8001 tgl@sss.pgh.pa.us 4701 [ - + ]:CBC 482 : if (target->savepointLevel != s->savepointLevel)
8001 tgl@sss.pgh.pa.us 4702 [ # # ]:UBC 0 : ereport(ERROR,
4703 : : (errcode(ERRCODE_S_E_INVALID_SPECIFICATION),
4704 : : errmsg("savepoint \"%s\" does not exist within current savepoint level", name)));
4705 : :
4706 : : /*
4707 : : * Mark "abort pending" all subtransactions up to the target
4708 : : * subtransaction. The actual aborts will happen when control gets to
4709 : : * CommitTransactionCommand.
4710 : : */
8008 tgl@sss.pgh.pa.us 4711 :CBC 482 : xact = CurrentTransactionState;
4712 : : for (;;)
4713 : : {
7957 4714 [ + + ]: 520 : if (xact == target)
4715 : 482 : break;
4716 [ + - ]: 38 : if (xact->blockState == TBLOCK_SUBINPROGRESS)
4717 : 38 : xact->blockState = TBLOCK_SUBABORT_PENDING;
7957 tgl@sss.pgh.pa.us 4718 [ # # ]:UBC 0 : else if (xact->blockState == TBLOCK_SUBABORT)
4719 : 0 : xact->blockState = TBLOCK_SUBABORT_END;
4720 : : else
4721 [ # # ]: 0 : elog(FATAL, "RollbackToSavepoint: unexpected state %s",
4722 : : BlockStateAsString(xact->blockState));
8008 tgl@sss.pgh.pa.us 4723 :CBC 38 : xact = xact->parent;
279 peter@eisentraut.org 4724 [ - + ]:GNC 38 : Assert(xact);
4725 : : }
4726 : :
4727 : : /* And mark the target as "restart pending" */
7957 tgl@sss.pgh.pa.us 4728 [ + + ]:CBC 482 : if (xact->blockState == TBLOCK_SUBINPROGRESS)
4729 : 337 : xact->blockState = TBLOCK_SUBRESTART;
4730 [ + - ]: 145 : else if (xact->blockState == TBLOCK_SUBABORT)
4731 : 145 : xact->blockState = TBLOCK_SUBABORT_RESTART;
4732 : : else
7957 tgl@sss.pgh.pa.us 4733 [ # # ]:UBC 0 : elog(FATAL, "RollbackToSavepoint: unexpected state %s",
4734 : : BlockStateAsString(xact->blockState));
8008 tgl@sss.pgh.pa.us 4735 :CBC 482 : }
4736 : :
4737 : : /*
4738 : : * BeginInternalSubTransaction
4739 : : * This is the same as DefineSavepoint except it allows TBLOCK_STARTED,
4740 : : * TBLOCK_IMPLICIT_INPROGRESS, TBLOCK_PARALLEL_INPROGRESS, TBLOCK_END,
4741 : : * and TBLOCK_PREPARE states, and therefore it can safely be used in
4742 : : * functions that might be called when not inside a BEGIN block or when
4743 : : * running deferred triggers at COMMIT/PREPARE time. Also, it
4744 : : * automatically does CommitTransactionCommand/StartTransactionCommand
4745 : : * instead of expecting the caller to do it.
4746 : : */
4747 : : void
3164 peter_e@gmx.net 4748 : 11038 : BeginInternalSubTransaction(const char *name)
4749 : : {
7975 bruce@momjian.us 4750 : 11038 : TransactionState s = CurrentTransactionState;
824 tgl@sss.pgh.pa.us 4751 : 11038 : bool save_ExitOnAnyError = ExitOnAnyError;
4752 : :
4753 : : /*
4754 : : * Errors within this function are improbable, but if one does happen we
4755 : : * force a FATAL exit. Callers generally aren't prepared to handle losing
4756 : : * control, and moreover our transaction state is probably corrupted if we
4757 : : * fail partway through; so an ordinary ERROR longjmp isn't okay.
4758 : : */
4759 : 11038 : ExitOnAnyError = true;
4760 : :
4761 : : /*
4762 : : * We do not check for parallel mode here. It's permissible to start and
4763 : : * end "internal" subtransactions while in parallel mode, so long as no
4764 : : * new XIDs or command IDs are assigned. Enforcement of that occurs in
4765 : : * AssignTransactionId() and CommandCounterIncrement().
4766 : : */
4767 : :
8004 4768 [ + - - ]: 11038 : switch (s->blockState)
4769 : : {
4770 : 11038 : case TBLOCK_STARTED:
4771 : : case TBLOCK_INPROGRESS:
4772 : : case TBLOCK_IMPLICIT_INPROGRESS:
4773 : : case TBLOCK_PARALLEL_INPROGRESS:
4774 : : case TBLOCK_END:
4775 : : case TBLOCK_PREPARE:
4776 : : case TBLOCK_SUBINPROGRESS:
4777 : : /* Normal subtransaction start */
4778 : 11038 : PushTransaction();
3296 4779 : 11038 : s = CurrentTransactionState; /* changed by push */
4780 : :
4781 : : /*
4782 : : * Savepoint names, like the TransactionState block itself, live
4783 : : * in TopTransactionContext.
4784 : : */
8004 4785 [ + + ]: 11038 : if (name)
7957 4786 : 985 : s->name = MemoryContextStrdup(TopTransactionContext, name);
8004 4787 : 11038 : break;
4788 : :
4789 : : /* These cases are invalid. */
8004 tgl@sss.pgh.pa.us 4790 :UBC 0 : case TBLOCK_DEFAULT:
4791 : : case TBLOCK_BEGIN:
4792 : : case TBLOCK_SUBBEGIN:
4793 : : case TBLOCK_SUBRELEASE:
4794 : : case TBLOCK_SUBCOMMIT:
4795 : : case TBLOCK_ABORT:
4796 : : case TBLOCK_SUBABORT:
4797 : : case TBLOCK_ABORT_END:
4798 : : case TBLOCK_SUBABORT_END:
4799 : : case TBLOCK_ABORT_PENDING:
4800 : : case TBLOCK_SUBABORT_PENDING:
4801 : : case TBLOCK_SUBRESTART:
4802 : : case TBLOCK_SUBABORT_RESTART:
4803 [ # # ]: 0 : elog(FATAL, "BeginInternalSubTransaction: unexpected state %s",
4804 : : BlockStateAsString(s->blockState));
4805 : : break;
4806 : : }
4807 : :
8004 tgl@sss.pgh.pa.us 4808 :CBC 11038 : CommitTransactionCommand();
4809 : 11038 : StartTransactionCommand();
4810 : :
824 4811 : 11038 : ExitOnAnyError = save_ExitOnAnyError;
8004 4812 : 11038 : }
4813 : :
4814 : : /*
4815 : : * ReleaseCurrentSubTransaction
4816 : : *
4817 : : * RELEASE (ie, commit) the innermost subtransaction, regardless of its
4818 : : * savepoint name (if any).
4819 : : * NB: do NOT use CommitTransactionCommand/StartTransactionCommand with this.
4820 : : */
4821 : : void
4822 : 6393 : ReleaseCurrentSubTransaction(void)
4823 : : {
4824 : 6393 : TransactionState s = CurrentTransactionState;
4825 : :
4826 : : /*
4827 : : * We do not check for parallel mode here. It's permissible to start and
4828 : : * end "internal" subtransactions while in parallel mode, so long as no
4829 : : * new XIDs or command IDs are assigned.
4830 : : */
4831 : :
4832 [ - + ]: 6393 : if (s->blockState != TBLOCK_SUBINPROGRESS)
8004 tgl@sss.pgh.pa.us 4833 [ # # ]:UBC 0 : elog(ERROR, "ReleaseCurrentSubTransaction: unexpected state %s",
4834 : : BlockStateAsString(s->blockState));
7963 tgl@sss.pgh.pa.us 4835 [ - + ]:CBC 6393 : Assert(s->state == TRANS_INPROGRESS);
8004 4836 : 6393 : MemoryContextSwitchTo(CurTransactionContext);
5410 simon@2ndQuadrant.co 4837 : 6393 : CommitSubTransaction();
7563 bruce@momjian.us 4838 : 6393 : s = CurrentTransactionState; /* changed by pop */
7963 tgl@sss.pgh.pa.us 4839 [ - + ]: 6393 : Assert(s->state == TRANS_INPROGRESS);
8004 4840 : 6393 : }
4841 : :
4842 : : /*
4843 : : * RollbackAndReleaseCurrentSubTransaction
4844 : : *
4845 : : * ROLLBACK and RELEASE (ie, abort) the innermost subtransaction, regardless
4846 : : * of its savepoint name (if any).
4847 : : * NB: do NOT use CommitTransactionCommand/StartTransactionCommand with this.
4848 : : */
4849 : : void
4850 : 4645 : RollbackAndReleaseCurrentSubTransaction(void)
4851 : : {
4852 : 4645 : TransactionState s = CurrentTransactionState;
4853 : :
4854 : : /*
4855 : : * We do not check for parallel mode here. It's permissible to start and
4856 : : * end "internal" subtransactions while in parallel mode, so long as no
4857 : : * new XIDs or command IDs are assigned.
4858 : : */
4859 : :
4860 [ + - - ]: 4645 : switch (s->blockState)
4861 : : {
4862 : : /* Must be in a subtransaction */
4863 : 4645 : case TBLOCK_SUBINPROGRESS:
4864 : : case TBLOCK_SUBABORT:
4865 : 4645 : break;
4866 : :
4867 : : /* These cases are invalid. */
8004 tgl@sss.pgh.pa.us 4868 :UBC 0 : case TBLOCK_DEFAULT:
4869 : : case TBLOCK_STARTED:
4870 : : case TBLOCK_BEGIN:
4871 : : case TBLOCK_IMPLICIT_INPROGRESS:
4872 : : case TBLOCK_PARALLEL_INPROGRESS:
4873 : : case TBLOCK_SUBBEGIN:
4874 : : case TBLOCK_INPROGRESS:
4875 : : case TBLOCK_END:
4876 : : case TBLOCK_SUBRELEASE:
4877 : : case TBLOCK_SUBCOMMIT:
4878 : : case TBLOCK_ABORT:
4879 : : case TBLOCK_ABORT_END:
4880 : : case TBLOCK_SUBABORT_END:
4881 : : case TBLOCK_ABORT_PENDING:
4882 : : case TBLOCK_SUBABORT_PENDING:
4883 : : case TBLOCK_SUBRESTART:
4884 : : case TBLOCK_SUBABORT_RESTART:
4885 : : case TBLOCK_PREPARE:
4886 [ # # ]: 0 : elog(FATAL, "RollbackAndReleaseCurrentSubTransaction: unexpected state %s",
4887 : : BlockStateAsString(s->blockState));
4888 : : break;
4889 : : }
4890 : :
4891 : : /*
4892 : : * Abort the current subtransaction, if needed.
4893 : : */
8004 tgl@sss.pgh.pa.us 4894 [ + + ]:CBC 4645 : if (s->blockState == TBLOCK_SUBINPROGRESS)
4895 : 3660 : AbortSubTransaction();
4896 : :
4897 : : /* And clean it up, too */
7957 4898 : 4645 : CleanupSubTransaction();
4899 : :
4900 : 4645 : s = CurrentTransactionState; /* changed by pop */
1341 peter@eisentraut.org 4901 [ + + + + : 4645 : Assert(s->blockState == TBLOCK_SUBINPROGRESS ||
+ - + + -
+ ]
4902 : : s->blockState == TBLOCK_INPROGRESS ||
4903 : : s->blockState == TBLOCK_IMPLICIT_INPROGRESS ||
4904 : : s->blockState == TBLOCK_PARALLEL_INPROGRESS ||
4905 : : s->blockState == TBLOCK_STARTED);
10948 scrappy@hub.org 4906 : 4645 : }
4907 : :
4908 : : /*
4909 : : * AbortOutOfAnyTransaction
4910 : : *
4911 : : * This routine is provided for error recovery purposes. It aborts any
4912 : : * active transaction or transaction block, leaving the system in a known
4913 : : * idle state.
4914 : : */
4915 : : void
9380 tgl@sss.pgh.pa.us 4916 : 18764 : AbortOutOfAnyTransaction(void)
4917 : : {
10129 4918 : 18764 : TransactionState s = CurrentTransactionState;
4919 : :
4920 : : /* Ensure we're not running in a doomed memory context */
3242 4921 : 18764 : AtAbort_Memory();
4922 : :
4923 : : /*
4924 : : * Get out of any transaction or nested transaction
4925 : : */
4926 : : do
4927 : : {
8034 4928 [ + + + + : 18766 : switch (s->blockState)
- - ]
4929 : : {
4930 : 18150 : case TBLOCK_DEFAULT:
5146 4931 [ - + ]: 18150 : if (s->state == TRANS_DEFAULT)
4932 : : {
4933 : : /* Not in a transaction, do nothing */
4934 : : }
4935 : : else
4936 : : {
4937 : : /*
4938 : : * We can get here after an error during transaction start
4939 : : * (state will be TRANS_START). Need to clean up the
4940 : : * incompletely started transaction. First, adjust the
4941 : : * low-level state to suppress warning message from
4942 : : * AbortTransaction.
4943 : : */
5146 tgl@sss.pgh.pa.us 4944 [ # # ]:UBC 0 : if (s->state == TRANS_START)
4945 : 0 : s->state = TRANS_INPROGRESS;
4946 : 0 : AbortTransaction();
4947 : 0 : CleanupTransaction();
4948 : : }
8034 tgl@sss.pgh.pa.us 4949 :CBC 18150 : break;
4950 : 600 : case TBLOCK_STARTED:
4951 : : case TBLOCK_BEGIN:
4952 : : case TBLOCK_INPROGRESS:
4953 : : case TBLOCK_IMPLICIT_INPROGRESS:
4954 : : case TBLOCK_PARALLEL_INPROGRESS:
4955 : : case TBLOCK_END:
4956 : : case TBLOCK_ABORT_PENDING:
4957 : : case TBLOCK_PREPARE:
4958 : : /* In a transaction, so clean up */
4959 : 600 : AbortTransaction();
4960 : 600 : CleanupTransaction();
4961 : 600 : s->blockState = TBLOCK_DEFAULT;
4962 : 600 : break;
4963 : 14 : case TBLOCK_ABORT:
4964 : : case TBLOCK_ABORT_END:
4965 : :
4966 : : /*
4967 : : * AbortTransaction is already done, still need Cleanup.
4968 : : * However, if we failed partway through running ROLLBACK,
4969 : : * there will be an active portal running that command, which
4970 : : * we need to shut down before doing CleanupTransaction.
4971 : : */
3242 4972 : 14 : AtAbort_Portals();
8034 4973 : 14 : CleanupTransaction();
4974 : 14 : s->blockState = TBLOCK_DEFAULT;
4975 : 14 : break;
4976 : :
4977 : : /*
4978 : : * In a subtransaction, so clean it up and abort parent too
4979 : : */
7957 4980 : 2 : case TBLOCK_SUBBEGIN:
4981 : : case TBLOCK_SUBINPROGRESS:
4982 : : case TBLOCK_SUBRELEASE:
4983 : : case TBLOCK_SUBCOMMIT:
4984 : : case TBLOCK_SUBABORT_PENDING:
4985 : : case TBLOCK_SUBRESTART:
8034 4986 : 2 : AbortSubTransaction();
4987 : 2 : CleanupSubTransaction();
7975 bruce@momjian.us 4988 : 2 : s = CurrentTransactionState; /* changed by pop */
8034 tgl@sss.pgh.pa.us 4989 : 2 : break;
4990 : :
8034 tgl@sss.pgh.pa.us 4991 :UBC 0 : case TBLOCK_SUBABORT:
4992 : : case TBLOCK_SUBABORT_END:
4993 : : case TBLOCK_SUBABORT_RESTART:
4994 : : /* As above, but AbortSubTransaction already done */
3242 4995 [ # # ]: 0 : if (s->curTransactionOwner)
4996 : : {
4997 : : /* As in TBLOCK_ABORT, might have a live portal to zap */
4998 : 0 : AtSubAbort_Portals(s->subTransactionId,
4999 : 0 : s->parent->subTransactionId,
5000 : : s->curTransactionOwner,
5001 : 0 : s->parent->curTransactionOwner);
5002 : : }
8034 5003 : 0 : CleanupSubTransaction();
7975 bruce@momjian.us 5004 : 0 : s = CurrentTransactionState; /* changed by pop */
8034 tgl@sss.pgh.pa.us 5005 : 0 : break;
5006 : : }
8034 tgl@sss.pgh.pa.us 5007 [ + + ]:CBC 18766 : } while (s->blockState != TBLOCK_DEFAULT);
5008 : :
5009 : : /* Should be out of all subxacts now */
5010 [ - + ]: 18764 : Assert(s->parent == NULL);
5011 : :
5012 : : /*
5013 : : * Revert to TopMemoryContext, to ensure we exit in a well-defined state
5014 : : * whether there were any transactions to close or not. (Callers that
5015 : : * don't intend to exit soon should switch to some other context to avoid
5016 : : * long-term memory leaks.)
5017 : : */
729 5018 : 18764 : MemoryContextSwitchTo(TopMemoryContext);
10129 5019 : 18764 : }
5020 : :
5021 : : /*
5022 : : * IsTransactionBlock --- are we within a transaction block?
5023 : : */
5024 : : bool
9380 5025 : 243711 : IsTransactionBlock(void)
5026 : : {
10523 bruce@momjian.us 5027 : 243711 : TransactionState s = CurrentTransactionState;
5028 : :
8121 5029 [ + + + + ]: 243711 : if (s->blockState == TBLOCK_DEFAULT || s->blockState == TBLOCK_STARTED)
8466 tgl@sss.pgh.pa.us 5030 : 175570 : return false;
5031 : :
5032 : 68141 : return true;
5033 : : }
5034 : :
5035 : : /*
5036 : : * IsTransactionOrTransactionBlock --- are we within either a transaction
5037 : : * or a transaction block? (The backend is only really "idle" when this
5038 : : * returns false.)
5039 : : *
5040 : : * This should match up with IsTransactionBlock and IsTransactionState.
5041 : : */
5042 : : bool
8293 5043 : 846979 : IsTransactionOrTransactionBlock(void)
5044 : : {
5045 : 846979 : TransactionState s = CurrentTransactionState;
5046 : :
8121 bruce@momjian.us 5047 [ + + ]: 846979 : if (s->blockState == TBLOCK_DEFAULT)
8293 tgl@sss.pgh.pa.us 5048 : 756271 : return false;
5049 : :
5050 : 90708 : return true;
5051 : : }
5052 : :
5053 : : /*
5054 : : * TransactionBlockStatusCode - return status code to send in ReadyForQuery
5055 : : */
5056 : : char
8466 5057 : 416451 : TransactionBlockStatusCode(void)
5058 : : {
5059 : 416451 : TransactionState s = CurrentTransactionState;
5060 : :
5061 [ + + + - ]: 416451 : switch (s->blockState)
5062 : : {
5063 : 326417 : case TBLOCK_DEFAULT:
5064 : : case TBLOCK_STARTED:
5065 : 326417 : return 'I'; /* idle --- not in transaction */
5066 : 88826 : case TBLOCK_BEGIN:
5067 : : case TBLOCK_SUBBEGIN:
5068 : : case TBLOCK_INPROGRESS:
5069 : : case TBLOCK_IMPLICIT_INPROGRESS:
5070 : : case TBLOCK_PARALLEL_INPROGRESS:
5071 : : case TBLOCK_SUBINPROGRESS:
5072 : : case TBLOCK_END:
5073 : : case TBLOCK_SUBRELEASE:
5074 : : case TBLOCK_SUBCOMMIT:
5075 : : case TBLOCK_PREPARE:
5076 : 88826 : return 'T'; /* in transaction */
5077 : 1208 : case TBLOCK_ABORT:
5078 : : case TBLOCK_SUBABORT:
5079 : : case TBLOCK_ABORT_END:
5080 : : case TBLOCK_SUBABORT_END:
5081 : : case TBLOCK_ABORT_PENDING:
5082 : : case TBLOCK_SUBABORT_PENDING:
5083 : : case TBLOCK_SUBRESTART:
5084 : : case TBLOCK_SUBABORT_RESTART:
5085 : 1208 : return 'E'; /* in failed transaction */
5086 : : }
5087 : :
5088 : : /* should never get here */
8034 tgl@sss.pgh.pa.us 5089 [ # # ]:UBC 0 : elog(FATAL, "invalid transaction block state: %s",
5090 : : BlockStateAsString(s->blockState));
5091 : : return 0; /* keep compiler quiet */
5092 : : }
5093 : :
5094 : : /*
5095 : : * IsSubTransaction
5096 : : */
5097 : : bool
8034 tgl@sss.pgh.pa.us 5098 :CBC 609698 : IsSubTransaction(void)
5099 : : {
5100 : 609698 : TransactionState s = CurrentTransactionState;
5101 : :
8003 5102 [ + + ]: 609698 : if (s->nestingLevel >= 2)
5103 : 473 : return true;
5104 : :
5105 : 609225 : return false;
5106 : : }
5107 : :
5108 : : /*
5109 : : * StartSubTransaction
5110 : : *
5111 : : * If you're wondering why this is separate from PushTransaction: it's because
5112 : : * we can't conveniently do this stuff right inside DefineSavepoint. The
5113 : : * SAVEPOINT utility command will be executed inside a Portal, and if we
5114 : : * muck with CurrentMemoryContext or CurrentResourceOwner then exit from
5115 : : * the Portal will undo those settings. So we make DefineSavepoint just
5116 : : * push a dummy transaction block, and when control returns to the main
5117 : : * idle loop, CommitTransactionCommand will be called, and we'll come here
5118 : : * to finish starting the subtransaction.
5119 : : */
5120 : : static void
8034 5121 : 12682 : StartSubTransaction(void)
5122 : : {
5123 : 12682 : TransactionState s = CurrentTransactionState;
5124 : :
5125 [ - + ]: 12682 : if (s->state != TRANS_DEFAULT)
8007 tgl@sss.pgh.pa.us 5126 [ # # ]:UBC 0 : elog(WARNING, "StartSubTransaction while in %s state",
5127 : : TransStateAsString(s->state));
5128 : :
8034 tgl@sss.pgh.pa.us 5129 :CBC 12682 : s->state = TRANS_START;
5130 : :
5131 : : /*
5132 : : * Initialize subsystems for new subtransaction
5133 : : *
5134 : : * must initialize resource-management stuff first
5135 : : */
8018 5136 : 12682 : AtSubStart_Memory();
5137 : 12682 : AtSubStart_ResourceOwner();
7963 5138 : 12682 : AfterTriggerBeginSubXact();
5139 : :
8034 5140 : 12682 : s->state = TRANS_INPROGRESS;
5141 : :
5142 : : /*
5143 : : * Call start-of-subxact callbacks
5144 : : */
7957 5145 : 12682 : CallSubXactCallbacks(SUBXACT_EVENT_START_SUB, s->subTransactionId,
5146 : 12682 : s->parent->subTransactionId);
5147 : :
8034 5148 : 12682 : ShowTransactionState("StartSubTransaction");
5149 : 12682 : }
5150 : :
5151 : : /*
5152 : : * CommitSubTransaction
5153 : : *
5154 : : * The caller has to make sure to always reassign CurrentTransactionState
5155 : : * if it has a local pointer to it after calling this function.
5156 : : */
5157 : : static void
5410 simon@2ndQuadrant.co 5158 : 7262 : CommitSubTransaction(void)
5159 : : {
8034 tgl@sss.pgh.pa.us 5160 : 7262 : TransactionState s = CurrentTransactionState;
5161 : :
5162 : 7262 : ShowTransactionState("CommitSubTransaction");
5163 : :
5164 [ - + ]: 7262 : if (s->state != TRANS_INPROGRESS)
8007 tgl@sss.pgh.pa.us 5165 [ # # ]:UBC 0 : elog(WARNING, "CommitSubTransaction while in %s state",
5166 : : TransStateAsString(s->state));
5167 : :
5168 : : /* Pre-commit processing goes here */
5169 : :
4884 tgl@sss.pgh.pa.us 5170 :CBC 7262 : CallSubXactCallbacks(SUBXACT_EVENT_PRE_COMMIT_SUB, s->subTransactionId,
5171 : 7262 : s->parent->subTransactionId);
5172 : :
5173 : : /*
5174 : : * If this subxact has started any unfinished parallel operation, clean up
5175 : : * its workers and exit parallel mode. Warn about leaked resources.
5176 : : */
824 5177 : 7262 : AtEOSubXact_Parallel(true, s->subTransactionId);
5178 [ - + ]: 7262 : if (s->parallelModeLevel != 0)
5179 : : {
824 tgl@sss.pgh.pa.us 5180 [ # # ]:UBC 0 : elog(WARNING, "parallelModeLevel is %d not 0 at end of subtransaction",
5181 : : s->parallelModeLevel);
4079 rhaas@postgresql.org 5182 : 0 : s->parallelModeLevel = 0;
5183 : : }
5184 : :
5185 : : /* Do the actual "commit", such as it is */
8034 tgl@sss.pgh.pa.us 5186 :CBC 7262 : s->state = TRANS_COMMIT;
5187 : :
5188 : : /* Must CCI to ensure commands of subtransaction are seen as done */
5189 : 7262 : CommandCounterIncrement();
5190 : :
5191 : : /*
5192 : : * Prior to 8.4 we marked subcommit in clog at this point. We now only
5193 : : * perform that step, if required, as part of the atomic update of the
5194 : : * whole transaction tree at top level commit or abort.
5195 : : */
5196 : :
5197 : : /* Post-commit cleanup */
2651 tmunro@postgresql.or 5198 [ + + ]: 7262 : if (FullTransactionIdIsValid(s->fullTransactionId))
6873 tgl@sss.pgh.pa.us 5199 : 5353 : AtSubCommit_childXids();
7963 5200 : 7262 : AfterTriggerEndSubXact(true);
7957 5201 : 7262 : AtSubCommit_Portals(s->subTransactionId,
5202 : 7262 : s->parent->subTransactionId,
1733 5203 : 7262 : s->parent->nestingLevel,
8003 5204 : 7262 : s->parent->curTransactionOwner);
7957 5205 : 7262 : AtEOSubXact_LargeObject(true, s->subTransactionId,
5206 : 7262 : s->parent->subTransactionId);
8003 5207 : 7262 : AtSubCommit_Notify();
5208 : :
7957 5209 : 7262 : CallSubXactCallbacks(SUBXACT_EVENT_COMMIT_SUB, s->subTransactionId,
5210 : 7262 : s->parent->subTransactionId);
5211 : :
8018 5212 : 7262 : ResourceOwnerRelease(s->curTransactionOwner,
5213 : : RESOURCE_RELEASE_BEFORE_LOCKS,
5214 : : true, false);
7957 5215 : 7262 : AtEOSubXact_RelationCache(true, s->subTransactionId,
5216 : 7262 : s->parent->subTransactionId);
614 akorotkov@postgresql 5217 : 7262 : AtEOSubXact_TypeCache();
8003 tgl@sss.pgh.pa.us 5218 : 7262 : AtEOSubXact_Inval(true);
7967 5219 : 7262 : AtSubCommit_smgr();
5220 : :
5221 : : /*
5222 : : * The only lock we actually release here is the subtransaction XID lock.
5223 : : */
7957 5224 : 7262 : CurrentResourceOwner = s->curTransactionOwner;
2651 tmunro@postgresql.or 5225 [ + + ]: 7262 : if (FullTransactionIdIsValid(s->fullTransactionId))
5226 : 5353 : XactLockTableDelete(XidFromFullTransactionId(s->fullTransactionId));
5227 : :
5228 : : /*
5229 : : * Other locks should get transferred to their parent resource owner.
5230 : : */
7979 tgl@sss.pgh.pa.us 5231 : 7262 : ResourceOwnerRelease(s->curTransactionOwner,
5232 : : RESOURCE_RELEASE_LOCKS,
5233 : : true, false);
8018 5234 : 7262 : ResourceOwnerRelease(s->curTransactionOwner,
5235 : : RESOURCE_RELEASE_AFTER_LOCKS,
5236 : : true, false);
5237 : :
6875 5238 : 7262 : AtEOXact_GUC(true, s->gucNestLevel);
7957 5239 : 7262 : AtEOSubXact_SPI(true, s->subTransactionId);
5240 : 7262 : AtEOSubXact_on_commit_actions(true, s->subTransactionId,
5241 : 7262 : s->parent->subTransactionId);
5242 : 7262 : AtEOSubXact_Namespace(true, s->subTransactionId,
5243 : 7262 : s->parent->subTransactionId);
5244 : 7262 : AtEOSubXact_Files(true, s->subTransactionId,
5245 : 7262 : s->parent->subTransactionId);
7005 5246 : 7262 : AtEOSubXact_HashTables(true, s->nestingLevel);
6974 5247 : 7262 : AtEOSubXact_PgStat(true, s->nestingLevel);
6623 alvherre@alvh.no-ip. 5248 : 7262 : AtSubCommit_Snapshot(s->nestingLevel);
5249 : :
5250 : : /*
5251 : : * We need to restore the upper transaction's read-only state, in case the
5252 : : * upper is read-write while the child is read-only; GUC will incorrectly
5253 : : * think it should leave the child state in place.
5254 : : */
8007 tgl@sss.pgh.pa.us 5255 : 7262 : XactReadOnly = s->prevXactReadOnly;
5256 : :
8018 5257 : 7262 : CurrentResourceOwner = s->parent->curTransactionOwner;
5258 : 7262 : CurTransactionResourceOwner = s->parent->curTransactionOwner;
7979 5259 : 7262 : ResourceOwnerDelete(s->curTransactionOwner);
8018 5260 : 7262 : s->curTransactionOwner = NULL;
5261 : :
8034 5262 : 7262 : AtSubCommit_Memory();
5263 : :
5264 : 7262 : s->state = TRANS_DEFAULT;
5265 : :
7957 5266 : 7262 : PopTransaction();
8034 5267 : 7262 : }
5268 : :
5269 : : /*
5270 : : * AbortSubTransaction
5271 : : */
5272 : : static void
5273 : 5420 : AbortSubTransaction(void)
5274 : : {
5275 : 5420 : TransactionState s = CurrentTransactionState;
5276 : :
5277 : : /* Prevent cancel/die interrupt while cleaning up */
5278 : 5420 : HOLD_INTERRUPTS();
5279 : :
5280 : : /* Make sure we have a valid memory context and resource owner */
7159 5281 : 5420 : AtSubAbort_Memory();
5282 : 5420 : AtSubAbort_ResourceOwner();
5283 : :
5284 : : /*
5285 : : * Release any LW locks we might be holding as quickly as possible.
5286 : : * (Regular locks, however, must be held till we finish aborting.)
5287 : : * Releasing LW locks is critical since we might try to grab them again
5288 : : * while cleaning up!
5289 : : *
5290 : : * FIXME This may be incorrect --- Are there some locks we should keep?
5291 : : * Buffer locks, for example? I don't think so but I'm not sure.
5292 : : */
8034 5293 : 5420 : LWLockReleaseAll();
5294 : :
5295 : : /*
5296 : : * Cleanup waiting for LSN if any.
5297 : : */
55 akorotkov@postgresql 5298 :GNC 5420 : WaitLSNCleanup();
5299 : :
3764 rhaas@postgresql.org 5300 :CBC 5420 : pgstat_report_wait_end();
5301 : 5420 : pgstat_progress_end_command();
5302 : :
470 andres@anarazel.de 5303 : 5420 : pgaio_error_cleanup();
5304 : :
8034 tgl@sss.pgh.pa.us 5305 : 5420 : UnlockBuffers();
5306 : :
5307 : : /* Reset WAL record construction state */
4240 heikki.linnakangas@i 5308 : 5420 : XLogResetInsertion();
5309 : :
5310 : : /* Cancel condition variable sleep */
3113 rhaas@postgresql.org 5311 : 5420 : ConditionVariableCancelSleep();
5312 : :
5313 : : /*
5314 : : * Also clean up any open wait for lock, since the lock manager will choke
5315 : : * if we try to wait for another lock before doing this.
5316 : : */
5186 5317 : 5420 : LockErrorCleanup();
5318 : :
5319 : : /*
5320 : : * If any timeout events are still active, make sure the timeout interrupt
5321 : : * is scheduled. This covers possible loss of a timeout interrupt due to
5322 : : * longjmp'ing out of the SIGINT handler (see notes in handle_sig_alarm).
5323 : : * We delay this till after LockErrorCleanup so that we don't uselessly
5324 : : * reschedule lock or deadlock check timeouts.
5325 : : */
4596 tgl@sss.pgh.pa.us 5326 : 5420 : reschedule_timeouts();
5327 : :
5328 : : /*
5329 : : * Re-enable signals, in case we got here by longjmp'ing out of a signal
5330 : : * handler. We do this fairly early in the sequence so that the timeout
5331 : : * infrastructure will be functional if needed while aborting.
5332 : : */
1243 tmunro@postgresql.or 5333 : 5420 : sigprocmask(SIG_SETMASK, &UnBlockSig, NULL);
5334 : :
5335 : : /*
5336 : : * check the current transaction state
5337 : : */
7159 tgl@sss.pgh.pa.us 5338 : 5420 : ShowTransactionState("AbortSubTransaction");
5339 : :
5340 [ - + ]: 5420 : if (s->state != TRANS_INPROGRESS)
7159 tgl@sss.pgh.pa.us 5341 [ # # ]:UBC 0 : elog(WARNING, "AbortSubTransaction while in %s state",
5342 : : TransStateAsString(s->state));
5343 : :
7159 tgl@sss.pgh.pa.us 5344 :CBC 5420 : s->state = TRANS_ABORT;
5345 : :
5346 : : /*
5347 : : * Reset user ID which might have been changed transiently. (See notes in
5348 : : * AbortTransaction.)
5349 : : */
6047 5350 : 5420 : SetUserIdAndSecContext(s->prevUser, s->prevSecContext);
5351 : :
5352 : : /* Forget about any active REINDEX. */
2261 5353 : 5420 : ResetReindexState(s->nestingLevel);
5354 : :
5355 : : /* Reset logical streaming state. */
2152 akapila@postgresql.o 5356 : 5420 : ResetLogicalStreamingState();
5357 : :
5358 : : /*
5359 : : * No need for SnapBuildResetExportedSnapshotState() here, snapshot
5360 : : * exports are not supported in subtransactions.
5361 : : */
5362 : :
5363 : : /*
5364 : : * If this subxact has started any unfinished parallel operation, clean up
5365 : : * its workers and exit parallel mode. Don't warn about leaked resources.
5366 : : */
824 tgl@sss.pgh.pa.us 5367 : 5420 : AtEOSubXact_Parallel(false, s->subTransactionId);
5368 : 5420 : s->parallelModeLevel = 0;
5369 : :
5370 : : /*
5371 : : * We can skip all this stuff if the subxact failed before creating a
5372 : : * ResourceOwner...
5373 : : */
7957 5374 [ + - ]: 5420 : if (s->curTransactionOwner)
5375 : : {
5376 : 5420 : AfterTriggerEndSubXact(false);
5377 : 5420 : AtSubAbort_Portals(s->subTransactionId,
5378 : 5420 : s->parent->subTransactionId,
5379 : : s->curTransactionOwner,
5380 : 5420 : s->parent->curTransactionOwner);
5381 : 5420 : AtEOSubXact_LargeObject(false, s->subTransactionId,
5382 : 5420 : s->parent->subTransactionId);
5383 : 5420 : AtSubAbort_Notify();
5384 : :
5385 : : /* Advertise the fact that we aborted in pg_xact. */
6870 5386 : 5420 : (void) RecordTransactionAbort(true);
5387 : :
5388 : : /* Post-abort cleanup */
2651 tmunro@postgresql.or 5389 [ + + ]: 5420 : if (FullTransactionIdIsValid(s->fullTransactionId))
7957 tgl@sss.pgh.pa.us 5390 : 859 : AtSubAbort_childXids();
5391 : :
5392 : 5420 : CallSubXactCallbacks(SUBXACT_EVENT_ABORT_SUB, s->subTransactionId,
5393 : 5420 : s->parent->subTransactionId);
5394 : :
5395 : 5420 : ResourceOwnerRelease(s->curTransactionOwner,
5396 : : RESOURCE_RELEASE_BEFORE_LOCKS,
5397 : : false, false);
5398 : :
470 andres@anarazel.de 5399 : 5420 : AtEOXact_Aio(false);
7957 tgl@sss.pgh.pa.us 5400 : 5420 : AtEOSubXact_RelationCache(false, s->subTransactionId,
5401 : 5420 : s->parent->subTransactionId);
614 akorotkov@postgresql 5402 : 5420 : AtEOSubXact_TypeCache();
7957 tgl@sss.pgh.pa.us 5403 : 5420 : AtEOSubXact_Inval(false);
5404 : 5420 : ResourceOwnerRelease(s->curTransactionOwner,
5405 : : RESOURCE_RELEASE_LOCKS,
5406 : : false, false);
5407 : 5420 : ResourceOwnerRelease(s->curTransactionOwner,
5408 : : RESOURCE_RELEASE_AFTER_LOCKS,
5409 : : false, false);
5129 rhaas@postgresql.org 5410 : 5420 : AtSubAbort_smgr();
5411 : :
6875 tgl@sss.pgh.pa.us 5412 : 5420 : AtEOXact_GUC(false, s->gucNestLevel);
7957 5413 : 5420 : AtEOSubXact_SPI(false, s->subTransactionId);
5414 : 5420 : AtEOSubXact_on_commit_actions(false, s->subTransactionId,
5415 : 5420 : s->parent->subTransactionId);
5416 : 5420 : AtEOSubXact_Namespace(false, s->subTransactionId,
5417 : 5420 : s->parent->subTransactionId);
5418 : 5420 : AtEOSubXact_Files(false, s->subTransactionId,
5419 : 5420 : s->parent->subTransactionId);
7005 5420 : 5420 : AtEOSubXact_HashTables(false, s->nestingLevel);
6974 5421 : 5420 : AtEOSubXact_PgStat(false, s->nestingLevel);
6623 alvherre@alvh.no-ip. 5422 : 5420 : AtSubAbort_Snapshot(s->nestingLevel);
5423 : : }
5424 : :
5425 : : /*
5426 : : * Restore the upper transaction's read-only state, too. This should be
5427 : : * redundant with GUC's cleanup but we may as well do it for consistency
5428 : : * with the commit case.
5429 : : */
8007 tgl@sss.pgh.pa.us 5430 : 5420 : XactReadOnly = s->prevXactReadOnly;
5431 : :
8034 5432 [ - + ]: 5420 : RESUME_INTERRUPTS();
5433 : 5420 : }
5434 : :
5435 : : /*
5436 : : * CleanupSubTransaction
5437 : : *
5438 : : * The caller has to make sure to always reassign CurrentTransactionState
5439 : : * if it has a local pointer to it after calling this function.
5440 : : */
5441 : : static void
5442 : 5420 : CleanupSubTransaction(void)
5443 : : {
5444 : 5420 : TransactionState s = CurrentTransactionState;
5445 : :
5446 : 5420 : ShowTransactionState("CleanupSubTransaction");
5447 : :
5448 [ - + ]: 5420 : if (s->state != TRANS_ABORT)
8007 tgl@sss.pgh.pa.us 5449 [ # # ]:UBC 0 : elog(WARNING, "CleanupSubTransaction while in %s state",
5450 : : TransStateAsString(s->state));
5451 : :
7957 tgl@sss.pgh.pa.us 5452 :CBC 5420 : AtSubCleanup_Portals(s->subTransactionId);
5453 : :
8018 5454 : 5420 : CurrentResourceOwner = s->parent->curTransactionOwner;
5455 : 5420 : CurTransactionResourceOwner = s->parent->curTransactionOwner;
7957 5456 [ + - ]: 5420 : if (s->curTransactionOwner)
5457 : 5420 : ResourceOwnerDelete(s->curTransactionOwner);
8018 5458 : 5420 : s->curTransactionOwner = NULL;
5459 : :
8034 5460 : 5420 : AtSubCleanup_Memory();
5461 : :
5462 : 5420 : s->state = TRANS_DEFAULT;
5463 : :
7957 5464 : 5420 : PopTransaction();
8034 5465 : 5420 : }
5466 : :
5467 : : /*
5468 : : * PushTransaction
5469 : : * Create transaction state stack entry for a subtransaction
5470 : : *
5471 : : * The caller has to make sure to always reassign CurrentTransactionState
5472 : : * if it has a local pointer to it after calling this function.
5473 : : */
5474 : : static void
5475 : 12682 : PushTransaction(void)
5476 : : {
7975 bruce@momjian.us 5477 : 12682 : TransactionState p = CurrentTransactionState;
5478 : : TransactionState s;
5479 : :
5480 : : /*
5481 : : * We keep subtransaction state nodes in TopTransactionContext.
5482 : : */
5483 : : s = (TransactionState)
8034 tgl@sss.pgh.pa.us 5484 : 12682 : MemoryContextAllocZero(TopTransactionContext,
5485 : : sizeof(TransactionStateData));
5486 : :
5487 : : /*
5488 : : * Assign a subtransaction ID, watching out for counter wraparound.
5489 : : */
7957 5490 : 12682 : currentSubTransactionId += 1;
5491 [ - + ]: 12682 : if (currentSubTransactionId == InvalidSubTransactionId)
5492 : : {
7957 tgl@sss.pgh.pa.us 5493 :UBC 0 : currentSubTransactionId -= 1;
5494 : 0 : pfree(s);
5495 [ # # ]: 0 : ereport(ERROR,
5496 : : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
5497 : : errmsg("cannot have more than 2^32-1 subtransactions in a transaction")));
5498 : : }
5499 : :
5500 : : /*
5501 : : * We can now stack a minimally valid subtransaction without fear of
5502 : : * failure.
5503 : : */
2651 tmunro@postgresql.or 5504 :CBC 12682 : s->fullTransactionId = InvalidFullTransactionId; /* until assigned */
7957 tgl@sss.pgh.pa.us 5505 : 12682 : s->subTransactionId = currentSubTransactionId;
8034 5506 : 12682 : s->parent = p;
5507 : 12682 : s->nestingLevel = p->nestingLevel + 1;
6875 5508 : 12682 : s->gucNestLevel = NewGUCNestLevel();
8008 5509 : 12682 : s->savepointLevel = p->savepointLevel;
8034 5510 : 12682 : s->state = TRANS_DEFAULT;
5511 : 12682 : s->blockState = TBLOCK_SUBBEGIN;
6047 5512 : 12682 : GetUserIdAndSecContext(&s->prevUser, &s->prevSecContext);
7957 5513 : 12682 : s->prevXactReadOnly = XactReadOnly;
931 michael@paquier.xyz 5514 : 12682 : s->startedInRecovery = p->startedInRecovery;
4079 rhaas@postgresql.org 5515 : 12682 : s->parallelModeLevel = 0;
824 tgl@sss.pgh.pa.us 5516 [ + + - + ]: 12682 : s->parallelChildXact = (p->parallelModeLevel != 0 || p->parallelChildXact);
1701 akapila@postgresql.o 5517 : 12682 : s->topXidLogged = false;
5518 : :
7957 tgl@sss.pgh.pa.us 5519 : 12682 : CurrentTransactionState = s;
5520 : :
5521 : : /*
5522 : : * AbortSubTransaction and CleanupSubTransaction have to be able to cope
5523 : : * with the subtransaction from here on out; in particular they should not
5524 : : * assume that it necessarily has a transaction context, resource owner,
5525 : : * or XID.
5526 : : */
8034 5527 : 12682 : }
5528 : :
5529 : : /*
5530 : : * PopTransaction
5531 : : * Pop back to parent transaction state
5532 : : *
5533 : : * The caller has to make sure to always reassign CurrentTransactionState
5534 : : * if it has a local pointer to it after calling this function.
5535 : : */
5536 : : static void
5537 : 12682 : PopTransaction(void)
5538 : : {
5539 : 12682 : TransactionState s = CurrentTransactionState;
5540 : :
5541 [ - + ]: 12682 : if (s->state != TRANS_DEFAULT)
8007 tgl@sss.pgh.pa.us 5542 [ # # ]:UBC 0 : elog(WARNING, "PopTransaction while in %s state",
5543 : : TransStateAsString(s->state));
5544 : :
8034 tgl@sss.pgh.pa.us 5545 [ - + ]:CBC 12682 : if (s->parent == NULL)
8034 tgl@sss.pgh.pa.us 5546 [ # # ]:UBC 0 : elog(FATAL, "PopTransaction with no parent");
5547 : :
8034 tgl@sss.pgh.pa.us 5548 :CBC 12682 : CurrentTransactionState = s->parent;
5549 : :
5550 : : /* Let's just make sure CurTransactionContext is good */
5551 : 12682 : CurTransactionContext = s->parent->curTransactionContext;
5552 : 12682 : MemoryContextSwitchTo(CurTransactionContext);
5553 : :
5554 : : /* Ditto for ResourceOwner links */
8018 5555 : 12682 : CurTransactionResourceOwner = s->parent->curTransactionOwner;
5556 : 12682 : CurrentResourceOwner = s->parent->curTransactionOwner;
5557 : :
5558 : : /* Free the old child structure */
8008 5559 [ + + ]: 12682 : if (s->name)
5560 : 2147 : pfree(s->name);
8034 5561 : 12682 : pfree(s);
5562 : 12682 : }
5563 : :
5564 : : /*
5565 : : * EstimateTransactionStateSpace
5566 : : * Estimate the amount of space that will be needed by
5567 : : * SerializeTransactionState. It would be OK to overestimate slightly,
5568 : : * but it's simple for us to work out the precise value, so we do.
5569 : : */
5570 : : Size
4079 rhaas@postgresql.org 5571 : 679 : EstimateTransactionStateSpace(void)
5572 : : {
5573 : : TransactionState s;
2651 tmunro@postgresql.or 5574 : 679 : Size nxids = 0;
5575 : 679 : Size size = SerializedTransactionStateHeaderSize;
5576 : :
4079 rhaas@postgresql.org 5577 [ + + ]: 3050 : for (s = CurrentTransactionState; s != NULL; s = s->parent)
5578 : : {
2651 tmunro@postgresql.or 5579 [ + + ]: 2371 : if (FullTransactionIdIsValid(s->fullTransactionId))
4079 rhaas@postgresql.org 5580 : 1359 : nxids = add_size(nxids, 1);
5581 : 2371 : nxids = add_size(nxids, s->nChildXids);
5582 : : }
5583 : :
2636 tmunro@postgresql.or 5584 : 679 : return add_size(size, mul_size(sizeof(TransactionId), nxids));
5585 : : }
5586 : :
5587 : : /*
5588 : : * SerializeTransactionState
5589 : : * Write out relevant details of our transaction state that will be
5590 : : * needed by a parallel worker.
5591 : : *
5592 : : * We need to save and restore XactDeferrable, XactIsoLevel, and the XIDs
5593 : : * associated with this transaction. These are serialized into a
5594 : : * caller-supplied buffer big enough to hold the number of bytes reported by
5595 : : * EstimateTransactionStateSpace(). We emit the XIDs in sorted order for the
5596 : : * convenience of the receiving process.
5597 : : */
5598 : : void
4079 rhaas@postgresql.org 5599 : 679 : SerializeTransactionState(Size maxsize, char *start_address)
5600 : : {
5601 : : TransactionState s;
4056 bruce@momjian.us 5602 : 679 : Size nxids = 0;
5603 : 679 : Size i = 0;
5604 : : TransactionId *workspace;
5605 : : SerializedTransactionState *result;
5606 : :
2651 tmunro@postgresql.or 5607 : 679 : result = (SerializedTransactionState *) start_address;
5608 : :
5609 : 679 : result->xactIsoLevel = XactIsoLevel;
5610 : 679 : result->xactDeferrable = XactDeferrable;
5611 : 679 : result->topFullTransactionId = XactTopFullTransactionId;
5612 : 679 : result->currentFullTransactionId =
5613 : 679 : CurrentTransactionState->fullTransactionId;
5614 : 679 : result->currentCommandId = currentCommandId;
5615 : :
5616 : : /*
5617 : : * If we're running in a parallel worker and launching a parallel worker
5618 : : * of our own, we can just pass along the information that was passed to
5619 : : * us.
5620 : : */
4079 rhaas@postgresql.org 5621 [ - + ]: 679 : if (nParallelCurrentXids > 0)
5622 : : {
2651 tmunro@postgresql.or 5623 :UBC 0 : result->nParallelCurrentXids = nParallelCurrentXids;
5624 : 0 : memcpy(&result->parallelCurrentXids[0], ParallelCurrentXids,
5625 : : nParallelCurrentXids * sizeof(TransactionId));
4079 rhaas@postgresql.org 5626 : 0 : return;
5627 : : }
5628 : :
5629 : : /*
5630 : : * OK, we need to generate a sorted list of XIDs that our workers should
5631 : : * view as current. First, figure out how many there are.
5632 : : */
4079 rhaas@postgresql.org 5633 [ + + ]:CBC 3050 : for (s = CurrentTransactionState; s != NULL; s = s->parent)
5634 : : {
2651 tmunro@postgresql.or 5635 [ + + ]: 2371 : if (FullTransactionIdIsValid(s->fullTransactionId))
4079 rhaas@postgresql.org 5636 : 1359 : nxids = add_size(nxids, 1);
5637 : 2371 : nxids = add_size(nxids, s->nChildXids);
5638 : : }
2651 tmunro@postgresql.or 5639 [ - + ]: 679 : Assert(SerializedTransactionStateHeaderSize + nxids * sizeof(TransactionId)
5640 : : <= maxsize);
5641 : :
5642 : : /* Copy them to our scratch space. */
4079 rhaas@postgresql.org 5643 : 679 : workspace = palloc(nxids * sizeof(TransactionId));
5644 [ + + ]: 3050 : for (s = CurrentTransactionState; s != NULL; s = s->parent)
5645 : : {
2651 tmunro@postgresql.or 5646 [ + + ]: 2371 : if (FullTransactionIdIsValid(s->fullTransactionId))
5647 : 1359 : workspace[i++] = XidFromFullTransactionId(s->fullTransactionId);
1580 tgl@sss.pgh.pa.us 5648 [ - + ]: 2371 : if (s->nChildXids > 0)
1580 tgl@sss.pgh.pa.us 5649 :UBC 0 : memcpy(&workspace[i], s->childXids,
5650 : 0 : s->nChildXids * sizeof(TransactionId));
4079 rhaas@postgresql.org 5651 :CBC 2371 : i += s->nChildXids;
5652 : : }
5653 [ - + ]: 679 : Assert(i == nxids);
5654 : :
5655 : : /* Sort them. */
5656 : 679 : qsort(workspace, nxids, sizeof(TransactionId), xidComparator);
5657 : :
5658 : : /* Copy data into output area. */
2651 tmunro@postgresql.or 5659 : 679 : result->nParallelCurrentXids = nxids;
5660 : 679 : memcpy(&result->parallelCurrentXids[0], workspace,
5661 : : nxids * sizeof(TransactionId));
5662 : : }
5663 : :
5664 : : /*
5665 : : * StartParallelWorkerTransaction
5666 : : * Start a parallel worker transaction, restoring the relevant
5667 : : * transaction state serialized by SerializeTransactionState.
5668 : : */
5669 : : void
4079 rhaas@postgresql.org 5670 : 2012 : StartParallelWorkerTransaction(char *tstatespace)
5671 : : {
5672 : : SerializedTransactionState *tstate;
5673 : :
5674 [ - + ]: 2012 : Assert(CurrentTransactionState->blockState == TBLOCK_DEFAULT);
5675 : 2012 : StartTransaction();
5676 : :
2651 tmunro@postgresql.or 5677 : 2012 : tstate = (SerializedTransactionState *) tstatespace;
5678 : 2012 : XactIsoLevel = tstate->xactIsoLevel;
5679 : 2012 : XactDeferrable = tstate->xactDeferrable;
5680 : 2012 : XactTopFullTransactionId = tstate->topFullTransactionId;
5681 : 2012 : CurrentTransactionState->fullTransactionId =
5682 : : tstate->currentFullTransactionId;
5683 : 2012 : currentCommandId = tstate->currentCommandId;
5684 : 2012 : nParallelCurrentXids = tstate->nParallelCurrentXids;
5685 : 2012 : ParallelCurrentXids = &tstate->parallelCurrentXids[0];
5686 : :
4079 rhaas@postgresql.org 5687 : 2012 : CurrentTransactionState->blockState = TBLOCK_PARALLEL_INPROGRESS;
5688 : 2012 : }
5689 : :
5690 : : /*
5691 : : * EndParallelWorkerTransaction
5692 : : * End a parallel worker transaction.
5693 : : */
5694 : : void
5695 : 2003 : EndParallelWorkerTransaction(void)
5696 : : {
5697 [ - + ]: 2003 : Assert(CurrentTransactionState->blockState == TBLOCK_PARALLEL_INPROGRESS);
5698 : 2003 : CommitTransaction();
5699 : 2003 : CurrentTransactionState->blockState = TBLOCK_DEFAULT;
5700 : 2003 : }
5701 : :
5702 : : /*
5703 : : * ShowTransactionState
5704 : : * Debug support
5705 : : */
5706 : : static void
8034 tgl@sss.pgh.pa.us 5707 : 846756 : ShowTransactionState(const char *str)
5708 : : {
5709 : : /* skip work if message will definitely not be printed */
2045 5710 [ - + ]: 846756 : if (message_level_is_interesting(DEBUG5))
3512 rhaas@postgresql.org 5711 :UBC 0 : ShowTransactionStateRec(str, CurrentTransactionState);
8034 tgl@sss.pgh.pa.us 5712 :CBC 846756 : }
5713 : :
5714 : : /*
5715 : : * ShowTransactionStateRec
5716 : : * Recursive subroutine for ShowTransactionState
5717 : : */
5718 : : static void
3512 rhaas@postgresql.org 5719 :UBC 0 : ShowTransactionStateRec(const char *str, TransactionState s)
5720 : : {
5721 : : StringInfoData buf;
5722 : :
844 akorotkov@postgresql 5723 [ # # ]: 0 : if (s->parent)
5724 : : {
5725 : : /*
5726 : : * Since this function recurses, it could be driven to stack overflow.
5727 : : * This is just a debugging aid, so we can leave out some details
5728 : : * instead of erroring out with check_stack_depth().
5729 : : */
5730 [ # # ]: 0 : if (stack_is_too_deep())
5731 [ # # ]: 0 : ereport(DEBUG5,
5732 : : (errmsg_internal("%s(%d): parent omitted to avoid stack overflow",
5733 : : str, s->nestingLevel)));
5734 : : else
5735 : 0 : ShowTransactionStateRec(str, s->parent);
5736 : : }
5737 : :
5738 : 0 : initStringInfo(&buf);
6679 tgl@sss.pgh.pa.us 5739 [ # # ]: 0 : if (s->nChildXids > 0)
5740 : : {
5741 : : int i;
5742 : :
3512 rhaas@postgresql.org 5743 : 0 : appendStringInfo(&buf, ", children: %u", s->childXids[0]);
6679 tgl@sss.pgh.pa.us 5744 [ # # ]: 0 : for (i = 1; i < s->nChildXids; i++)
5745 : 0 : appendStringInfo(&buf, " %u", s->childXids[i]);
5746 : : }
3512 rhaas@postgresql.org 5747 [ # # # # : 0 : ereport(DEBUG5,
# # ]
5748 : : (errmsg_internal("%s(%d) name: %s; blockState: %s; state: %s, xid/subid/cid: %u/%u/%u%s%s",
5749 : : str, s->nestingLevel,
5750 : : s->name ? s->name : "unnamed",
5751 : : BlockStateAsString(s->blockState),
5752 : : TransStateAsString(s->state),
5753 : : XidFromFullTransactionId(s->fullTransactionId),
5754 : : s->subTransactionId,
5755 : : currentCommandId,
5756 : : currentCommandIdUsed ? " (used)" : "",
5757 : : buf.data)));
6679 tgl@sss.pgh.pa.us 5758 : 0 : pfree(buf.data);
8034 5759 : 0 : }
5760 : :
5761 : : /*
5762 : : * BlockStateAsString
5763 : : * Debug support
5764 : : */
5765 : : static const char *
5766 : 0 : BlockStateAsString(TBlockState blockState)
5767 : : {
8008 5768 [ # # # # : 0 : switch (blockState)
# # # # #
# # # # #
# # # # #
# # ]
5769 : : {
8034 5770 : 0 : case TBLOCK_DEFAULT:
5771 : 0 : return "DEFAULT";
5772 : 0 : case TBLOCK_STARTED:
5773 : 0 : return "STARTED";
5774 : 0 : case TBLOCK_BEGIN:
5775 : 0 : return "BEGIN";
5776 : 0 : case TBLOCK_INPROGRESS:
5777 : 0 : return "INPROGRESS";
3218 5778 : 0 : case TBLOCK_IMPLICIT_INPROGRESS:
5779 : 0 : return "IMPLICIT_INPROGRESS";
4079 rhaas@postgresql.org 5780 : 0 : case TBLOCK_PARALLEL_INPROGRESS:
5781 : 0 : return "PARALLEL_INPROGRESS";
8034 tgl@sss.pgh.pa.us 5782 : 0 : case TBLOCK_END:
5783 : 0 : return "END";
5784 : 0 : case TBLOCK_ABORT:
5785 : 0 : return "ABORT";
7957 5786 : 0 : case TBLOCK_ABORT_END:
3055 peter_e@gmx.net 5787 : 0 : return "ABORT_END";
7957 tgl@sss.pgh.pa.us 5788 : 0 : case TBLOCK_ABORT_PENDING:
3055 peter_e@gmx.net 5789 : 0 : return "ABORT_PENDING";
7683 tgl@sss.pgh.pa.us 5790 : 0 : case TBLOCK_PREPARE:
5791 : 0 : return "PREPARE";
8034 5792 : 0 : case TBLOCK_SUBBEGIN:
3055 peter_e@gmx.net 5793 : 0 : return "SUBBEGIN";
8034 tgl@sss.pgh.pa.us 5794 : 0 : case TBLOCK_SUBINPROGRESS:
3055 peter_e@gmx.net 5795 : 0 : return "SUBINPROGRESS";
5460 simon@2ndQuadrant.co 5796 : 0 : case TBLOCK_SUBRELEASE:
3055 peter_e@gmx.net 5797 : 0 : return "SUBRELEASE";
5460 simon@2ndQuadrant.co 5798 : 0 : case TBLOCK_SUBCOMMIT:
3055 peter_e@gmx.net 5799 : 0 : return "SUBCOMMIT";
8034 tgl@sss.pgh.pa.us 5800 : 0 : case TBLOCK_SUBABORT:
3055 peter_e@gmx.net 5801 : 0 : return "SUBABORT";
7957 tgl@sss.pgh.pa.us 5802 : 0 : case TBLOCK_SUBABORT_END:
3055 peter_e@gmx.net 5803 : 0 : return "SUBABORT_END";
8008 tgl@sss.pgh.pa.us 5804 : 0 : case TBLOCK_SUBABORT_PENDING:
3055 peter_e@gmx.net 5805 : 0 : return "SUBABORT_PENDING";
7957 tgl@sss.pgh.pa.us 5806 : 0 : case TBLOCK_SUBRESTART:
3055 peter_e@gmx.net 5807 : 0 : return "SUBRESTART";
7957 tgl@sss.pgh.pa.us 5808 : 0 : case TBLOCK_SUBABORT_RESTART:
3055 peter_e@gmx.net 5809 : 0 : return "SUBABORT_RESTART";
5810 : : }
8034 tgl@sss.pgh.pa.us 5811 : 0 : return "UNRECOGNIZED";
5812 : : }
5813 : :
5814 : : /*
5815 : : * TransStateAsString
5816 : : * Debug support
5817 : : */
5818 : : static const char *
5819 : 0 : TransStateAsString(TransState state)
5820 : : {
8008 5821 [ # # # # : 0 : switch (state)
# # # ]
5822 : : {
8034 5823 : 0 : case TRANS_DEFAULT:
5824 : 0 : return "DEFAULT";
5825 : 0 : case TRANS_START:
5826 : 0 : return "START";
7683 5827 : 0 : case TRANS_INPROGRESS:
3055 peter_e@gmx.net 5828 : 0 : return "INPROGRESS";
8034 tgl@sss.pgh.pa.us 5829 : 0 : case TRANS_COMMIT:
5830 : 0 : return "COMMIT";
5831 : 0 : case TRANS_ABORT:
5832 : 0 : return "ABORT";
7683 5833 : 0 : case TRANS_PREPARE:
5834 : 0 : return "PREPARE";
5835 : : }
8034 5836 : 0 : return "UNRECOGNIZED";
5837 : : }
5838 : :
5839 : : /*
5840 : : * xactGetCommittedChildren
5841 : : *
5842 : : * Gets the list of committed children of the current transaction. The return
5843 : : * value is the number of child transactions. *ptr is set to point to an
5844 : : * array of TransactionIds. The array is allocated in TopTransactionContext;
5845 : : * the caller should *not* pfree() it (this is a change from pre-8.4 code!).
5846 : : * If there are no subxacts, *ptr is set to NULL.
5847 : : */
5848 : : int
8018 tgl@sss.pgh.pa.us 5849 :CBC 397574 : xactGetCommittedChildren(TransactionId **ptr)
5850 : : {
7975 bruce@momjian.us 5851 : 397574 : TransactionState s = CurrentTransactionState;
5852 : :
6679 tgl@sss.pgh.pa.us 5853 [ + + ]: 397574 : if (s->nChildXids == 0)
8034 5854 : 396938 : *ptr = NULL;
5855 : : else
6679 5856 : 636 : *ptr = s->childXids;
5857 : :
5858 : 397574 : return s->nChildXids;
5859 : : }
5860 : :
5861 : : /*
5862 : : * XLOG support routines
5863 : : */
5864 : :
5865 : :
5866 : : /*
5867 : : * Log the commit record for a plain or twophase transaction commit.
5868 : : *
5869 : : * A 2pc commit will be emitted when twophase_xid is valid, a plain one
5870 : : * otherwise.
5871 : : */
5872 : : XLogRecPtr
4125 andres@anarazel.de 5873 : 157668 : XactLogCommitRecord(TimestampTz commit_time,
5874 : : int nsubxacts, TransactionId *subxacts,
5875 : : int nrels, RelFileLocator *rels,
5876 : : int ndroppedstats, xl_xact_stats_item *droppedstats,
5877 : : int nmsgs, SharedInvalidationMessage *msgs,
5878 : : bool relcacheInval,
5879 : : int xactflags, TransactionId twophase_xid,
5880 : : const char *twophase_gid)
5881 : : {
5882 : : xl_xact_commit xlrec;
5883 : : xl_xact_xinfo xl_xinfo;
5884 : : xl_xact_dbinfo xl_dbinfo;
5885 : : xl_xact_subxacts xl_subxacts;
5886 : : xl_xact_relfilelocators xl_relfilelocators;
5887 : : xl_xact_stats_items xl_dropped_stats;
5888 : : xl_xact_invals xl_invals;
5889 : : xl_xact_twophase xl_twophase;
5890 : : xl_xact_origin xl_origin;
5891 : : uint8 info;
5892 : :
5893 [ - + ]: 157668 : Assert(CritSectionCount > 0);
5894 : :
5895 : 157668 : xl_xinfo.xinfo = 0;
5896 : :
5897 : : /* decide between a plain and 2pc commit */
5898 [ + + ]: 157668 : if (!TransactionIdIsValid(twophase_xid))
5899 : 157403 : info = XLOG_XACT_COMMIT;
5900 : : else
5901 : 265 : info = XLOG_XACT_COMMIT_PREPARED;
5902 : :
5903 : : /* First figure out and collect all the information needed */
5904 : :
5905 : 157668 : xlrec.xact_time = commit_time;
5906 : :
5907 [ + + ]: 157668 : if (relcacheInval)
5908 : 4766 : xl_xinfo.xinfo |= XACT_COMPLETION_UPDATE_RELCACHE_FILE;
5909 [ + + ]: 157668 : if (forceSyncCommit)
5910 : 556 : xl_xinfo.xinfo |= XACT_COMPLETION_FORCE_SYNC_COMMIT;
3387 simon@2ndQuadrant.co 5911 [ + + ]: 157668 : if ((xactflags & XACT_FLAGS_ACQUIREDACCESSEXCLUSIVELOCK))
5912 : 65099 : xl_xinfo.xinfo |= XACT_XINFO_HAS_AE_LOCKS;
5913 : :
5914 : : /*
5915 : : * Check if the caller would like to ask standbys for immediate feedback
5916 : : * once this commit is applied.
5917 : : */
3745 rhaas@postgresql.org 5918 [ + + ]: 157668 : if (synchronous_commit >= SYNCHRONOUS_COMMIT_REMOTE_APPLY)
5919 : 2 : xl_xinfo.xinfo |= XACT_COMPLETION_APPLY_FEEDBACK;
5920 : :
5921 : : /*
5922 : : * Relcache invalidations requires information about the current database
5923 : : * and so does logical decoding.
5924 : : */
4125 andres@anarazel.de 5925 [ + + + + : 157668 : if (nmsgs > 0 || XLogLogicalInfoActive())
+ + ]
5926 : : {
5927 : 113267 : xl_xinfo.xinfo |= XACT_XINFO_HAS_DBINFO;
5928 : 113267 : xl_dbinfo.dbId = MyDatabaseId;
5929 : 113267 : xl_dbinfo.tsId = MyDatabaseTableSpace;
5930 : : }
5931 : :
5932 [ + + ]: 157668 : if (nsubxacts > 0)
5933 : : {
5934 : 533 : xl_xinfo.xinfo |= XACT_XINFO_HAS_SUBXACTS;
5935 : 533 : xl_subxacts.nsubxacts = nsubxacts;
5936 : : }
5937 : :
5938 [ + + ]: 157668 : if (nrels > 0)
5939 : : {
1455 rhaas@postgresql.org 5940 : 12732 : xl_xinfo.xinfo |= XACT_XINFO_HAS_RELFILELOCATORS;
5941 : 12732 : xl_relfilelocators.nrels = nrels;
2143 heikki.linnakangas@i 5942 : 12732 : info |= XLR_SPECIAL_REL_UPDATE;
5943 : : }
5944 : :
1546 andres@anarazel.de 5945 [ + + ]: 157668 : if (ndroppedstats > 0)
5946 : : {
5947 : 15136 : xl_xinfo.xinfo |= XACT_XINFO_HAS_DROPPED_STATS;
5948 : 15136 : xl_dropped_stats.nitems = ndroppedstats;
5949 : : }
5950 : :
4125 5951 [ + + ]: 157668 : if (nmsgs > 0)
5952 : : {
5953 : 112329 : xl_xinfo.xinfo |= XACT_XINFO_HAS_INVALS;
5954 : 112329 : xl_invals.nmsgs = nmsgs;
5955 : : }
5956 : :
5957 [ + + ]: 157668 : if (TransactionIdIsValid(twophase_xid))
5958 : : {
5959 : 265 : xl_xinfo.xinfo |= XACT_XINFO_HAS_TWOPHASE;
5960 : 265 : xl_twophase.xid = twophase_xid;
3016 simon@2ndQuadrant.co 5961 [ - + ]: 265 : Assert(twophase_gid != NULL);
5962 : :
5963 [ + + - + ]: 265 : if (XLogLogicalInfoActive())
5964 : 41 : xl_xinfo.xinfo |= XACT_XINFO_HAS_GID;
5965 : : }
5966 : :
5967 : : /* dump transaction origin information */
153 msawada@postgresql.o 5968 [ + + ]:GNC 157668 : if (replorigin_xact_state.origin != InvalidReplOriginId)
5969 : : {
4080 andres@anarazel.de 5970 :CBC 1085 : xl_xinfo.xinfo |= XACT_XINFO_HAS_ORIGIN;
5971 : :
153 msawada@postgresql.o 5972 :GNC 1085 : xl_origin.origin_lsn = replorigin_xact_state.origin_lsn;
5973 : 1085 : xl_origin.origin_timestamp = replorigin_xact_state.origin_timestamp;
5974 : : }
5975 : :
4125 andres@anarazel.de 5976 [ + + ]:CBC 157668 : if (xl_xinfo.xinfo != 0)
5977 : 116329 : info |= XLOG_XACT_HAS_INFO;
5978 : :
5979 : : /* Then include all the collected data into the commit record. */
5980 : :
5981 : 157668 : XLogBeginInsert();
5982 : :
504 peter@eisentraut.org 5983 : 157668 : XLogRegisterData(&xlrec, sizeof(xl_xact_commit));
5984 : :
4125 andres@anarazel.de 5985 [ + + ]: 157668 : if (xl_xinfo.xinfo != 0)
504 peter@eisentraut.org 5986 : 116329 : XLogRegisterData(&xl_xinfo.xinfo, sizeof(xl_xinfo.xinfo));
5987 : :
4125 andres@anarazel.de 5988 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_DBINFO)
504 peter@eisentraut.org 5989 : 113267 : XLogRegisterData(&xl_dbinfo, sizeof(xl_dbinfo));
5990 : :
4125 andres@anarazel.de 5991 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_SUBXACTS)
5992 : : {
504 peter@eisentraut.org 5993 : 533 : XLogRegisterData(&xl_subxacts,
5994 : : MinSizeOfXactSubxacts);
5995 : 533 : XLogRegisterData(subxacts,
5996 : : nsubxacts * sizeof(TransactionId));
5997 : : }
5998 : :
1455 rhaas@postgresql.org 5999 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
6000 : : {
504 peter@eisentraut.org 6001 : 12732 : XLogRegisterData(&xl_relfilelocators,
6002 : : MinSizeOfXactRelfileLocators);
6003 : 12732 : XLogRegisterData(rels,
6004 : : nrels * sizeof(RelFileLocator));
6005 : : }
6006 : :
1546 andres@anarazel.de 6007 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_DROPPED_STATS)
6008 : : {
504 peter@eisentraut.org 6009 : 15136 : XLogRegisterData(&xl_dropped_stats,
6010 : : MinSizeOfXactStatsItems);
6011 : 15136 : XLogRegisterData(droppedstats,
6012 : : ndroppedstats * sizeof(xl_xact_stats_item));
6013 : : }
6014 : :
4125 andres@anarazel.de 6015 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_INVALS)
6016 : : {
504 peter@eisentraut.org 6017 : 112329 : XLogRegisterData(&xl_invals, MinSizeOfXactInvals);
6018 : 112329 : XLogRegisterData(msgs,
6019 : : nmsgs * sizeof(SharedInvalidationMessage));
6020 : : }
6021 : :
4125 andres@anarazel.de 6022 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_TWOPHASE)
6023 : : {
504 peter@eisentraut.org 6024 : 265 : XLogRegisterData(&xl_twophase, sizeof(xl_xact_twophase));
3016 simon@2ndQuadrant.co 6025 [ + + ]: 265 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_GID)
665 peter@eisentraut.org 6026 : 41 : XLogRegisterData(twophase_gid, strlen(twophase_gid) + 1);
6027 : : }
6028 : :
4080 andres@anarazel.de 6029 [ + + ]: 157668 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_ORIGIN)
504 peter@eisentraut.org 6030 : 1085 : XLogRegisterData(&xl_origin, sizeof(xl_xact_origin));
6031 : :
6032 : : /* we allow filtering by xacts */
3477 andres@anarazel.de 6033 : 157668 : XLogSetRecordFlags(XLOG_INCLUDE_ORIGIN);
6034 : :
4125 6035 : 157668 : return XLogInsert(RM_XACT_ID, info);
6036 : : }
6037 : :
6038 : : /*
6039 : : * Log the commit record for a plain or twophase transaction abort.
6040 : : *
6041 : : * A 2pc abort will be emitted when twophase_xid is valid, a plain one
6042 : : * otherwise.
6043 : : */
6044 : : XLogRecPtr
6045 : 9462 : XactLogAbortRecord(TimestampTz abort_time,
6046 : : int nsubxacts, TransactionId *subxacts,
6047 : : int nrels, RelFileLocator *rels,
6048 : : int ndroppedstats, xl_xact_stats_item *droppedstats,
6049 : : int xactflags, TransactionId twophase_xid,
6050 : : const char *twophase_gid)
6051 : : {
6052 : : xl_xact_abort xlrec;
6053 : : xl_xact_xinfo xl_xinfo;
6054 : : xl_xact_subxacts xl_subxacts;
6055 : : xl_xact_relfilelocators xl_relfilelocators;
6056 : : xl_xact_stats_items xl_dropped_stats;
6057 : : xl_xact_twophase xl_twophase;
6058 : : xl_xact_dbinfo xl_dbinfo;
6059 : : xl_xact_origin xl_origin;
6060 : :
6061 : : uint8 info;
6062 : :
6063 [ - + ]: 9462 : Assert(CritSectionCount > 0);
6064 : :
6065 : 9462 : xl_xinfo.xinfo = 0;
6066 : :
6067 : : /* decide between a plain and 2pc abort */
6068 [ + + ]: 9462 : if (!TransactionIdIsValid(twophase_xid))
6069 : 9415 : info = XLOG_XACT_ABORT;
6070 : : else
6071 : 47 : info = XLOG_XACT_ABORT_PREPARED;
6072 : :
6073 : :
6074 : : /* First figure out and collect all the information needed */
6075 : :
6076 : 9462 : xlrec.xact_time = abort_time;
6077 : :
3387 simon@2ndQuadrant.co 6078 [ + + ]: 9462 : if ((xactflags & XACT_FLAGS_ACQUIREDACCESSEXCLUSIVELOCK))
6079 : 5285 : xl_xinfo.xinfo |= XACT_XINFO_HAS_AE_LOCKS;
6080 : :
4125 andres@anarazel.de 6081 [ + + ]: 9462 : if (nsubxacts > 0)
6082 : : {
6083 : 107 : xl_xinfo.xinfo |= XACT_XINFO_HAS_SUBXACTS;
6084 : 107 : xl_subxacts.nsubxacts = nsubxacts;
6085 : : }
6086 : :
6087 [ + + ]: 9462 : if (nrels > 0)
6088 : : {
1455 rhaas@postgresql.org 6089 : 1358 : xl_xinfo.xinfo |= XACT_XINFO_HAS_RELFILELOCATORS;
6090 : 1358 : xl_relfilelocators.nrels = nrels;
2143 heikki.linnakangas@i 6091 : 1358 : info |= XLR_SPECIAL_REL_UPDATE;
6092 : : }
6093 : :
1546 andres@anarazel.de 6094 [ + + ]: 9462 : if (ndroppedstats > 0)
6095 : : {
6096 : 1934 : xl_xinfo.xinfo |= XACT_XINFO_HAS_DROPPED_STATS;
6097 : 1934 : xl_dropped_stats.nitems = ndroppedstats;
6098 : : }
6099 : :
4125 6100 [ + + ]: 9462 : if (TransactionIdIsValid(twophase_xid))
6101 : : {
6102 : 47 : xl_xinfo.xinfo |= XACT_XINFO_HAS_TWOPHASE;
6103 : 47 : xl_twophase.xid = twophase_xid;
3016 simon@2ndQuadrant.co 6104 [ - + ]: 47 : Assert(twophase_gid != NULL);
6105 : :
6106 [ + + - + ]: 47 : if (XLogLogicalInfoActive())
6107 : 13 : xl_xinfo.xinfo |= XACT_XINFO_HAS_GID;
6108 : : }
6109 : :
6110 [ + + + + : 9462 : if (TransactionIdIsValid(twophase_xid) && XLogLogicalInfoActive())
- + ]
6111 : : {
6112 : 13 : xl_xinfo.xinfo |= XACT_XINFO_HAS_DBINFO;
6113 : 13 : xl_dbinfo.dbId = MyDatabaseId;
6114 : 13 : xl_dbinfo.tsId = MyDatabaseTableSpace;
6115 : : }
6116 : :
6117 : : /*
6118 : : * Dump transaction origin information. We need this during recovery to
6119 : : * update the replication origin progress.
6120 : : */
153 msawada@postgresql.o 6121 [ + + ]:GNC 9462 : if (replorigin_xact_state.origin != InvalidReplOriginId)
6122 : : {
3016 simon@2ndQuadrant.co 6123 :CBC 27 : xl_xinfo.xinfo |= XACT_XINFO_HAS_ORIGIN;
6124 : :
153 msawada@postgresql.o 6125 :GNC 27 : xl_origin.origin_lsn = replorigin_xact_state.origin_lsn;
6126 : 27 : xl_origin.origin_timestamp = replorigin_xact_state.origin_timestamp;
6127 : : }
6128 : :
4125 andres@anarazel.de 6129 [ + + ]:CBC 9462 : if (xl_xinfo.xinfo != 0)
6130 : 5538 : info |= XLOG_XACT_HAS_INFO;
6131 : :
6132 : : /* Then include all the collected data into the abort record. */
6133 : :
6134 : 9462 : XLogBeginInsert();
6135 : :
504 peter@eisentraut.org 6136 : 9462 : XLogRegisterData(&xlrec, MinSizeOfXactAbort);
6137 : :
4125 andres@anarazel.de 6138 [ + + ]: 9462 : if (xl_xinfo.xinfo != 0)
504 peter@eisentraut.org 6139 : 5538 : XLogRegisterData(&xl_xinfo, sizeof(xl_xinfo));
6140 : :
3016 simon@2ndQuadrant.co 6141 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_DBINFO)
504 peter@eisentraut.org 6142 : 13 : XLogRegisterData(&xl_dbinfo, sizeof(xl_dbinfo));
6143 : :
4125 andres@anarazel.de 6144 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_SUBXACTS)
6145 : : {
504 peter@eisentraut.org 6146 : 107 : XLogRegisterData(&xl_subxacts,
6147 : : MinSizeOfXactSubxacts);
6148 : 107 : XLogRegisterData(subxacts,
6149 : : nsubxacts * sizeof(TransactionId));
6150 : : }
6151 : :
1455 rhaas@postgresql.org 6152 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_RELFILELOCATORS)
6153 : : {
504 peter@eisentraut.org 6154 : 1358 : XLogRegisterData(&xl_relfilelocators,
6155 : : MinSizeOfXactRelfileLocators);
6156 : 1358 : XLogRegisterData(rels,
6157 : : nrels * sizeof(RelFileLocator));
6158 : : }
6159 : :
1546 andres@anarazel.de 6160 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_DROPPED_STATS)
6161 : : {
504 peter@eisentraut.org 6162 : 1934 : XLogRegisterData(&xl_dropped_stats,
6163 : : MinSizeOfXactStatsItems);
6164 : 1934 : XLogRegisterData(droppedstats,
6165 : : ndroppedstats * sizeof(xl_xact_stats_item));
6166 : : }
6167 : :
4125 andres@anarazel.de 6168 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_TWOPHASE)
6169 : : {
504 peter@eisentraut.org 6170 : 47 : XLogRegisterData(&xl_twophase, sizeof(xl_xact_twophase));
3016 simon@2ndQuadrant.co 6171 [ + + ]: 47 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_GID)
665 peter@eisentraut.org 6172 : 13 : XLogRegisterData(twophase_gid, strlen(twophase_gid) + 1);
6173 : : }
6174 : :
3016 simon@2ndQuadrant.co 6175 [ + + ]: 9462 : if (xl_xinfo.xinfo & XACT_XINFO_HAS_ORIGIN)
504 peter@eisentraut.org 6176 : 27 : XLogRegisterData(&xl_origin, sizeof(xl_xact_origin));
6177 : :
6178 : : /* Include the replication origin */
1268 akapila@postgresql.o 6179 : 9462 : XLogSetRecordFlags(XLOG_INCLUDE_ORIGIN);
6180 : :
4125 andres@anarazel.de 6181 : 9462 : return XLogInsert(RM_XACT_ID, info);
6182 : : }
6183 : :
6184 : : /*
6185 : : * Before 9.0 this was a fairly short function, but now it performs many
6186 : : * actions for which the order of execution is critical.
6187 : : */
6188 : : static void
6189 : 24157 : xact_redo_commit(xl_xact_parsed_commit *parsed,
6190 : : TransactionId xid,
6191 : : XLogRecPtr lsn,
6192 : : ReplOriginId origin_id)
6193 : : {
6194 : : TransactionId max_xid;
6195 : : TimestampTz commit_time;
6196 : :
3303 alvherre@alvh.no-ip. 6197 [ - + ]: 24157 : Assert(TransactionIdIsValid(xid));
6198 : :
4125 andres@anarazel.de 6199 : 24157 : max_xid = TransactionIdLatest(xid, parsed->nsubxacts, parsed->subxacts);
6200 : :
6201 : : /* Make sure nextXid is beyond any XID mentioned in the record. */
2651 tmunro@postgresql.or 6202 : 24157 : AdvanceNextFullTransactionIdPastXid(max_xid);
6203 : :
3747 andres@anarazel.de 6204 [ - + ]: 24157 : Assert(((parsed->xinfo & XACT_XINFO_HAS_ORIGIN) == 0) ==
6205 : : (origin_id == InvalidReplOriginId));
6206 : :
4080 6207 [ + + ]: 24157 : if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
6208 : 20 : commit_time = parsed->origin_timestamp;
6209 : : else
6210 : 24137 : commit_time = parsed->xact_time;
6211 : :
6212 : : /* Set the transaction commit timestamp and metadata */
4125 6213 : 24157 : TransactionTreeSetCommitTsData(xid, parsed->nsubxacts, parsed->subxacts,
6214 : : commit_time, origin_id);
6215 : :
5892 simon@2ndQuadrant.co 6216 [ + + ]: 24157 : if (standbyState == STANDBY_DISABLED)
6217 : : {
6218 : : /*
6219 : : * Mark the transaction committed in pg_xact.
6220 : : */
4125 andres@anarazel.de 6221 : 2225 : TransactionIdCommitTree(xid, parsed->nsubxacts, parsed->subxacts);
6222 : : }
6223 : : else
6224 : : {
6225 : : /*
6226 : : * If a transaction completion record arrives that has as-yet
6227 : : * unobserved subtransactions then this will not have been fully
6228 : : * handled by the call to RecordKnownAssignedTransactionIds() in the
6229 : : * main recovery loop in PerformWalRecovery(). So we need to do
6230 : : * bookkeeping again to cover that case. This is confusing and it is
6231 : : * easy to think this call is irrelevant, which has happened three
6232 : : * times in development already. Leave it in.
6233 : : */
6037 simon@2ndQuadrant.co 6234 : 21932 : RecordKnownAssignedTransactionIds(max_xid);
6235 : :
6236 : : /*
6237 : : * Mark the transaction committed in pg_xact. We use async commit
6238 : : * protocol during recovery to provide information on database
6239 : : * consistency for when users try to set hint bits. It is important
6240 : : * that we do not set hint bits until the minRecoveryPoint is past
6241 : : * this commit record. This ensures that if we crash we don't see hint
6242 : : * bits set on changes made by transactions that haven't yet
6243 : : * recovered. It's unlikely but it's good to be safe.
6244 : : */
2343 alvherre@alvh.no-ip. 6245 : 21932 : TransactionIdAsyncCommitTree(xid, parsed->nsubxacts, parsed->subxacts, lsn);
6246 : :
6247 : : /*
6248 : : * We must mark clog before we update the ProcArray.
6249 : : */
6250 : 21932 : ExpireTreeKnownAssignedTransactionIds(xid, parsed->nsubxacts, parsed->subxacts, max_xid);
6251 : :
6252 : : /*
6253 : : * Send any cache invalidations attached to the commit. We must
6254 : : * maintain the same order of invalidation then release locks as
6255 : : * occurs in CommitTransaction().
6256 : : */
6257 : 21932 : ProcessCommittedInvalidationMessages(parsed->msgs, parsed->nmsgs,
3296 tgl@sss.pgh.pa.us 6258 : 21932 : XactCompletionRelcacheInitFileInval(parsed->xinfo),
6259 : : parsed->dbId, parsed->tsId);
6260 : :
6261 : : /*
6262 : : * Release locks, if any. We do this for both two phase and normal one
6263 : : * phase transactions. In effect we are ignoring the prepare phase and
6264 : : * just going straight to lock release.
6265 : : */
3387 simon@2ndQuadrant.co 6266 [ + + ]: 21932 : if (parsed->xinfo & XACT_XINFO_HAS_AE_LOCKS)
2936 6267 : 10745 : StandbyReleaseLockTree(xid, parsed->nsubxacts, parsed->subxacts);
6268 : : }
6269 : :
4080 andres@anarazel.de 6270 [ + + ]: 24157 : if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
6271 : : {
6272 : : /* recover apply progress */
6273 : 20 : replorigin_advance(origin_id, parsed->origin_lsn, lsn,
6274 : : false /* backward */ , false /* WAL */ );
6275 : : }
6276 : :
6277 : : /* Make sure files supposed to be dropped are dropped */
4125 6278 [ + + ]: 24157 : if (parsed->nrels > 0)
6279 : : {
6280 : : /*
6281 : : * First update minimum recovery point to cover this WAL record. Once
6282 : : * a relation is deleted, there's no going back. The buffer manager
6283 : : * enforces the WAL-first rule for normal updates to relation files,
6284 : : * so that the minimum recovery point is always updated before the
6285 : : * corresponding change in the data file is flushed to disk, but we
6286 : : * have to do the same here since we're bypassing the buffer manager.
6287 : : *
6288 : : * Doing this before deleting the files means that if a deletion fails
6289 : : * for some reason, you cannot start up the system even after restart,
6290 : : * until you fix the underlying situation so that the deletion will
6291 : : * succeed. Alternatively, we could update the minimum recovery point
6292 : : * after deletion, but that would leave a small window where the
6293 : : * WAL-first rule would be violated.
6294 : : */
4950 heikki.linnakangas@i 6295 : 2329 : XLogFlush(lsn);
6296 : :
6297 : : /* Make sure files supposed to be dropped are dropped */
1455 rhaas@postgresql.org 6298 : 2329 : DropRelationFiles(parsed->xlocators, parsed->nrels, true);
6299 : : }
6300 : :
1546 andres@anarazel.de 6301 [ + + ]: 24157 : if (parsed->nstats > 0)
6302 : : {
6303 : : /* see equivalent call for relations above */
6304 : 3015 : XLogFlush(lsn);
6305 : :
6306 : 3015 : pgstat_execute_transactional_drops(parsed->nstats, parsed->stats, true);
6307 : : }
6308 : :
6309 : : /*
6310 : : * We issue an XLogFlush() for the same reason we emit ForceSyncCommit()
6311 : : * in normal operation. For example, in CREATE DATABASE, we copy all files
6312 : : * from the template database, and then commit the transaction. If we
6313 : : * crash after all the files have been copied but before the commit, you
6314 : : * have files in the data directory without an entry in pg_database. To
6315 : : * minimize the window for that, we use ForceSyncCommit() to rush the
6316 : : * commit record to disk as quick as possible. We have the same window
6317 : : * during recovery, and forcing an XLogFlush() (which updates
6318 : : * minRecoveryPoint during recovery) helps to reduce that problem window,
6319 : : * for any user that requested ForceSyncCommit().
6320 : : */
4125 6321 [ + + ]: 24157 : if (XactCompletionForceSyncCommit(parsed->xinfo))
6037 simon@2ndQuadrant.co 6322 : 51 : XLogFlush(lsn);
6323 : :
6324 : : /*
6325 : : * If asked by the primary (because someone is waiting for a synchronous
6326 : : * commit = remote_apply), we will need to ask walreceiver to send a reply
6327 : : * immediately.
6328 : : */
3745 rhaas@postgresql.org 6329 [ + + ]: 24157 : if (XactCompletionApplyFeedback(parsed->xinfo))
6330 : 2 : XLogRequestWalReceiverReply();
5481 simon@2ndQuadrant.co 6331 : 24157 : }
6332 : :
6333 : : /*
6334 : : * Be careful with the order of execution, as with xact_redo_commit().
6335 : : * The two functions are similar but differ in key places.
6336 : : *
6337 : : * Note also that an abort can be for a subtransaction and its children,
6338 : : * not just for a top level abort. That means we have to consider
6339 : : * topxid != xid, whereas in commit we would find topxid == xid always
6340 : : * because subtransaction commit is never WAL logged.
6341 : : */
6342 : : static void
1940 akapila@postgresql.o 6343 : 2031 : xact_redo_abort(xl_xact_parsed_abort *parsed, TransactionId xid,
6344 : : XLogRecPtr lsn, ReplOriginId origin_id)
6345 : : {
6346 : : TransactionId max_xid;
6347 : :
3303 alvherre@alvh.no-ip. 6348 [ - + ]: 2031 : Assert(TransactionIdIsValid(xid));
6349 : :
6350 : : /* Make sure nextXid is beyond any XID mentioned in the record. */
4125 andres@anarazel.de 6351 : 2031 : max_xid = TransactionIdLatest(xid,
6352 : : parsed->nsubxacts,
6353 : 2031 : parsed->subxacts);
2651 tmunro@postgresql.or 6354 : 2031 : AdvanceNextFullTransactionIdPastXid(max_xid);
6355 : :
5892 simon@2ndQuadrant.co 6356 [ + + ]: 2031 : if (standbyState == STANDBY_DISABLED)
6357 : : {
6358 : : /* Mark the transaction aborted in pg_xact, no need for async stuff */
4125 andres@anarazel.de 6359 : 20 : TransactionIdAbortTree(xid, parsed->nsubxacts, parsed->subxacts);
6360 : : }
6361 : : else
6362 : : {
6363 : : /*
6364 : : * If a transaction completion record arrives that has as-yet
6365 : : * unobserved subtransactions then this will not have been fully
6366 : : * handled by the call to RecordKnownAssignedTransactionIds() in the
6367 : : * main recovery loop in PerformWalRecovery(). So we need to do
6368 : : * bookkeeping again to cover that case. This is confusing and it is
6369 : : * easy to think this call is irrelevant, which has happened three
6370 : : * times in development already. Leave it in.
6371 : : */
6037 simon@2ndQuadrant.co 6372 : 2011 : RecordKnownAssignedTransactionIds(max_xid);
6373 : :
6374 : : /* Mark the transaction aborted in pg_xact, no need for async stuff */
4125 andres@anarazel.de 6375 : 2011 : TransactionIdAbortTree(xid, parsed->nsubxacts, parsed->subxacts);
6376 : :
6377 : : /*
6378 : : * We must update the ProcArray after we have marked clog.
6379 : : */
2343 alvherre@alvh.no-ip. 6380 : 2011 : ExpireTreeKnownAssignedTransactionIds(xid, parsed->nsubxacts, parsed->subxacts, max_xid);
6381 : :
6382 : : /*
6383 : : * There are no invalidation messages to send or undo.
6384 : : */
6385 : :
6386 : : /*
6387 : : * Release locks, if any. There are no invalidations to send.
6388 : : */
3387 simon@2ndQuadrant.co 6389 [ + + ]: 2011 : if (parsed->xinfo & XACT_XINFO_HAS_AE_LOCKS)
6390 : 1274 : StandbyReleaseLockTree(xid, parsed->nsubxacts, parsed->subxacts);
6391 : : }
6392 : :
1940 akapila@postgresql.o 6393 [ + + ]: 2031 : if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
6394 : : {
6395 : : /* recover apply progress */
6396 : 5 : replorigin_advance(origin_id, parsed->origin_lsn, lsn,
6397 : : false /* backward */ , false /* WAL */ );
6398 : : }
6399 : :
6400 : : /* Make sure files supposed to be dropped are dropped */
1797 fujii@postgresql.org 6401 [ + + ]: 2031 : if (parsed->nrels > 0)
6402 : : {
6403 : : /*
6404 : : * See comments about update of minimum recovery point on truncation,
6405 : : * in xact_redo_commit().
6406 : : */
6407 : 331 : XLogFlush(lsn);
6408 : :
1455 rhaas@postgresql.org 6409 : 331 : DropRelationFiles(parsed->xlocators, parsed->nrels, true);
6410 : : }
6411 : :
1546 andres@anarazel.de 6412 [ + + ]: 2031 : if (parsed->nstats > 0)
6413 : : {
6414 : : /* see equivalent call for relations above */
6415 : 457 : XLogFlush(lsn);
6416 : :
6417 : 457 : pgstat_execute_transactional_drops(parsed->nstats, parsed->stats, true);
6418 : : }
7683 tgl@sss.pgh.pa.us 6419 : 2031 : }
6420 : :
6421 : : void
4240 heikki.linnakangas@i 6422 : 26561 : xact_redo(XLogReaderState *record)
6423 : : {
4125 andres@anarazel.de 6424 : 26561 : uint8 info = XLogRecGetInfo(record) & XLOG_XACT_OPMASK;
6425 : :
6426 : : /* Backup blocks are not used in xact records */
4240 heikki.linnakangas@i 6427 [ - + ]: 26561 : Assert(!XLogRecHasAnyBlockRefs(record));
6428 : :
3303 alvherre@alvh.no-ip. 6429 [ + + ]: 26561 : if (info == XLOG_XACT_COMMIT)
6430 : : {
8175 tgl@sss.pgh.pa.us 6431 : 24111 : xl_xact_commit *xlrec = (xl_xact_commit *) XLogRecGetData(record);
6432 : : xl_xact_parsed_commit parsed;
6433 : :
3303 alvherre@alvh.no-ip. 6434 : 24111 : ParseCommitRecord(XLogRecGetInfo(record), xlrec, &parsed);
6435 : 24111 : xact_redo_commit(&parsed, XLogRecGetXid(record),
6436 : 24111 : record->EndRecPtr, XLogRecGetOrigin(record));
6437 : : }
6438 [ + + ]: 2450 : else if (info == XLOG_XACT_COMMIT_PREPARED)
6439 : : {
6440 : 46 : xl_xact_commit *xlrec = (xl_xact_commit *) XLogRecGetData(record);
6441 : : xl_xact_parsed_commit parsed;
6442 : :
6443 : 46 : ParseCommitRecord(XLogRecGetInfo(record), xlrec, &parsed);
6444 : 46 : xact_redo_commit(&parsed, parsed.twophase_xid,
6445 : 46 : record->EndRecPtr, XLogRecGetOrigin(record));
6446 : :
6447 : : /* Delete TwoPhaseState gxact entry and/or 2PC file. */
6448 : 46 : LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
6449 : 46 : PrepareRedoRemove(parsed.twophase_xid, false);
6450 : 46 : LWLockRelease(TwoPhaseStateLock);
6451 : : }
6452 [ + + ]: 2404 : else if (info == XLOG_XACT_ABORT)
6453 : : {
7683 tgl@sss.pgh.pa.us 6454 : 2006 : xl_xact_abort *xlrec = (xl_xact_abort *) XLogRecGetData(record);
6455 : : xl_xact_parsed_abort parsed;
6456 : :
3303 alvherre@alvh.no-ip. 6457 : 2006 : ParseAbortRecord(XLogRecGetInfo(record), xlrec, &parsed);
1940 akapila@postgresql.o 6458 : 2006 : xact_redo_abort(&parsed, XLogRecGetXid(record),
6459 : 2006 : record->EndRecPtr, XLogRecGetOrigin(record));
6460 : : }
3303 alvherre@alvh.no-ip. 6461 [ + + ]: 398 : else if (info == XLOG_XACT_ABORT_PREPARED)
6462 : : {
6463 : 25 : xl_xact_abort *xlrec = (xl_xact_abort *) XLogRecGetData(record);
6464 : : xl_xact_parsed_abort parsed;
6465 : :
6466 : 25 : ParseAbortRecord(XLogRecGetInfo(record), xlrec, &parsed);
1940 akapila@postgresql.o 6467 : 25 : xact_redo_abort(&parsed, parsed.twophase_xid,
6468 : 25 : record->EndRecPtr, XLogRecGetOrigin(record));
6469 : :
6470 : : /* Delete TwoPhaseState gxact entry and/or 2PC file. */
3303 alvherre@alvh.no-ip. 6471 : 25 : LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
6472 : 25 : PrepareRedoRemove(parsed.twophase_xid, false);
6473 : 25 : LWLockRelease(TwoPhaseStateLock);
6474 : : }
7683 tgl@sss.pgh.pa.us 6475 [ + + ]: 373 : else if (info == XLOG_XACT_PREPARE)
6476 : : {
6477 : : /*
6478 : : * Store xid and start/end pointers of the WAL record in TwoPhaseState
6479 : : * gxact entry.
6480 : : */
3303 alvherre@alvh.no-ip. 6481 : 81 : LWLockAcquire(TwoPhaseStateLock, LW_EXCLUSIVE);
358 michael@paquier.xyz 6482 :GNC 81 : PrepareRedoAdd(InvalidFullTransactionId,
6483 : 81 : XLogRecGetData(record),
6484 : : record->ReadRecPtr,
6485 : : record->EndRecPtr,
3016 simon@2ndQuadrant.co 6486 :CBC 81 : XLogRecGetOrigin(record));
3303 alvherre@alvh.no-ip. 6487 : 81 : LWLockRelease(TwoPhaseStateLock);
6488 : : }
6037 simon@2ndQuadrant.co 6489 [ + + ]: 292 : else if (info == XLOG_XACT_ASSIGNMENT)
6490 : : {
6491 : 21 : xl_xact_assignment *xlrec = (xl_xact_assignment *) XLogRecGetData(record);
6492 : :
5892 6493 [ + - ]: 21 : if (standbyState >= STANDBY_INITIALIZED)
6037 6494 : 21 : ProcArrayApplyXidAssignment(xlrec->xtop,
6495 : 21 : xlrec->nsubxacts, xlrec->xsub);
6496 : : }
2168 akapila@postgresql.o 6497 [ - + ]: 271 : else if (info == XLOG_XACT_INVALIDATIONS)
6498 : : {
6499 : : /*
6500 : : * XXX we do ignore this for now, what matters are invalidations
6501 : : * written into the commit record.
6502 : : */
6503 : : }
6504 : : else
7683 tgl@sss.pgh.pa.us 6505 [ # # ]:UBC 0 : elog(PANIC, "xact_redo: unknown op code %u", info);
7683 tgl@sss.pgh.pa.us 6506 :CBC 26561 : }
|