Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * spi.c
4 : : * Server Programming Interface
5 : : *
6 : : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : : * Portions Copyright (c) 1994, Regents of the University of California
8 : : *
9 : : *
10 : : * IDENTIFICATION
11 : : * src/backend/executor/spi.c
12 : : *
13 : : *-------------------------------------------------------------------------
14 : : */
15 : : #include "postgres.h"
16 : :
17 : : #include "access/htup_details.h"
18 : : #include "access/printtup.h"
19 : : #include "access/sysattr.h"
20 : : #include "access/xact.h"
21 : : #include "catalog/heap.h"
22 : : #include "catalog/pg_type.h"
23 : : #include "commands/trigger.h"
24 : : #include "executor/executor.h"
25 : : #include "executor/spi_priv.h"
26 : : #include "tcop/pquery.h"
27 : : #include "tcop/utility.h"
28 : : #include "utils/builtins.h"
29 : : #include "utils/datum.h"
30 : : #include "utils/lsyscache.h"
31 : : #include "utils/memutils.h"
32 : : #include "utils/rel.h"
33 : : #include "utils/snapmgr.h"
34 : : #include "utils/syscache.h"
35 : : #include "utils/typcache.h"
36 : :
37 : :
38 : : /*
39 : : * These global variables are part of the API for various SPI functions
40 : : * (a horrible API choice, but it's too late now). To reduce the risk of
41 : : * interference between different SPI callers, we save and restore them
42 : : * when entering/exiting a SPI nesting level.
43 : : */
44 : : uint64 SPI_processed = 0;
45 : : SPITupleTable *SPI_tuptable = NULL;
46 : : int SPI_result = 0;
47 : :
48 : : static _SPI_connection *_SPI_stack = NULL;
49 : : static _SPI_connection *_SPI_current = NULL;
50 : : static int _SPI_stack_depth = 0; /* allocated size of _SPI_stack */
51 : : static int _SPI_connected = -1; /* current stack index */
52 : :
53 : : typedef struct SPICallbackArg
54 : : {
55 : : const char *query;
56 : : RawParseMode mode;
57 : : } SPICallbackArg;
58 : :
59 : : static Portal SPI_cursor_open_internal(const char *name, SPIPlanPtr plan,
60 : : ParamListInfo paramLI, bool read_only);
61 : :
62 : : static void _SPI_prepare_plan(const char *src, SPIPlanPtr plan);
63 : :
64 : : static void _SPI_prepare_oneshot_plan(const char *src, SPIPlanPtr plan);
65 : :
66 : : static int _SPI_execute_plan(SPIPlanPtr plan, const SPIExecuteOptions *options,
67 : : Snapshot snapshot, Snapshot crosscheck_snapshot,
68 : : bool fire_triggers);
69 : :
70 : : static ParamListInfo _SPI_convert_params(int nargs, Oid *argtypes,
71 : : const Datum *Values, const char *Nulls);
72 : :
73 : : static int _SPI_pquery(QueryDesc *queryDesc, bool fire_triggers, uint64 tcount);
74 : :
75 : : static void _SPI_error_callback(void *arg);
76 : :
77 : : static void _SPI_cursor_operation(Portal portal,
78 : : FetchDirection direction, long count,
79 : : DestReceiver *dest);
80 : :
81 : : static SPIPlanPtr _SPI_make_plan_non_temp(SPIPlanPtr plan);
82 : : static SPIPlanPtr _SPI_save_plan(SPIPlanPtr plan);
83 : :
84 : : static int _SPI_begin_call(bool use_exec);
85 : : static int _SPI_end_call(bool use_exec);
86 : : static MemoryContext _SPI_execmem(void);
87 : : static MemoryContext _SPI_procmem(void);
88 : : static bool _SPI_checktuples(void);
89 : :
90 : :
91 : : /* =================== interface functions =================== */
92 : :
93 : : int
8964 tgl@sss.pgh.pa.us 94 :CBC 8719 : SPI_connect(void)
95 : : {
2885 peter_e@gmx.net 96 : 8719 : return SPI_connect_ext(0);
97 : : }
98 : :
99 : : int
100 : 52270 : SPI_connect_ext(int options)
101 : : {
102 : : int newdepth;
103 : :
104 : : /* Enlarge stack if necessary */
10327 bruce@momjian.us 105 [ + + ]: 52270 : if (_SPI_stack == NULL)
106 : : {
7838 tgl@sss.pgh.pa.us 107 [ + - - + ]: 1008 : if (_SPI_connected != -1 || _SPI_stack_depth != 0)
8184 tgl@sss.pgh.pa.us 108 [ # # ]:UBC 0 : elog(ERROR, "SPI stack corrupted");
7838 tgl@sss.pgh.pa.us 109 :CBC 1008 : newdepth = 16;
110 : 1008 : _SPI_stack = (_SPI_connection *)
2885 peter_e@gmx.net 111 : 1008 : MemoryContextAlloc(TopMemoryContext,
112 : : newdepth * sizeof(_SPI_connection));
7838 tgl@sss.pgh.pa.us 113 : 1008 : _SPI_stack_depth = newdepth;
114 : : }
115 : : else
116 : : {
117 [ + - - + ]: 51262 : if (_SPI_stack_depth <= 0 || _SPI_stack_depth <= _SPI_connected)
8184 tgl@sss.pgh.pa.us 118 [ # # ]:UBC 0 : elog(ERROR, "SPI stack corrupted");
7838 tgl@sss.pgh.pa.us 119 [ + + ]:CBC 51262 : if (_SPI_stack_depth == _SPI_connected + 1)
120 : : {
121 : 13 : newdepth = _SPI_stack_depth * 2;
122 : 13 : _SPI_stack = (_SPI_connection *)
123 : 13 : repalloc(_SPI_stack,
124 : : newdepth * sizeof(_SPI_connection));
125 : 13 : _SPI_stack_depth = newdepth;
126 : : }
127 : : }
128 : :
129 : : /* Enter new stack level */
10327 bruce@momjian.us 130 : 52270 : _SPI_connected++;
7838 tgl@sss.pgh.pa.us 131 [ + - - + ]: 52270 : Assert(_SPI_connected >= 0 && _SPI_connected < _SPI_stack_depth);
132 : :
10327 bruce@momjian.us 133 : 52270 : _SPI_current = &(_SPI_stack[_SPI_connected]);
134 : 52270 : _SPI_current->processed = 0;
135 : 52270 : _SPI_current->tuptable = NULL;
2993 tgl@sss.pgh.pa.us 136 : 52270 : _SPI_current->execSubid = InvalidSubTransactionId;
4527 137 : 52270 : slist_init(&_SPI_current->tuptables);
3100 138 : 52270 : _SPI_current->procCxt = NULL; /* in case we fail to create 'em */
7761 139 : 52270 : _SPI_current->execCxt = NULL;
140 : 52270 : _SPI_current->connectSubid = GetCurrentSubTransactionId();
3182 kgrittn@postgresql.o 141 : 52270 : _SPI_current->queryEnv = NULL;
2885 peter_e@gmx.net 142 : 52270 : _SPI_current->atomic = (options & SPI_OPT_NONATOMIC ? false : true);
143 : 52270 : _SPI_current->internal_xact = false;
2657 tgl@sss.pgh.pa.us 144 : 52270 : _SPI_current->outer_processed = SPI_processed;
145 : 52270 : _SPI_current->outer_tuptable = SPI_tuptable;
146 : 52270 : _SPI_current->outer_result = SPI_result;
147 : :
148 : : /*
149 : : * Create memory contexts for this procedure
150 : : *
151 : : * In atomic contexts (the normal case), we use TopTransactionContext,
152 : : * otherwise PortalContext, so that it lives across transaction
153 : : * boundaries.
154 : : *
155 : : * XXX It could be better to use PortalContext as the parent context in
156 : : * all cases, but we may not be inside a portal (consider deferred-trigger
157 : : * execution). Perhaps CurTransactionContext could be an option? For now
158 : : * it doesn't matter because we clean up explicitly in AtEOSubXact_SPI();
159 : : * but see also AtEOXact_SPI().
160 : : */
2885 peter_e@gmx.net 161 [ + + ]: 52270 : _SPI_current->procCxt = AllocSetContextCreate(_SPI_current->atomic ? TopTransactionContext : PortalContext,
162 : : "SPI Proc",
163 : : ALLOCSET_DEFAULT_SIZES);
164 [ + + ]: 52270 : _SPI_current->execCxt = AllocSetContextCreate(_SPI_current->atomic ? TopTransactionContext : _SPI_current->procCxt,
165 : : "SPI Exec",
166 : : ALLOCSET_DEFAULT_SIZES);
167 : : /* ... and switch to procedure's context */
9302 tgl@sss.pgh.pa.us 168 : 52270 : _SPI_current->savedcxt = MemoryContextSwitchTo(_SPI_current->procCxt);
169 : :
170 : : /*
171 : : * Reset API global variables so that current caller cannot accidentally
172 : : * depend on state of an outer caller.
173 : : */
2657 174 : 52270 : SPI_processed = 0;
175 : 52270 : SPI_tuptable = NULL;
176 : 52270 : SPI_result = 0;
177 : :
9968 bruce@momjian.us 178 : 52270 : return SPI_OK_CONNECT;
179 : : }
180 : :
181 : : int
8964 tgl@sss.pgh.pa.us 182 : 50823 : SPI_finish(void)
183 : : {
184 : : int res;
185 : :
2993 186 : 50823 : res = _SPI_begin_call(false); /* just check we're connected */
10327 bruce@momjian.us 187 [ - + ]: 50823 : if (res < 0)
9968 bruce@momjian.us 188 :UBC 0 : return res;
189 : :
190 : : /* Restore memory context as it was before procedure call */
10327 bruce@momjian.us 191 :CBC 50823 : MemoryContextSwitchTo(_SPI_current->savedcxt);
192 : :
193 : : /* Release memory used in procedure call (including tuptables) */
9302 tgl@sss.pgh.pa.us 194 : 50823 : MemoryContextDelete(_SPI_current->execCxt);
7761 195 : 50823 : _SPI_current->execCxt = NULL;
9302 196 : 50823 : MemoryContextDelete(_SPI_current->procCxt);
7761 197 : 50823 : _SPI_current->procCxt = NULL;
198 : :
199 : : /*
200 : : * Restore outer API variables, especially SPI_tuptable which is probably
201 : : * pointing at a just-deleted tuptable
202 : : */
2657 203 : 50823 : SPI_processed = _SPI_current->outer_processed;
204 : 50823 : SPI_tuptable = _SPI_current->outer_tuptable;
205 : 50823 : SPI_result = _SPI_current->outer_result;
206 : :
207 : : /* Exit stack level */
10327 bruce@momjian.us 208 : 50823 : _SPI_connected--;
3325 tgl@sss.pgh.pa.us 209 [ + + ]: 50823 : if (_SPI_connected < 0)
9302 210 : 44679 : _SPI_current = NULL;
211 : : else
10327 bruce@momjian.us 212 : 6144 : _SPI_current = &(_SPI_stack[_SPI_connected]);
213 : :
9968 214 : 50823 : return SPI_OK_FINISH;
215 : : }
216 : :
217 : : /*
218 : : * SPI_start_transaction is a no-op, kept for backwards compatibility.
219 : : * SPI callers are *always* inside a transaction.
220 : : */
221 : : void
2885 peter_e@gmx.net 222 :UBC 0 : SPI_start_transaction(void)
223 : : {
224 : 0 : }
225 : :
226 : : static void
2459 peter@eisentraut.org 227 :CBC 2144 : _SPI_commit(bool chain)
228 : : {
2885 peter_e@gmx.net 229 : 2144 : MemoryContext oldcontext = CurrentMemoryContext;
230 : : SavedTransactionCharacteristics savetc;
231 : :
232 : : /*
233 : : * Complain if we are in a context that doesn't permit transaction
234 : : * termination. (Note: here and _SPI_rollback should be the only places
235 : : * that throw ERRCODE_INVALID_TRANSACTION_TERMINATION, so that callers can
236 : : * test for that with security that they know what happened.)
237 : : */
238 [ + + ]: 2144 : if (_SPI_current->atomic)
239 [ + - ]: 16 : ereport(ERROR,
240 : : (errcode(ERRCODE_INVALID_TRANSACTION_TERMINATION),
241 : : errmsg("invalid transaction termination")));
242 : :
243 : : /*
244 : : * This restriction is required by PLs implemented on top of SPI. They
245 : : * use subtransactions to establish exception blocks that are supposed to
246 : : * be rolled back together if there is an error. Terminating the
247 : : * top-level transaction in such a block violates that idea. A future PL
248 : : * implementation might have different ideas about this, in which case
249 : : * this restriction would have to be refined or the check possibly be
250 : : * moved out of SPI into the PLs. Note however that the code below relies
251 : : * on not being within a subtransaction.
252 : : */
253 [ + + ]: 2128 : if (IsSubTransaction())
254 [ + - ]: 3 : ereport(ERROR,
255 : : (errcode(ERRCODE_INVALID_TRANSACTION_TERMINATION),
256 : : errmsg("cannot commit while a subtransaction is active")));
257 : :
1387 tgl@sss.pgh.pa.us 258 [ + + ]: 2125 : if (chain)
259 : 2 : SaveTransactionCharacteristics(&savetc);
260 : :
261 : : /* Catch any error occurring during the COMMIT */
262 [ + + ]: 2125 : PG_TRY();
263 : : {
264 : : /* Protect current SPI stack entry against deletion */
265 : 2125 : _SPI_current->internal_xact = true;
266 : :
267 : : /*
268 : : * Hold any pinned portals that any PLs might be using. We have to do
269 : : * this before changing transaction state, since this will run
270 : : * user-defined code that might throw an error.
271 : : */
272 : 2125 : HoldPinnedPortals();
273 : :
274 : : /* Release snapshots associated with portals */
275 : 2124 : ForgetPortalSnapshots();
276 : :
277 : : /* Do the deed */
278 : 2124 : CommitTransactionCommand();
279 : :
280 : : /* Immediately start a new transaction */
2459 peter@eisentraut.org 281 : 2117 : StartTransactionCommand();
1387 tgl@sss.pgh.pa.us 282 [ + + ]: 2117 : if (chain)
283 : 2 : RestoreTransactionCharacteristics(&savetc);
284 : :
285 : 2117 : MemoryContextSwitchTo(oldcontext);
286 : :
287 : 2117 : _SPI_current->internal_xact = false;
288 : : }
289 : 8 : PG_CATCH();
290 : : {
291 : : ErrorData *edata;
292 : :
293 : : /* Save error info in caller's context */
294 : 8 : MemoryContextSwitchTo(oldcontext);
295 : 8 : edata = CopyErrorData();
296 : 8 : FlushErrorState();
297 : :
298 : : /*
299 : : * Abort the failed transaction. If this fails too, we'll just
300 : : * propagate the error out ... there's not that much we can do.
301 : : */
302 : 8 : AbortCurrentTransaction();
303 : :
304 : : /* ... and start a new one */
305 : 8 : StartTransactionCommand();
306 [ - + ]: 8 : if (chain)
1387 tgl@sss.pgh.pa.us 307 :UBC 0 : RestoreTransactionCharacteristics(&savetc);
308 : :
1387 tgl@sss.pgh.pa.us 309 :CBC 8 : MemoryContextSwitchTo(oldcontext);
310 : :
311 : 8 : _SPI_current->internal_xact = false;
312 : :
313 : : /* Now that we've cleaned up the transaction, re-throw the error */
314 : 8 : ReThrowError(edata);
315 : : }
316 [ - + ]: 2117 : PG_END_TRY();
2885 peter_e@gmx.net 317 : 2117 : }
318 : :
319 : : void
2459 peter@eisentraut.org 320 : 2142 : SPI_commit(void)
321 : : {
322 : 2142 : _SPI_commit(false);
323 : 2115 : }
324 : :
325 : : void
326 : 2 : SPI_commit_and_chain(void)
327 : : {
328 : 2 : _SPI_commit(true);
329 : 2 : }
330 : :
331 : : static void
332 : 84 : _SPI_rollback(bool chain)
333 : : {
2885 peter_e@gmx.net 334 : 84 : MemoryContext oldcontext = CurrentMemoryContext;
335 : : SavedTransactionCharacteristics savetc;
336 : :
337 : : /* see comments in _SPI_commit() */
338 [ - + ]: 84 : if (_SPI_current->atomic)
2885 peter_e@gmx.net 339 [ # # ]:UBC 0 : ereport(ERROR,
340 : : (errcode(ERRCODE_INVALID_TRANSACTION_TERMINATION),
341 : : errmsg("invalid transaction termination")));
342 : :
343 : : /* see comments in _SPI_commit() */
2885 peter_e@gmx.net 344 [ + + ]:CBC 84 : if (IsSubTransaction())
345 [ + - ]: 2 : ereport(ERROR,
346 : : (errcode(ERRCODE_INVALID_TRANSACTION_TERMINATION),
347 : : errmsg("cannot roll back while a subtransaction is active")));
348 : :
1387 tgl@sss.pgh.pa.us 349 [ + + ]: 82 : if (chain)
350 : 2 : SaveTransactionCharacteristics(&savetc);
351 : :
352 : : /* Catch any error occurring during the ROLLBACK */
353 [ + + ]: 82 : PG_TRY();
354 : : {
355 : : /* Protect current SPI stack entry against deletion */
356 : 82 : _SPI_current->internal_xact = true;
357 : :
358 : : /*
359 : : * Hold any pinned portals that any PLs might be using. We have to do
360 : : * this before changing transaction state, since this will run
361 : : * user-defined code that might throw an error, and in any case
362 : : * couldn't be run in an already-aborted transaction.
363 : : */
364 : 82 : HoldPinnedPortals();
365 : :
366 : : /* Release snapshots associated with portals */
367 : 80 : ForgetPortalSnapshots();
368 : :
369 : : /* Do the deed */
370 : 80 : AbortCurrentTransaction();
371 : :
372 : : /* Immediately start a new transaction */
2459 peter@eisentraut.org 373 : 80 : StartTransactionCommand();
1387 tgl@sss.pgh.pa.us 374 [ + + ]: 80 : if (chain)
375 : 2 : RestoreTransactionCharacteristics(&savetc);
376 : :
377 : 80 : MemoryContextSwitchTo(oldcontext);
378 : :
379 : 80 : _SPI_current->internal_xact = false;
380 : : }
381 : 2 : PG_CATCH();
382 : : {
383 : : ErrorData *edata;
384 : :
385 : : /* Save error info in caller's context */
386 : 2 : MemoryContextSwitchTo(oldcontext);
387 : 2 : edata = CopyErrorData();
388 : 2 : FlushErrorState();
389 : :
390 : : /*
391 : : * Try again to abort the failed transaction. If this fails too,
392 : : * we'll just propagate the error out ... there's not that much we can
393 : : * do.
394 : : */
395 : 2 : AbortCurrentTransaction();
396 : :
397 : : /* ... and start a new one */
398 : 2 : StartTransactionCommand();
399 [ - + ]: 2 : if (chain)
1387 tgl@sss.pgh.pa.us 400 :UBC 0 : RestoreTransactionCharacteristics(&savetc);
401 : :
1387 tgl@sss.pgh.pa.us 402 :CBC 2 : MemoryContextSwitchTo(oldcontext);
403 : :
404 : 2 : _SPI_current->internal_xact = false;
405 : :
406 : : /* Now that we've cleaned up the transaction, re-throw the error */
407 : 2 : ReThrowError(edata);
408 : : }
409 [ - + ]: 80 : PG_END_TRY();
2885 peter_e@gmx.net 410 : 80 : }
411 : :
412 : : void
2459 peter@eisentraut.org 413 : 82 : SPI_rollback(void)
414 : : {
415 : 82 : _SPI_rollback(false);
416 : 78 : }
417 : :
418 : : void
419 : 2 : SPI_rollback_and_chain(void)
420 : : {
421 : 2 : _SPI_rollback(true);
422 : 2 : }
423 : :
424 : : /*
425 : : * Clean up SPI state at transaction commit or abort.
426 : : */
427 : : void
8050 mail@joeconway.com 428 : 330688 : AtEOXact_SPI(bool isCommit)
429 : : {
1387 tgl@sss.pgh.pa.us 430 : 330688 : bool found = false;
431 : :
432 : : /*
433 : : * Pop stack entries, stopping if we find one marked internal_xact (that
434 : : * one belongs to the caller of SPI_commit or SPI_rollback).
435 : : */
436 [ + + ]: 332024 : while (_SPI_connected >= 0)
437 : : {
438 : 3543 : _SPI_connection *connection = &(_SPI_stack[_SPI_connected]);
439 : :
440 [ + + ]: 3543 : if (connection->internal_xact)
441 : 2207 : break;
442 : :
443 : 1336 : found = true;
444 : :
445 : : /*
446 : : * We need not release the procedure's memory contexts explicitly, as
447 : : * they'll go away automatically when their parent context does; see
448 : : * notes in SPI_connect_ext.
449 : : */
450 : :
451 : : /*
452 : : * Restore outer global variables and pop the stack entry. Unlike
453 : : * SPI_finish(), we don't risk switching to memory contexts that might
454 : : * be already gone.
455 : : */
456 : 1336 : SPI_processed = connection->outer_processed;
457 : 1336 : SPI_tuptable = connection->outer_tuptable;
458 : 1336 : SPI_result = connection->outer_result;
459 : :
460 : 1336 : _SPI_connected--;
461 [ + + ]: 1336 : if (_SPI_connected < 0)
462 : 1304 : _SPI_current = NULL;
463 : : else
464 : 32 : _SPI_current = &(_SPI_stack[_SPI_connected]);
465 : : }
466 : :
467 : : /* We should only find entries to pop during an ABORT. */
468 [ + + - + ]: 330688 : if (found && isCommit)
7838 tgl@sss.pgh.pa.us 469 [ # # ]:UBC 0 : ereport(WARNING,
470 : : (errcode(ERRCODE_WARNING),
471 : : errmsg("transaction left non-empty SPI stack"),
472 : : errhint("Check for missing \"SPI_finish\" calls.")));
9302 tgl@sss.pgh.pa.us 473 :CBC 330688 : }
474 : :
475 : : /*
476 : : * Clean up SPI state at subtransaction commit or abort.
477 : : *
478 : : * During commit, there shouldn't be any unclosed entries remaining from
479 : : * the current subtransaction; we emit a warning if any are found.
480 : : */
481 : : void
7761 482 : 9102 : AtEOSubXact_SPI(bool isCommit, SubTransactionId mySubid)
483 : : {
7779 bruce@momjian.us 484 : 9102 : bool found = false;
485 : :
7838 tgl@sss.pgh.pa.us 486 [ + + ]: 9213 : while (_SPI_connected >= 0)
487 : : {
488 : 6976 : _SPI_connection *connection = &(_SPI_stack[_SPI_connected]);
489 : :
7761 490 [ + + ]: 6976 : if (connection->connectSubid != mySubid)
7838 491 : 6865 : break; /* couldn't be any underneath it either */
492 : :
2885 peter_e@gmx.net 493 [ - + ]: 111 : if (connection->internal_xact)
2885 peter_e@gmx.net 494 :UBC 0 : break;
495 : :
7838 tgl@sss.pgh.pa.us 496 :CBC 111 : found = true;
497 : :
498 : : /*
499 : : * Release procedure memory explicitly (see note in SPI_connect)
500 : : */
7761 501 [ + - ]: 111 : if (connection->execCxt)
502 : : {
503 : 111 : MemoryContextDelete(connection->execCxt);
504 : 111 : connection->execCxt = NULL;
505 : : }
506 [ + - ]: 111 : if (connection->procCxt)
507 : : {
508 : 111 : MemoryContextDelete(connection->procCxt);
509 : 111 : connection->procCxt = NULL;
510 : : }
511 : :
512 : : /*
513 : : * Restore outer global variables and pop the stack entry. Unlike
514 : : * SPI_finish(), we don't risk switching to memory contexts that might
515 : : * be already gone.
516 : : */
2657 517 : 111 : SPI_processed = connection->outer_processed;
518 : 111 : SPI_tuptable = connection->outer_tuptable;
519 : 111 : SPI_result = connection->outer_result;
520 : :
7838 521 : 111 : _SPI_connected--;
3325 522 [ + + ]: 111 : if (_SPI_connected < 0)
7838 523 : 39 : _SPI_current = NULL;
524 : : else
525 : 72 : _SPI_current = &(_SPI_stack[_SPI_connected]);
526 : : }
527 : :
528 [ + + - + ]: 9102 : if (found && isCommit)
7838 tgl@sss.pgh.pa.us 529 [ # # ]:UBC 0 : ereport(WARNING,
530 : : (errcode(ERRCODE_WARNING),
531 : : errmsg("subtransaction left non-empty SPI stack"),
532 : : errhint("Check for missing \"SPI_finish\" calls.")));
533 : :
534 : : /*
535 : : * If we are aborting a subtransaction and there is an open SPI context
536 : : * surrounding the subxact, clean up to prevent memory leakage.
537 : : */
6965 tgl@sss.pgh.pa.us 538 [ + + + + ]:CBC 9102 : if (_SPI_current && !isCommit)
539 : : {
540 : : slist_mutable_iter siter;
541 : :
542 : : /*
543 : : * Throw away executor state if current executor operation was started
544 : : * within current subxact (essentially, force a _SPI_end_call(true)).
545 : : */
2993 546 [ + + ]: 3179 : if (_SPI_current->execSubid >= mySubid)
547 : : {
548 : 2783 : _SPI_current->execSubid = InvalidSubTransactionId;
762 nathan@postgresql.or 549 : 2783 : MemoryContextReset(_SPI_current->execCxt);
550 : : }
551 : :
552 : : /* throw away any tuple tables created within current subxact */
4527 tgl@sss.pgh.pa.us 553 [ + + + + : 7701 : slist_foreach_modify(siter, &_SPI_current->tuptables)
+ + ]
554 : : {
555 : : SPITupleTable *tuptable;
556 : :
557 : 4522 : tuptable = slist_container(SPITupleTable, next, siter.cur);
558 [ + + ]: 4522 : if (tuptable->subid >= mySubid)
559 : : {
560 : : /*
561 : : * If we used SPI_freetuptable() here, its internal search of
562 : : * the tuptables list would make this operation O(N^2).
563 : : * Instead, just free the tuptable manually. This should
564 : : * match what SPI_freetuptable() does.
565 : : */
566 : 2668 : slist_delete_current(&siter);
567 [ + + ]: 2668 : if (tuptable == _SPI_current->tuptable)
568 : 2665 : _SPI_current->tuptable = NULL;
569 [ + + ]: 2668 : if (tuptable == SPI_tuptable)
570 : 3 : SPI_tuptable = NULL;
571 : 2668 : MemoryContextDelete(tuptable->tuptabcxt);
572 : : }
573 : : }
574 : : }
7838 575 : 9102 : }
576 : :
577 : : /*
578 : : * Are we executing inside a procedure (that is, a nonatomic SPI context)?
579 : : */
580 : : bool
2626 581 : 326353 : SPI_inside_nonatomic_context(void)
582 : : {
583 [ + + ]: 326353 : if (_SPI_current == NULL)
584 : 324146 : return false; /* not in any SPI context at all */
585 : : /* these tests must match _SPI_commit's opinion of what's atomic: */
586 [ - + ]: 2207 : if (_SPI_current->atomic)
2626 tgl@sss.pgh.pa.us 587 :UBC 0 : return false; /* it's atomic (ie function not procedure) */
426 tgl@sss.pgh.pa.us 588 [ - + ]:CBC 2207 : if (IsSubTransaction())
426 tgl@sss.pgh.pa.us 589 :UBC 0 : return false; /* if within subtransaction, it's atomic */
2626 tgl@sss.pgh.pa.us 590 :CBC 2207 : return true;
591 : : }
592 : :
593 : :
594 : : /* Parse, plan, and execute a query string */
595 : : int
7533 neilc@samurai.com 596 : 744 : SPI_execute(const char *src, bool read_only, long tcount)
597 : : {
598 : : _SPI_plan plan;
599 : : SPIExecuteOptions options;
600 : : int res;
601 : :
10327 bruce@momjian.us 602 [ + - - + ]: 744 : if (src == NULL || tcount < 0)
9968 bruce@momjian.us 603 :UBC 0 : return SPI_ERROR_ARGUMENT;
604 : :
10327 bruce@momjian.us 605 :CBC 744 : res = _SPI_begin_call(true);
606 [ - + ]: 744 : if (res < 0)
9968 bruce@momjian.us 607 :UBC 0 : return res;
608 : :
6851 tgl@sss.pgh.pa.us 609 :CBC 744 : memset(&plan, 0, sizeof(_SPI_plan));
610 : 744 : plan.magic = _SPI_PLAN_MAGIC;
1807 611 : 744 : plan.parse_mode = RAW_PARSE_DEFAULT;
3189 rhaas@postgresql.org 612 : 744 : plan.cursor_options = CURSOR_OPT_PARALLEL_OK;
613 : :
4729 tgl@sss.pgh.pa.us 614 : 744 : _SPI_prepare_oneshot_plan(src, &plan);
615 : :
1535 616 : 738 : memset(&options, 0, sizeof(options));
617 : 738 : options.read_only = read_only;
618 : 738 : options.tcount = tcount;
619 : :
620 : 738 : res = _SPI_execute_plan(&plan, &options,
621 : : InvalidSnapshot, InvalidSnapshot,
622 : : true);
623 : :
10327 bruce@momjian.us 624 : 712 : _SPI_end_call(true);
9968 625 : 712 : return res;
626 : : }
627 : :
628 : : /* Obsolete version of SPI_execute */
629 : : int
7533 neilc@samurai.com 630 : 265 : SPI_exec(const char *src, long tcount)
631 : : {
7764 tgl@sss.pgh.pa.us 632 : 265 : return SPI_execute(src, false, tcount);
633 : : }
634 : :
635 : : /* Parse, plan, and execute a query string, with extensible options */
636 : : int
1785 637 : 7267 : SPI_execute_extended(const char *src,
638 : : const SPIExecuteOptions *options)
639 : : {
640 : : int res;
641 : : _SPI_plan plan;
642 : :
643 [ + - - + ]: 7267 : if (src == NULL || options == NULL)
1785 tgl@sss.pgh.pa.us 644 :UBC 0 : return SPI_ERROR_ARGUMENT;
645 : :
1785 tgl@sss.pgh.pa.us 646 :CBC 7267 : res = _SPI_begin_call(true);
647 [ - + ]: 7267 : if (res < 0)
1785 tgl@sss.pgh.pa.us 648 :UBC 0 : return res;
649 : :
1785 tgl@sss.pgh.pa.us 650 :CBC 7267 : memset(&plan, 0, sizeof(_SPI_plan));
651 : 7267 : plan.magic = _SPI_PLAN_MAGIC;
652 : 7267 : plan.parse_mode = RAW_PARSE_DEFAULT;
653 : 7267 : plan.cursor_options = CURSOR_OPT_PARALLEL_OK;
654 [ + + ]: 7267 : if (options->params)
655 : : {
656 : 285 : plan.parserSetup = options->params->parserSetup;
657 : 285 : plan.parserSetupArg = options->params->parserSetupArg;
658 : : }
659 : :
660 : 7267 : _SPI_prepare_oneshot_plan(src, &plan);
661 : :
1535 662 : 7267 : res = _SPI_execute_plan(&plan, options,
663 : : InvalidSnapshot, InvalidSnapshot,
664 : : true);
665 : :
1785 666 : 7157 : _SPI_end_call(true);
667 : 7157 : return res;
668 : : }
669 : :
670 : : /* Execute a previously prepared plan */
671 : : int
46 peter@eisentraut.org 672 :GNC 2220 : SPI_execute_plan(SPIPlanPtr plan, const Datum *Values, const char *Nulls,
673 : : bool read_only, long tcount)
674 : : {
675 : : SPIExecuteOptions options;
676 : : int res;
677 : :
6851 tgl@sss.pgh.pa.us 678 [ + - + - :CBC 2220 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC || tcount < 0)
- + ]
9968 bruce@momjian.us 679 :UBC 0 : return SPI_ERROR_ARGUMENT;
680 : :
6851 tgl@sss.pgh.pa.us 681 [ + + - + ]:CBC 2220 : if (plan->nargs > 0 && Values == NULL)
9968 bruce@momjian.us 682 :UBC 0 : return SPI_ERROR_PARAM;
683 : :
10327 bruce@momjian.us 684 :CBC 2220 : res = _SPI_begin_call(true);
685 [ - + ]: 2220 : if (res < 0)
9968 bruce@momjian.us 686 :UBC 0 : return res;
687 : :
1535 tgl@sss.pgh.pa.us 688 :CBC 2220 : memset(&options, 0, sizeof(options));
689 : 2220 : options.params = _SPI_convert_params(plan->nargs, plan->argtypes,
690 : : Values, Nulls);
691 : 2220 : options.read_only = read_only;
692 : 2220 : options.tcount = tcount;
693 : :
694 : 2220 : res = _SPI_execute_plan(plan, &options,
695 : : InvalidSnapshot, InvalidSnapshot,
696 : : true);
697 : :
8118 698 : 2219 : _SPI_end_call(true);
699 : 2219 : return res;
700 : : }
701 : :
702 : : /* Obsolete version of SPI_execute_plan */
703 : : int
6851 704 : 27 : SPI_execp(SPIPlanPtr plan, Datum *Values, const char *Nulls, long tcount)
705 : : {
7764 706 : 27 : return SPI_execute_plan(plan, Values, Nulls, false, tcount);
707 : : }
708 : :
709 : : /* Execute a previously prepared plan */
710 : : int
1786 711 : 1277 : SPI_execute_plan_extended(SPIPlanPtr plan,
712 : : const SPIExecuteOptions *options)
713 : : {
714 : : int res;
715 : :
716 [ + - + - : 1277 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC || options == NULL)
- + ]
5886 tgl@sss.pgh.pa.us 717 :UBC 0 : return SPI_ERROR_ARGUMENT;
718 : :
5886 tgl@sss.pgh.pa.us 719 :CBC 1277 : res = _SPI_begin_call(true);
720 [ - + ]: 1277 : if (res < 0)
5886 tgl@sss.pgh.pa.us 721 :UBC 0 : return res;
722 : :
1535 tgl@sss.pgh.pa.us 723 :CBC 1277 : res = _SPI_execute_plan(plan, options,
724 : : InvalidSnapshot, InvalidSnapshot,
725 : : true);
726 : :
2013 727 : 1272 : _SPI_end_call(true);
728 : 1272 : return res;
729 : : }
730 : :
731 : : /* Execute a previously prepared plan */
732 : : int
1786 733 : 36664 : SPI_execute_plan_with_paramlist(SPIPlanPtr plan, ParamListInfo params,
734 : : bool read_only, long tcount)
735 : : {
736 : : SPIExecuteOptions options;
737 : : int res;
738 : :
2013 739 [ + - + - : 36664 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC || tcount < 0)
- + ]
2013 tgl@sss.pgh.pa.us 740 :UBC 0 : return SPI_ERROR_ARGUMENT;
741 : :
2013 tgl@sss.pgh.pa.us 742 :CBC 36664 : res = _SPI_begin_call(true);
743 [ - + ]: 36664 : if (res < 0)
2013 tgl@sss.pgh.pa.us 744 :UBC 0 : return res;
745 : :
1535 tgl@sss.pgh.pa.us 746 :CBC 36664 : memset(&options, 0, sizeof(options));
747 : 36664 : options.params = params;
748 : 36664 : options.read_only = read_only;
749 : 36664 : options.tcount = tcount;
750 : :
751 : 36664 : res = _SPI_execute_plan(plan, &options,
752 : : InvalidSnapshot, InvalidSnapshot,
753 : : true);
754 : :
5886 755 : 33964 : _SPI_end_call(true);
756 : 33964 : return res;
757 : : }
758 : :
759 : : /*
760 : : * SPI_execute_snapshot -- identical to SPI_execute_plan, except that we allow
761 : : * the caller to specify exactly which snapshots to use, which will be
762 : : * registered here. Also, the caller may specify that AFTER triggers should be
763 : : * queued as part of the outer query rather than being fired immediately at the
764 : : * end of the command.
765 : : *
766 : : * This is currently not documented in spi.sgml because it is only intended
767 : : * for use by RI triggers.
768 : : *
769 : : * Passing snapshot == InvalidSnapshot will select the normal behavior of
770 : : * fetching a new snapshot for each query.
771 : : */
772 : : int
6851 773 : 3995 : SPI_execute_snapshot(SPIPlanPtr plan,
774 : : const Datum *Values, const char *Nulls,
775 : : Snapshot snapshot, Snapshot crosscheck_snapshot,
776 : : bool read_only, bool fire_triggers, long tcount)
777 : : {
778 : : SPIExecuteOptions options;
779 : : int res;
780 : :
781 [ + - + - : 3995 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC || tcount < 0)
- + ]
8118 tgl@sss.pgh.pa.us 782 :UBC 0 : return SPI_ERROR_ARGUMENT;
783 : :
6851 tgl@sss.pgh.pa.us 784 [ + + - + ]:CBC 3995 : if (plan->nargs > 0 && Values == NULL)
8118 tgl@sss.pgh.pa.us 785 :UBC 0 : return SPI_ERROR_PARAM;
786 : :
8118 tgl@sss.pgh.pa.us 787 :CBC 3995 : res = _SPI_begin_call(true);
788 [ - + ]: 3995 : if (res < 0)
8118 tgl@sss.pgh.pa.us 789 :UBC 0 : return res;
790 : :
1535 tgl@sss.pgh.pa.us 791 :CBC 3995 : memset(&options, 0, sizeof(options));
792 : 3995 : options.params = _SPI_convert_params(plan->nargs, plan->argtypes,
793 : : Values, Nulls);
794 : 3995 : options.read_only = read_only;
795 : 3995 : options.tcount = tcount;
796 : :
797 : 3995 : res = _SPI_execute_plan(plan, &options,
798 : : snapshot, crosscheck_snapshot,
799 : : fire_triggers);
800 : :
10327 bruce@momjian.us 801 : 3980 : _SPI_end_call(true);
9968 802 : 3980 : return res;
803 : : }
804 : :
805 : : /*
806 : : * SPI_execute_with_args -- plan and execute a query with supplied arguments
807 : : *
808 : : * This is functionally equivalent to SPI_prepare followed by
809 : : * SPI_execute_plan.
810 : : */
811 : : int
6468 tgl@sss.pgh.pa.us 812 :UBC 0 : SPI_execute_with_args(const char *src,
813 : : int nargs, Oid *argtypes,
814 : : const Datum *Values, const char *Nulls,
815 : : bool read_only, long tcount)
816 : : {
817 : : int res;
818 : : _SPI_plan plan;
819 : : ParamListInfo paramLI;
820 : : SPIExecuteOptions options;
821 : :
822 [ # # # # : 0 : if (src == NULL || nargs < 0 || tcount < 0)
# # ]
823 : 0 : return SPI_ERROR_ARGUMENT;
824 : :
825 [ # # # # : 0 : if (nargs > 0 && (argtypes == NULL || Values == NULL))
# # ]
826 : 0 : return SPI_ERROR_PARAM;
827 : :
828 : 0 : res = _SPI_begin_call(true);
829 [ # # ]: 0 : if (res < 0)
830 : 0 : return res;
831 : :
832 : 0 : memset(&plan, 0, sizeof(_SPI_plan));
833 : 0 : plan.magic = _SPI_PLAN_MAGIC;
1807 834 : 0 : plan.parse_mode = RAW_PARSE_DEFAULT;
3189 rhaas@postgresql.org 835 : 0 : plan.cursor_options = CURSOR_OPT_PARALLEL_OK;
6468 tgl@sss.pgh.pa.us 836 : 0 : plan.nargs = nargs;
837 : 0 : plan.argtypes = argtypes;
5886 838 : 0 : plan.parserSetup = NULL;
839 : 0 : plan.parserSetupArg = NULL;
840 : :
6468 841 : 0 : paramLI = _SPI_convert_params(nargs, argtypes,
842 : : Values, Nulls);
843 : :
4729 844 : 0 : _SPI_prepare_oneshot_plan(src, &plan);
845 : :
1535 846 : 0 : memset(&options, 0, sizeof(options));
847 : 0 : options.params = paramLI;
848 : 0 : options.read_only = read_only;
849 : 0 : options.tcount = tcount;
850 : :
851 : 0 : res = _SPI_execute_plan(&plan, &options,
852 : : InvalidSnapshot, InvalidSnapshot,
853 : : true);
854 : :
2013 855 : 0 : _SPI_end_call(true);
856 : 0 : return res;
857 : : }
858 : :
859 : : SPIPlanPtr
8387 tgl@sss.pgh.pa.us 860 :CBC 2697 : SPI_prepare(const char *src, int nargs, Oid *argtypes)
861 : : {
6819 862 : 2697 : return SPI_prepare_cursor(src, nargs, argtypes, 0);
863 : : }
864 : :
865 : : SPIPlanPtr
866 : 2697 : SPI_prepare_cursor(const char *src, int nargs, Oid *argtypes,
867 : : int cursorOptions)
868 : : {
869 : : _SPI_plan plan;
870 : : SPIPlanPtr result;
871 : :
10308 vadim4o@yahoo.com 872 [ + - + - : 2697 : if (src == NULL || nargs < 0 || (nargs > 0 && argtypes == NULL))
+ + - + ]
873 : : {
10327 bruce@momjian.us 874 :UBC 0 : SPI_result = SPI_ERROR_ARGUMENT;
9968 875 : 0 : return NULL;
876 : : }
877 : :
10327 bruce@momjian.us 878 :CBC 2697 : SPI_result = _SPI_begin_call(true);
879 [ - + ]: 2697 : if (SPI_result < 0)
9968 bruce@momjian.us 880 :UBC 0 : return NULL;
881 : :
6851 tgl@sss.pgh.pa.us 882 :CBC 2697 : memset(&plan, 0, sizeof(_SPI_plan));
883 : 2697 : plan.magic = _SPI_PLAN_MAGIC;
1807 884 : 2697 : plan.parse_mode = RAW_PARSE_DEFAULT;
6819 885 : 2697 : plan.cursor_options = cursorOptions;
7940 886 : 2697 : plan.nargs = nargs;
887 : 2697 : plan.argtypes = argtypes;
5886 888 : 2697 : plan.parserSetup = NULL;
889 : 2697 : plan.parserSetupArg = NULL;
890 : :
4729 891 : 2697 : _SPI_prepare_plan(src, &plan);
892 : :
893 : : /* copy plan to procedure context */
5205 894 : 2696 : result = _SPI_make_plan_non_temp(&plan);
895 : :
5886 896 : 2696 : _SPI_end_call(true);
897 : :
898 : 2696 : return result;
899 : : }
900 : :
901 : : SPIPlanPtr
1807 902 : 14837 : SPI_prepare_extended(const char *src,
903 : : const SPIPrepareOptions *options)
904 : : {
905 : : _SPI_plan plan;
906 : : SPIPlanPtr result;
907 : :
908 [ + - - + ]: 14837 : if (src == NULL || options == NULL)
909 : : {
1807 tgl@sss.pgh.pa.us 910 :UBC 0 : SPI_result = SPI_ERROR_ARGUMENT;
911 : 0 : return NULL;
912 : : }
913 : :
1807 tgl@sss.pgh.pa.us 914 :CBC 14837 : SPI_result = _SPI_begin_call(true);
915 [ - + ]: 14837 : if (SPI_result < 0)
1807 tgl@sss.pgh.pa.us 916 :UBC 0 : return NULL;
917 : :
1807 tgl@sss.pgh.pa.us 918 :CBC 14837 : memset(&plan, 0, sizeof(_SPI_plan));
919 : 14837 : plan.magic = _SPI_PLAN_MAGIC;
920 : 14837 : plan.parse_mode = options->parseMode;
921 : 14837 : plan.cursor_options = options->cursorOptions;
922 : 14837 : plan.nargs = 0;
923 : 14837 : plan.argtypes = NULL;
924 : 14837 : plan.parserSetup = options->parserSetup;
925 : 14837 : plan.parserSetupArg = options->parserSetupArg;
926 : :
927 : 14837 : _SPI_prepare_plan(src, &plan);
928 : :
929 : : /* copy plan to procedure context */
930 : 14786 : result = _SPI_make_plan_non_temp(&plan);
931 : :
932 : 14786 : _SPI_end_call(true);
933 : :
934 : 14786 : return result;
935 : : }
936 : :
937 : : SPIPlanPtr
5886 tgl@sss.pgh.pa.us 938 :UBC 0 : SPI_prepare_params(const char *src,
939 : : ParserSetupHook parserSetup,
940 : : void *parserSetupArg,
941 : : int cursorOptions)
942 : : {
943 : : _SPI_plan plan;
944 : : SPIPlanPtr result;
945 : :
946 [ # # ]: 0 : if (src == NULL)
947 : : {
948 : 0 : SPI_result = SPI_ERROR_ARGUMENT;
949 : 0 : return NULL;
950 : : }
951 : :
952 : 0 : SPI_result = _SPI_begin_call(true);
953 [ # # ]: 0 : if (SPI_result < 0)
954 : 0 : return NULL;
955 : :
956 : 0 : memset(&plan, 0, sizeof(_SPI_plan));
957 : 0 : plan.magic = _SPI_PLAN_MAGIC;
1807 958 : 0 : plan.parse_mode = RAW_PARSE_DEFAULT;
5886 959 : 0 : plan.cursor_options = cursorOptions;
960 : 0 : plan.nargs = 0;
961 : 0 : plan.argtypes = NULL;
962 : 0 : plan.parserSetup = parserSetup;
963 : 0 : plan.parserSetupArg = parserSetupArg;
964 : :
4729 965 : 0 : _SPI_prepare_plan(src, &plan);
966 : :
967 : : /* copy plan to procedure context */
5205 968 : 0 : result = _SPI_make_plan_non_temp(&plan);
969 : :
10327 bruce@momjian.us 970 : 0 : _SPI_end_call(true);
971 : :
6851 tgl@sss.pgh.pa.us 972 : 0 : return result;
973 : : }
974 : :
975 : : int
5205 tgl@sss.pgh.pa.us 976 :CBC 16717 : SPI_keepplan(SPIPlanPtr plan)
977 : : {
978 : : ListCell *lc;
979 : :
4729 980 [ + - + - ]: 16717 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC ||
981 [ + - - + ]: 16717 : plan->saved || plan->oneshot)
5205 tgl@sss.pgh.pa.us 982 :UBC 0 : return SPI_ERROR_ARGUMENT;
983 : :
984 : : /*
985 : : * Mark it saved, reparent it under CacheMemoryContext, and mark all the
986 : : * component CachedPlanSources as saved. This sequence cannot fail
987 : : * partway through, so there's no risk of long-term memory leakage.
988 : : */
5205 tgl@sss.pgh.pa.us 989 :CBC 16717 : plan->saved = true;
990 : 16717 : MemoryContextSetParent(plan->plancxt, CacheMemoryContext);
991 : :
992 [ + - + + : 33434 : foreach(lc, plan->plancache_list)
+ + ]
993 : : {
994 : 16717 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
995 : :
996 : 16717 : SaveCachedPlan(plansource);
997 : : }
998 : :
999 : 16717 : return 0;
1000 : : }
1001 : :
1002 : : SPIPlanPtr
6851 tgl@sss.pgh.pa.us 1003 :UBC 0 : SPI_saveplan(SPIPlanPtr plan)
1004 : : {
1005 : : SPIPlanPtr newplan;
1006 : :
5205 1007 [ # # # # ]: 0 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC)
1008 : : {
10327 bruce@momjian.us 1009 : 0 : SPI_result = SPI_ERROR_ARGUMENT;
9968 1010 : 0 : return NULL;
1011 : : }
1012 : :
3100 tgl@sss.pgh.pa.us 1013 : 0 : SPI_result = _SPI_begin_call(false); /* don't change context */
10327 bruce@momjian.us 1014 [ # # ]: 0 : if (SPI_result < 0)
9968 1015 : 0 : return NULL;
1016 : :
6851 tgl@sss.pgh.pa.us 1017 : 0 : newplan = _SPI_save_plan(plan);
1018 : :
5205 1019 : 0 : SPI_result = _SPI_end_call(false);
1020 : :
6851 1021 : 0 : return newplan;
1022 : : }
1023 : :
1024 : : int
6851 tgl@sss.pgh.pa.us 1025 :CBC 3839 : SPI_freeplan(SPIPlanPtr plan)
1026 : : {
1027 : : ListCell *lc;
1028 : :
1029 [ + - - + ]: 3839 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC)
8975 JanWieck@Yahoo.com 1030 :UBC 0 : return SPI_ERROR_ARGUMENT;
1031 : :
1032 : : /* Release the plancache entries */
5205 tgl@sss.pgh.pa.us 1033 [ + - + + :CBC 7678 : foreach(lc, plan->plancache_list)
+ + ]
1034 : : {
1035 : 3839 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
1036 : :
1037 : 3839 : DropCachedPlan(plansource);
1038 : : }
1039 : :
1040 : : /* Now get rid of the _SPI_plan and subsidiary data in its plancxt */
6851 1041 : 3839 : MemoryContextDelete(plan->plancxt);
1042 : :
8975 JanWieck@Yahoo.com 1043 : 3839 : return 0;
1044 : : }
1045 : :
1046 : : HeapTuple
10322 vadim4o@yahoo.com 1047 : 1146 : SPI_copytuple(HeapTuple tuple)
1048 : : {
1049 : : MemoryContext oldcxt;
1050 : : HeapTuple ctuple;
1051 : :
1052 [ - + ]: 1146 : if (tuple == NULL)
1053 : : {
10322 vadim4o@yahoo.com 1054 :UBC 0 : SPI_result = SPI_ERROR_ARGUMENT;
9968 bruce@momjian.us 1055 : 0 : return NULL;
1056 : : }
1057 : :
3325 tgl@sss.pgh.pa.us 1058 [ - + ]:CBC 1146 : if (_SPI_current == NULL)
1059 : : {
3325 tgl@sss.pgh.pa.us 1060 :UBC 0 : SPI_result = SPI_ERROR_UNCONNECTED;
1061 : 0 : return NULL;
1062 : : }
1063 : :
3325 tgl@sss.pgh.pa.us 1064 :CBC 1146 : oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);
1065 : :
10322 vadim4o@yahoo.com 1066 : 1146 : ctuple = heap_copytuple(tuple);
1067 : :
3325 tgl@sss.pgh.pa.us 1068 : 1146 : MemoryContextSwitchTo(oldcxt);
1069 : :
9968 bruce@momjian.us 1070 : 1146 : return ctuple;
1071 : : }
1072 : :
1073 : : HeapTupleHeader
7929 tgl@sss.pgh.pa.us 1074 : 3094 : SPI_returntuple(HeapTuple tuple, TupleDesc tupdesc)
1075 : : {
1076 : : MemoryContext oldcxt;
1077 : : HeapTupleHeader dtup;
1078 : :
8807 1079 [ + - - + ]: 3094 : if (tuple == NULL || tupdesc == NULL)
1080 : : {
8807 tgl@sss.pgh.pa.us 1081 :UBC 0 : SPI_result = SPI_ERROR_ARGUMENT;
1082 : 0 : return NULL;
1083 : : }
1084 : :
3325 tgl@sss.pgh.pa.us 1085 [ - + ]:CBC 3094 : if (_SPI_current == NULL)
1086 : : {
3325 tgl@sss.pgh.pa.us 1087 :UBC 0 : SPI_result = SPI_ERROR_UNCONNECTED;
1088 : 0 : return NULL;
1089 : : }
1090 : :
1091 : : /* For RECORD results, make sure a typmod has been assigned */
7929 tgl@sss.pgh.pa.us 1092 [ + + ]:CBC 3094 : if (tupdesc->tdtypeid == RECORDOID &&
1093 [ - + ]: 3086 : tupdesc->tdtypmod < 0)
7929 tgl@sss.pgh.pa.us 1094 :UBC 0 : assign_record_type_typmod(tupdesc);
1095 : :
3325 tgl@sss.pgh.pa.us 1096 :CBC 3094 : oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);
1097 : :
4247 1098 : 3094 : dtup = DatumGetHeapTupleHeader(heap_copy_tuple_as_datum(tuple, tupdesc));
1099 : :
3325 1100 : 3094 : MemoryContextSwitchTo(oldcxt);
1101 : :
7929 1102 : 3094 : return dtup;
1103 : : }
1104 : :
1105 : : HeapTuple
10322 vadim4o@yahoo.com 1106 :UBC 0 : SPI_modifytuple(Relation rel, HeapTuple tuple, int natts, int *attnum,
1107 : : Datum *Values, const char *Nulls)
1108 : : {
1109 : : MemoryContext oldcxt;
1110 : : HeapTuple mtuple;
1111 : : int numberOfAttributes;
1112 : : Datum *v;
1113 : : bool *n;
1114 : : int i;
1115 : :
8127 tgl@sss.pgh.pa.us 1116 [ # # # # : 0 : if (rel == NULL || tuple == NULL || natts < 0 || attnum == NULL || Values == NULL)
# # # # #
# ]
1117 : : {
10322 vadim4o@yahoo.com 1118 : 0 : SPI_result = SPI_ERROR_ARGUMENT;
9968 bruce@momjian.us 1119 : 0 : return NULL;
1120 : : }
1121 : :
3325 tgl@sss.pgh.pa.us 1122 [ # # ]: 0 : if (_SPI_current == NULL)
1123 : : {
1124 : 0 : SPI_result = SPI_ERROR_UNCONNECTED;
1125 : 0 : return NULL;
1126 : : }
1127 : :
1128 : 0 : oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);
1129 : :
10322 vadim4o@yahoo.com 1130 : 0 : SPI_result = 0;
1131 : :
1132 : 0 : numberOfAttributes = rel->rd_att->natts;
6 michael@paquier.xyz 1133 :UNC 0 : v = palloc_array(Datum, numberOfAttributes);
1134 : 0 : n = palloc_array(bool, numberOfAttributes);
1135 : :
1136 : : /* fetch old values and nulls */
6253 tgl@sss.pgh.pa.us 1137 :UBC 0 : heap_deform_tuple(tuple, rel->rd_att, v, n);
1138 : :
1139 : : /* replace values and nulls */
10322 vadim4o@yahoo.com 1140 [ # # ]: 0 : for (i = 0; i < natts; i++)
1141 : : {
1142 [ # # # # ]: 0 : if (attnum[i] <= 0 || attnum[i] > numberOfAttributes)
1143 : : break;
1144 : 0 : v[attnum[i] - 1] = Values[i];
1560 michael@paquier.xyz 1145 [ # # # # ]: 0 : n[attnum[i] - 1] = (Nulls && Nulls[i] == 'n');
1146 : : }
1147 : :
9968 bruce@momjian.us 1148 [ # # ]: 0 : if (i == natts) /* no errors in *attnum */
1149 : : {
6253 tgl@sss.pgh.pa.us 1150 : 0 : mtuple = heap_form_tuple(rel->rd_att, v, n);
1151 : :
1152 : : /*
1153 : : * copy the identification info of the old tuple: t_ctid, t_self, and
1154 : : * OID (if any)
1155 : : */
8506 1156 : 0 : mtuple->t_data->t_ctid = tuple->t_data->t_ctid;
1157 : 0 : mtuple->t_self = tuple->t_self;
1158 : 0 : mtuple->t_tableOid = tuple->t_tableOid;
1159 : : }
1160 : : else
1161 : : {
10322 vadim4o@yahoo.com 1162 : 0 : mtuple = NULL;
1163 : 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
1164 : : }
1165 : :
1166 : 0 : pfree(v);
1167 : 0 : pfree(n);
1168 : :
3325 tgl@sss.pgh.pa.us 1169 : 0 : MemoryContextSwitchTo(oldcxt);
1170 : :
9968 bruce@momjian.us 1171 : 0 : return mtuple;
1172 : : }
1173 : :
1174 : : int
8387 tgl@sss.pgh.pa.us 1175 :CBC 10856 : SPI_fnumber(TupleDesc tupdesc, const char *fname)
1176 : : {
1177 : : int res;
1178 : : const FormData_pg_attribute *sysatt;
1179 : :
10327 bruce@momjian.us 1180 [ + + ]: 58365 : for (res = 0; res < tupdesc->natts; res++)
1181 : : {
3040 andres@anarazel.de 1182 : 58358 : Form_pg_attribute attr = TupleDescAttr(tupdesc, res);
1183 : :
1184 [ + + ]: 58358 : if (namestrcmp(&attr->attname, fname) == 0 &&
1185 [ + - ]: 10849 : !attr->attisdropped)
9968 bruce@momjian.us 1186 : 10849 : return res + 1;
1187 : : }
1188 : :
2583 andres@anarazel.de 1189 : 7 : sysatt = SystemAttributeByName(fname);
8820 tgl@sss.pgh.pa.us 1190 [ - + ]: 7 : if (sysatt != NULL)
8820 tgl@sss.pgh.pa.us 1191 :UBC 0 : return sysatt->attnum;
1192 : :
1193 : : /* SPI_ERROR_NOATTRIBUTE is different from all sys column numbers */
9968 bruce@momjian.us 1194 :CBC 7 : return SPI_ERROR_NOATTRIBUTE;
1195 : : }
1196 : :
1197 : : char *
10323 vadim4o@yahoo.com 1198 : 486 : SPI_fname(TupleDesc tupdesc, int fnumber)
1199 : : {
1200 : : const FormData_pg_attribute *att;
1201 : :
1202 : 486 : SPI_result = 0;
1203 : :
8820 tgl@sss.pgh.pa.us 1204 [ + - + - : 486 : if (fnumber > tupdesc->natts || fnumber == 0 ||
- + ]
1205 : : fnumber <= FirstLowInvalidHeapAttributeNumber)
1206 : : {
10323 vadim4o@yahoo.com 1207 :UBC 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
9968 bruce@momjian.us 1208 : 0 : return NULL;
1209 : : }
1210 : :
8820 tgl@sss.pgh.pa.us 1211 [ + - ]:CBC 486 : if (fnumber > 0)
3040 andres@anarazel.de 1212 : 486 : att = TupleDescAttr(tupdesc, fnumber - 1);
1213 : : else
2583 andres@anarazel.de 1214 :UBC 0 : att = SystemAttributeDefinition(fnumber);
1215 : :
8820 tgl@sss.pgh.pa.us 1216 :CBC 486 : return pstrdup(NameStr(att->attname));
1217 : : }
1218 : :
1219 : : char *
10327 bruce@momjian.us 1220 : 4722 : SPI_getvalue(HeapTuple tuple, TupleDesc tupdesc, int fnumber)
1221 : : {
1222 : : Datum val;
1223 : : bool isnull;
1224 : : Oid typoid,
1225 : : foutoid;
1226 : : bool typisvarlena;
1227 : :
1228 : 4722 : SPI_result = 0;
1229 : :
6270 tgl@sss.pgh.pa.us 1230 [ + - + - : 4722 : if (fnumber > tupdesc->natts || fnumber == 0 ||
- + ]
1231 : : fnumber <= FirstLowInvalidHeapAttributeNumber)
1232 : : {
10305 vadim4o@yahoo.com 1233 :UBC 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
9968 bruce@momjian.us 1234 : 0 : return NULL;
1235 : : }
1236 : :
4426 tgl@sss.pgh.pa.us 1237 :CBC 4722 : val = heap_getattr(tuple, fnumber, tupdesc, &isnull);
10327 bruce@momjian.us 1238 [ + + ]: 4722 : if (isnull)
9968 1239 : 66 : return NULL;
1240 : :
8820 tgl@sss.pgh.pa.us 1241 [ + - ]: 4656 : if (fnumber > 0)
3040 andres@anarazel.de 1242 : 4656 : typoid = TupleDescAttr(tupdesc, fnumber - 1)->atttypid;
1243 : : else
2583 andres@anarazel.de 1244 :UBC 0 : typoid = (SystemAttributeDefinition(fnumber))->atttypid;
1245 : :
7534 tgl@sss.pgh.pa.us 1246 :CBC 4656 : getTypeOutputInfo(typoid, &foutoid, &typisvarlena);
1247 : :
4426 1248 : 4656 : return OidOutputFunctionCall(foutoid, val);
1249 : : }
1250 : :
1251 : : Datum
10155 bruce@momjian.us 1252 : 28710 : SPI_getbinval(HeapTuple tuple, TupleDesc tupdesc, int fnumber, bool *isnull)
1253 : : {
10327 1254 : 28710 : SPI_result = 0;
1255 : :
6270 tgl@sss.pgh.pa.us 1256 [ + - + - : 28710 : if (fnumber > tupdesc->natts || fnumber == 0 ||
- + ]
1257 : : fnumber <= FirstLowInvalidHeapAttributeNumber)
1258 : : {
10305 vadim4o@yahoo.com 1259 :UBC 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
8820 tgl@sss.pgh.pa.us 1260 : 0 : *isnull = true;
130 tgl@sss.pgh.pa.us 1261 :UNC 0 : return (Datum) 0;
1262 : : }
1263 : :
8820 tgl@sss.pgh.pa.us 1264 :CBC 28710 : return heap_getattr(tuple, fnumber, tupdesc, isnull);
1265 : : }
1266 : :
1267 : : char *
10327 bruce@momjian.us 1268 : 4 : SPI_gettype(TupleDesc tupdesc, int fnumber)
1269 : : {
1270 : : Oid typoid;
1271 : : HeapTuple typeTuple;
1272 : : char *result;
1273 : :
1274 : 4 : SPI_result = 0;
1275 : :
8820 tgl@sss.pgh.pa.us 1276 [ + - + - : 4 : if (fnumber > tupdesc->natts || fnumber == 0 ||
- + ]
1277 : : fnumber <= FirstLowInvalidHeapAttributeNumber)
1278 : : {
10327 bruce@momjian.us 1279 :UBC 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
9968 1280 : 0 : return NULL;
1281 : : }
1282 : :
8820 tgl@sss.pgh.pa.us 1283 [ + - ]:CBC 4 : if (fnumber > 0)
3040 andres@anarazel.de 1284 : 4 : typoid = TupleDescAttr(tupdesc, fnumber - 1)->atttypid;
1285 : : else
2583 andres@anarazel.de 1286 :UBC 0 : typoid = (SystemAttributeDefinition(fnumber))->atttypid;
1287 : :
5784 rhaas@postgresql.org 1288 :CBC 4 : typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typoid));
1289 : :
10327 bruce@momjian.us 1290 [ - + ]: 4 : if (!HeapTupleIsValid(typeTuple))
1291 : : {
10327 bruce@momjian.us 1292 :UBC 0 : SPI_result = SPI_ERROR_TYPUNKNOWN;
9968 1293 : 0 : return NULL;
1294 : : }
1295 : :
9161 tgl@sss.pgh.pa.us 1296 :CBC 4 : result = pstrdup(NameStr(((Form_pg_type) GETSTRUCT(typeTuple))->typname));
1297 : 4 : ReleaseSysCache(typeTuple);
1298 : 4 : return result;
1299 : : }
1300 : :
1301 : : /*
1302 : : * Get the data type OID for a column.
1303 : : *
1304 : : * There's nothing similar for typmod and typcollation. The rare consumers
1305 : : * thereof should inspect the TupleDesc directly.
1306 : : */
1307 : : Oid
10327 bruce@momjian.us 1308 : 602 : SPI_gettypeid(TupleDesc tupdesc, int fnumber)
1309 : : {
1310 : 602 : SPI_result = 0;
1311 : :
8820 tgl@sss.pgh.pa.us 1312 [ + - + - : 602 : if (fnumber > tupdesc->natts || fnumber == 0 ||
- + ]
1313 : : fnumber <= FirstLowInvalidHeapAttributeNumber)
1314 : : {
10327 bruce@momjian.us 1315 :UBC 0 : SPI_result = SPI_ERROR_NOATTRIBUTE;
9968 1316 : 0 : return InvalidOid;
1317 : : }
1318 : :
8820 tgl@sss.pgh.pa.us 1319 [ + - ]:CBC 602 : if (fnumber > 0)
3040 andres@anarazel.de 1320 : 602 : return TupleDescAttr(tupdesc, fnumber - 1)->atttypid;
1321 : : else
2583 andres@anarazel.de 1322 :UBC 0 : return (SystemAttributeDefinition(fnumber))->atttypid;
1323 : : }
1324 : :
1325 : : char *
10327 bruce@momjian.us 1326 :CBC 183 : SPI_getrelname(Relation rel)
1327 : : {
9536 1328 : 183 : return pstrdup(RelationGetRelationName(rel));
1329 : : }
1330 : :
1331 : : char *
7567 neilc@samurai.com 1332 : 141 : SPI_getnspname(Relation rel)
1333 : : {
7367 bruce@momjian.us 1334 : 141 : return get_namespace_name(RelationGetNamespace(rel));
1335 : : }
1336 : :
1337 : : void *
10155 1338 : 19 : SPI_palloc(Size size)
1339 : : {
3325 tgl@sss.pgh.pa.us 1340 [ - + ]: 19 : if (_SPI_current == NULL)
3325 tgl@sss.pgh.pa.us 1341 [ # # ]:UBC 0 : elog(ERROR, "SPI_palloc called while not connected to SPI");
1342 : :
3325 tgl@sss.pgh.pa.us 1343 :CBC 19 : return MemoryContextAlloc(_SPI_current->savedcxt, size);
1344 : : }
1345 : :
1346 : : void *
10155 bruce@momjian.us 1347 :UBC 0 : SPI_repalloc(void *pointer, Size size)
1348 : : {
1349 : : /* No longer need to worry which context chunk was in... */
9302 tgl@sss.pgh.pa.us 1350 : 0 : return repalloc(pointer, size);
1351 : : }
1352 : :
1353 : : void
10155 bruce@momjian.us 1354 : 0 : SPI_pfree(void *pointer)
1355 : : {
1356 : : /* No longer need to worry which context chunk was in... */
1357 : 0 : pfree(pointer);
10310 vadim4o@yahoo.com 1358 : 0 : }
1359 : :
1360 : : Datum
3869 tgl@sss.pgh.pa.us 1361 :CBC 3265 : SPI_datumTransfer(Datum value, bool typByVal, int typLen)
1362 : : {
1363 : : MemoryContext oldcxt;
1364 : : Datum result;
1365 : :
3325 1366 [ - + ]: 3265 : if (_SPI_current == NULL)
3325 tgl@sss.pgh.pa.us 1367 [ # # ]:UBC 0 : elog(ERROR, "SPI_datumTransfer called while not connected to SPI");
1368 : :
3325 tgl@sss.pgh.pa.us 1369 :CBC 3265 : oldcxt = MemoryContextSwitchTo(_SPI_current->savedcxt);
1370 : :
3869 1371 : 3265 : result = datumTransfer(value, typByVal, typLen);
1372 : :
3325 1373 : 3265 : MemoryContextSwitchTo(oldcxt);
1374 : :
3869 1375 : 3265 : return result;
1376 : : }
1377 : :
1378 : : void
9497 JanWieck@Yahoo.com 1379 :UBC 0 : SPI_freetuple(HeapTuple tuple)
1380 : : {
1381 : : /* No longer need to worry which context tuple was in... */
1382 : 0 : heap_freetuple(tuple);
1383 : 0 : }
1384 : :
1385 : : void
8975 JanWieck@Yahoo.com 1386 :CBC 89832 : SPI_freetuptable(SPITupleTable *tuptable)
1387 : : {
4527 tgl@sss.pgh.pa.us 1388 : 89832 : bool found = false;
1389 : :
1390 : : /* ignore call if NULL pointer */
1391 [ + + ]: 89832 : if (tuptable == NULL)
1392 : 50517 : return;
1393 : :
1394 : : /*
1395 : : * Search only the topmost SPI context for a matching tuple table.
1396 : : */
3325 1397 [ + - ]: 39315 : if (_SPI_current != NULL)
1398 : : {
1399 : : slist_mutable_iter siter;
1400 : :
1401 : : /* find tuptable in active list, then remove it */
4527 1402 [ + - + - : 39317 : slist_foreach_modify(siter, &_SPI_current->tuptables)
+ - ]
1403 : : {
1404 : : SPITupleTable *tt;
1405 : :
1406 : 39317 : tt = slist_container(SPITupleTable, next, siter.cur);
1407 [ + + ]: 39317 : if (tt == tuptable)
1408 : : {
1409 : 39315 : slist_delete_current(&siter);
1410 : 39315 : found = true;
1411 : 39315 : break;
1412 : : }
1413 : : }
1414 : : }
1415 : :
1416 : : /*
1417 : : * Refuse the deletion if we didn't find it in the topmost SPI context.
1418 : : * This is primarily a guard against double deletion, but might prevent
1419 : : * other errors as well. Since the worst consequence of not deleting a
1420 : : * tuptable would be a transient memory leak, this is just a WARNING.
1421 : : */
1422 [ - + ]: 39315 : if (!found)
1423 : : {
4527 tgl@sss.pgh.pa.us 1424 [ # # ]:UBC 0 : elog(WARNING, "attempt to delete invalid SPITupleTable %p", tuptable);
1425 : 0 : return;
1426 : : }
1427 : :
1428 : : /* for safety, reset global variables that might point at tuptable */
4527 tgl@sss.pgh.pa.us 1429 [ - + ]:CBC 39315 : if (tuptable == _SPI_current->tuptable)
4527 tgl@sss.pgh.pa.us 1430 :UBC 0 : _SPI_current->tuptable = NULL;
4527 tgl@sss.pgh.pa.us 1431 [ + + ]:CBC 39315 : if (tuptable == SPI_tuptable)
1432 : 35438 : SPI_tuptable = NULL;
1433 : :
1434 : : /* release all memory belonging to tuptable */
1435 : 39315 : MemoryContextDelete(tuptable->tuptabcxt);
1436 : : }
1437 : :
1438 : :
1439 : : /*
1440 : : * SPI_cursor_open()
1441 : : *
1442 : : * Open a prepared SPI plan as a portal
1443 : : */
1444 : : Portal
6851 1445 : 107 : SPI_cursor_open(const char *name, SPIPlanPtr plan,
1446 : : const Datum *Values, const char *Nulls,
1447 : : bool read_only)
1448 : : {
1449 : : Portal portal;
1450 : : ParamListInfo paramLI;
1451 : :
1452 : : /* build transient ParamListInfo in caller's context */
5886 1453 : 107 : paramLI = _SPI_convert_params(plan->nargs, plan->argtypes,
1454 : : Values, Nulls);
1455 : :
1456 : 107 : portal = SPI_cursor_open_internal(name, plan, paramLI, read_only);
1457 : :
1458 : : /* done with the transient ParamListInfo */
1459 [ + + ]: 107 : if (paramLI)
1460 : 4 : pfree(paramLI);
1461 : :
1462 : 107 : return portal;
1463 : : }
1464 : :
1465 : :
1466 : : /*
1467 : : * SPI_cursor_open_with_args()
1468 : : *
1469 : : * Parse and plan a query and open it as a portal.
1470 : : */
1471 : : Portal
6407 tgl@sss.pgh.pa.us 1472 :UBC 0 : SPI_cursor_open_with_args(const char *name,
1473 : : const char *src,
1474 : : int nargs, Oid *argtypes,
1475 : : Datum *Values, const char *Nulls,
1476 : : bool read_only, int cursorOptions)
1477 : : {
1478 : : Portal result;
1479 : : _SPI_plan plan;
1480 : : ParamListInfo paramLI;
1481 : :
1482 [ # # # # ]: 0 : if (src == NULL || nargs < 0)
1483 [ # # ]: 0 : elog(ERROR, "SPI_cursor_open_with_args called with invalid arguments");
1484 : :
1485 [ # # # # : 0 : if (nargs > 0 && (argtypes == NULL || Values == NULL))
# # ]
1486 [ # # ]: 0 : elog(ERROR, "SPI_cursor_open_with_args called with missing parameters");
1487 : :
1488 : 0 : SPI_result = _SPI_begin_call(true);
1489 [ # # ]: 0 : if (SPI_result < 0)
1490 [ # # ]: 0 : elog(ERROR, "SPI_cursor_open_with_args called while not connected");
1491 : :
1492 : 0 : memset(&plan, 0, sizeof(_SPI_plan));
1493 : 0 : plan.magic = _SPI_PLAN_MAGIC;
1807 1494 : 0 : plan.parse_mode = RAW_PARSE_DEFAULT;
6407 1495 : 0 : plan.cursor_options = cursorOptions;
1496 : 0 : plan.nargs = nargs;
1497 : 0 : plan.argtypes = argtypes;
5886 1498 : 0 : plan.parserSetup = NULL;
1499 : 0 : plan.parserSetupArg = NULL;
1500 : :
1501 : : /* build transient ParamListInfo in executor context */
6407 1502 : 0 : paramLI = _SPI_convert_params(nargs, argtypes,
1503 : : Values, Nulls);
1504 : :
4729 1505 : 0 : _SPI_prepare_plan(src, &plan);
1506 : :
1507 : : /* We needn't copy the plan; SPI_cursor_open_internal will do so */
1508 : :
5886 1509 : 0 : result = SPI_cursor_open_internal(name, &plan, paramLI, read_only);
1510 : :
1511 : : /* And clean up */
6407 1512 : 0 : _SPI_end_call(true);
1513 : :
1514 : 0 : return result;
1515 : : }
1516 : :
1517 : :
1518 : : /*
1519 : : * SPI_cursor_open_with_paramlist()
1520 : : *
1521 : : * Same as SPI_cursor_open except that parameters (if any) are passed
1522 : : * as a ParamListInfo, which supports dynamic parameter set determination
1523 : : */
1524 : : Portal
5886 tgl@sss.pgh.pa.us 1525 :CBC 1364 : SPI_cursor_open_with_paramlist(const char *name, SPIPlanPtr plan,
1526 : : ParamListInfo params, bool read_only)
1527 : : {
1528 : 1364 : return SPI_cursor_open_internal(name, plan, params, read_only);
1529 : : }
1530 : :
1531 : : /* Parse a query and open it as a cursor */
1532 : : Portal
1785 1533 : 4716 : SPI_cursor_parse_open(const char *name,
1534 : : const char *src,
1535 : : const SPIParseOpenOptions *options)
1536 : : {
1537 : : Portal result;
1538 : : _SPI_plan plan;
1539 : :
1540 [ + - - + ]: 4716 : if (src == NULL || options == NULL)
1785 tgl@sss.pgh.pa.us 1541 [ # # ]:UBC 0 : elog(ERROR, "SPI_cursor_parse_open called with invalid arguments");
1542 : :
2013 tgl@sss.pgh.pa.us 1543 :CBC 4716 : SPI_result = _SPI_begin_call(true);
1544 [ - + ]: 4716 : if (SPI_result < 0)
1785 tgl@sss.pgh.pa.us 1545 [ # # ]:UBC 0 : elog(ERROR, "SPI_cursor_parse_open called while not connected");
1546 : :
2013 tgl@sss.pgh.pa.us 1547 :CBC 4716 : memset(&plan, 0, sizeof(_SPI_plan));
1548 : 4716 : plan.magic = _SPI_PLAN_MAGIC;
1807 1549 : 4716 : plan.parse_mode = RAW_PARSE_DEFAULT;
1785 1550 : 4716 : plan.cursor_options = options->cursorOptions;
1551 [ + + ]: 4716 : if (options->params)
1552 : : {
1553 : 6 : plan.parserSetup = options->params->parserSetup;
1554 : 6 : plan.parserSetupArg = options->params->parserSetupArg;
1555 : : }
1556 : :
2013 1557 : 4716 : _SPI_prepare_plan(src, &plan);
1558 : :
1559 : : /* We needn't copy the plan; SPI_cursor_open_internal will do so */
1560 : :
1785 1561 : 4716 : result = SPI_cursor_open_internal(name, &plan,
1562 : 4716 : options->params, options->read_only);
1563 : :
1564 : : /* And clean up */
2013 1565 : 4716 : _SPI_end_call(true);
1566 : :
1567 : 4716 : return result;
1568 : : }
1569 : :
1570 : :
1571 : : /*
1572 : : * SPI_cursor_open_internal()
1573 : : *
1574 : : * Common code for SPI_cursor_open variants
1575 : : */
1576 : : static Portal
6407 1577 : 6187 : SPI_cursor_open_internal(const char *name, SPIPlanPtr plan,
1578 : : ParamListInfo paramLI, bool read_only)
1579 : : {
1580 : : CachedPlanSource *plansource;
1581 : : CachedPlan *cplan;
1582 : : List *stmt_list;
1583 : : char *query_string;
1584 : : Snapshot snapshot;
1585 : : MemoryContext oldcontext;
1586 : : Portal portal;
1587 : : SPICallbackArg spicallbackarg;
1588 : : ErrorContextCallback spierrcontext;
1589 : :
1590 : : /*
1591 : : * Check that the plan is something the Portal code will special-case as
1592 : : * returning one tupleset.
1593 : : */
6851 1594 [ - + ]: 6187 : if (!SPI_is_cursor_plan(plan))
1595 : : {
1596 : : /* try to give a good error message */
1597 : : const char *cmdtag;
1598 : :
6851 tgl@sss.pgh.pa.us 1599 [ # # ]:UBC 0 : if (list_length(plan->plancache_list) != 1)
7064 1600 [ # # ]: 0 : ereport(ERROR,
1601 : : (errcode(ERRCODE_INVALID_CURSOR_DEFINITION),
1602 : : errmsg("cannot open multi-query plan as cursor")));
6851 1603 : 0 : plansource = (CachedPlanSource *) linitial(plan->plancache_list);
1604 : : /* A SELECT that fails SPI_is_cursor_plan() must be SELECT INTO */
1578 1605 [ # # ]: 0 : if (plansource->commandTag == CMDTAG_SELECT)
1606 : 0 : cmdtag = "SELECT INTO";
1607 : : else
1608 : 0 : cmdtag = GetCommandTagName(plansource->commandTag);
7066 1609 [ # # ]: 0 : ereport(ERROR,
1610 : : (errcode(ERRCODE_INVALID_CURSOR_DEFINITION),
1611 : : /* translator: %s is name of a SQL command, eg INSERT */
1612 : : errmsg("cannot open %s query as cursor", cmdtag)));
1613 : : }
1614 : :
6851 tgl@sss.pgh.pa.us 1615 [ - + ]:CBC 6187 : Assert(list_length(plan->plancache_list) == 1);
1616 : 6187 : plansource = (CachedPlanSource *) linitial(plan->plancache_list);
1617 : :
1618 : : /* Push the SPI stack */
5205 1619 [ - + ]: 6187 : if (_SPI_begin_call(true) < 0)
6517 tgl@sss.pgh.pa.us 1620 [ # # ]:UBC 0 : elog(ERROR, "SPI_cursor_open called while not connected");
1621 : :
1622 : : /* Reset SPI result (note we deliberately don't touch lastoid) */
8975 JanWieck@Yahoo.com 1623 :CBC 6187 : SPI_processed = 0;
1624 : 6187 : SPI_tuptable = NULL;
1625 : 6187 : _SPI_current->processed = 0;
1626 : 6187 : _SPI_current->tuptable = NULL;
1627 : :
1628 : : /* Create the portal */
8264 tgl@sss.pgh.pa.us 1629 [ + + - + ]: 6187 : if (name == NULL || name[0] == '\0')
1630 : : {
1631 : : /* Use a random nonconflicting name */
1632 : 6163 : portal = CreateNewPortal();
1633 : : }
1634 : : else
1635 : : {
1636 : : /* In this path, error if portal of same name already exists */
1637 : 24 : portal = CreatePortal(name, false, false);
1638 : : }
1639 : :
1640 : : /* Copy the plan's query string into the portal */
2922 peter_e@gmx.net 1641 : 6187 : query_string = MemoryContextStrdup(portal->portalContext,
1642 : : plansource->query_string);
1643 : :
1644 : : /*
1645 : : * Setup error traceback support for ereport(), in case GetCachedPlan
1646 : : * throws an error.
1647 : : */
1807 tgl@sss.pgh.pa.us 1648 : 6187 : spicallbackarg.query = plansource->query_string;
1649 : 6187 : spicallbackarg.mode = plan->parse_mode;
4703 1650 : 6187 : spierrcontext.callback = _SPI_error_callback;
1807 1651 : 6187 : spierrcontext.arg = &spicallbackarg;
4703 1652 : 6187 : spierrcontext.previous = error_context_stack;
1653 : 6187 : error_context_stack = &spierrcontext;
1654 : :
1655 : : /*
1656 : : * Note: for a saved plan, we mustn't have any failure occur between
1657 : : * GetCachedPlan and PortalDefineQuery; that would result in leaking our
1658 : : * plancache refcount.
1659 : : */
1660 : :
1661 : : /* Replan if needed, and increment plan refcount for portal */
1786 1662 : 6187 : cplan = GetCachedPlan(plansource, paramLI, NULL, _SPI_current->queryEnv);
5205 1663 : 6187 : stmt_list = cplan->stmt_list;
1664 : :
1665 [ + + ]: 6187 : if (!plan->saved)
1666 : : {
1667 : : /*
1668 : : * We don't want the portal to depend on an unsaved CachedPlanSource,
1669 : : * so must copy the plan into the portal's context. An error here
1670 : : * will result in leaking our refcount on the plan, but it doesn't
1671 : : * matter because the plan is unsaved and hence transient anyway.
1672 : : */
2922 peter_e@gmx.net 1673 : 4817 : oldcontext = MemoryContextSwitchTo(portal->portalContext);
5205 tgl@sss.pgh.pa.us 1674 : 4817 : stmt_list = copyObject(stmt_list);
6851 1675 : 4817 : MemoryContextSwitchTo(oldcontext);
1786 1676 : 4817 : ReleaseCachedPlan(cplan, NULL);
6851 1677 : 4817 : cplan = NULL; /* portal shouldn't depend on cplan */
1678 : : }
1679 : :
1680 : : /*
1681 : : * Set up the portal.
1682 : : */
8264 1683 : 6187 : PortalDefineQuery(portal,
1684 : : NULL, /* no statement name */
1685 : : query_string,
1686 : : plansource->commandTag,
1687 : : stmt_list,
1688 : : cplan);
1689 : :
1690 : : /*
1691 : : * Set up options for portal. Default SCROLL type is chosen the same way
1692 : : * as PerformCursorOpen does it.
1693 : : */
6819 1694 : 6187 : portal->cursorOptions = plan->cursor_options;
1695 [ + + ]: 6187 : if (!(portal->cursorOptions & (CURSOR_OPT_SCROLL | CURSOR_OPT_NO_SCROLL)))
1696 : : {
1697 [ + - ]: 210 : if (list_length(stmt_list) == 1 &&
3100 1698 [ + - ]: 210 : linitial_node(PlannedStmt, stmt_list)->commandType != CMD_UTILITY &&
3172 1699 [ + + + + ]: 419 : linitial_node(PlannedStmt, stmt_list)->rowMarks == NIL &&
1700 : 209 : ExecSupportsBackwardScan(linitial_node(PlannedStmt, stmt_list)->planTree))
6819 1701 : 194 : portal->cursorOptions |= CURSOR_OPT_SCROLL;
1702 : : else
1703 : 16 : portal->cursorOptions |= CURSOR_OPT_NO_SCROLL;
1704 : : }
1705 : :
1706 : : /*
1707 : : * Disallow SCROLL with SELECT FOR UPDATE. This is not redundant with the
1708 : : * check in transformDeclareCursorStmt because the cursor options might
1709 : : * not have come through there.
1710 : : */
6628 1711 [ + + ]: 6187 : if (portal->cursorOptions & CURSOR_OPT_SCROLL)
1712 : : {
1713 [ + - ]: 207 : if (list_length(stmt_list) == 1 &&
3100 1714 [ + - ]: 207 : linitial_node(PlannedStmt, stmt_list)->commandType != CMD_UTILITY &&
3172 1715 [ - + ]: 207 : linitial_node(PlannedStmt, stmt_list)->rowMarks != NIL)
6628 tgl@sss.pgh.pa.us 1716 [ # # ]:UBC 0 : ereport(ERROR,
1717 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
1718 : : errmsg("DECLARE SCROLL CURSOR ... FOR UPDATE/SHARE is not supported"),
1719 : : errdetail("Scrollable cursors must be READ ONLY.")));
1720 : : }
1721 : :
1722 : : /* Make current query environment available to portal at execution time. */
3178 kgrittn@postgresql.o 1723 :CBC 6187 : portal->queryEnv = _SPI_current->queryEnv;
1724 : :
1725 : : /*
1726 : : * If told to be read-only, we'd better check for read-only queries. This
1727 : : * can't be done earlier because we need to look at the finished, planned
1728 : : * queries. (In particular, we don't want to do it between GetCachedPlan
1729 : : * and PortalDefineQuery, because throwing an error between those steps
1730 : : * would result in leaking our plancache refcount.)
1731 : : */
2161 rhaas@postgresql.org 1732 [ + + ]: 6187 : if (read_only)
1733 : : {
1734 : : ListCell *lc;
1735 : :
6849 tgl@sss.pgh.pa.us 1736 [ + - + + : 156 : foreach(lc, stmt_list)
+ + ]
1737 : : {
3172 1738 : 78 : PlannedStmt *pstmt = lfirst_node(PlannedStmt, lc);
1739 : :
6849 1740 [ - + ]: 78 : if (!CommandIsReadOnly(pstmt))
2161 rhaas@postgresql.org 1741 [ # # ]:UBC 0 : ereport(ERROR,
1742 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
1743 : : /* translator: %s is a SQL statement name */
1744 : : errmsg("%s is not allowed in a non-volatile function",
1745 : : CreateCommandName((Node *) pstmt))));
1746 : : }
1747 : : }
1748 : :
1749 : : /* Set up the snapshot to use. */
4768 tgl@sss.pgh.pa.us 1750 [ + + ]:CBC 6187 : if (read_only)
1751 : 78 : snapshot = GetActiveSnapshot();
1752 : : else
1753 : : {
1754 : 6109 : CommandCounterIncrement();
1755 : 6109 : snapshot = GetTransactionSnapshot();
1756 : : }
1757 : :
1758 : : /*
1759 : : * If the plan has parameters, copy them into the portal. Note that this
1760 : : * must be done after revalidating the plan, because in dynamic parameter
1761 : : * cases the set of parameters could have changed during re-parsing.
1762 : : */
5886 1763 [ + + ]: 6187 : if (paramLI)
1764 : : {
2922 peter_e@gmx.net 1765 : 338 : oldcontext = MemoryContextSwitchTo(portal->portalContext);
5886 tgl@sss.pgh.pa.us 1766 : 338 : paramLI = copyParamList(paramLI);
1767 : 338 : MemoryContextSwitchTo(oldcontext);
1768 : : }
1769 : :
1770 : : /*
1771 : : * Start portal execution.
1772 : : */
4768 1773 : 6187 : PortalStart(portal, paramLI, 0, snapshot);
1774 : :
7064 1775 [ - + ]: 6187 : Assert(portal->strategy != PORTAL_MULTI_QUERY);
1776 : :
1777 : : /* Pop the error context stack */
3164 1778 : 6187 : error_context_stack = spierrcontext.previous;
1779 : :
1780 : : /* Pop the SPI stack */
5205 1781 : 6187 : _SPI_end_call(true);
1782 : :
1783 : : /* Return the created portal */
8975 JanWieck@Yahoo.com 1784 : 6187 : return portal;
1785 : : }
1786 : :
1787 : :
1788 : : /*
1789 : : * SPI_cursor_find()
1790 : : *
1791 : : * Find the portal of an existing open cursor
1792 : : */
1793 : : Portal
8387 tgl@sss.pgh.pa.us 1794 : 281 : SPI_cursor_find(const char *name)
1795 : : {
8975 JanWieck@Yahoo.com 1796 : 281 : return GetPortalByName(name);
1797 : : }
1798 : :
1799 : :
1800 : : /*
1801 : : * SPI_cursor_fetch()
1802 : : *
1803 : : * Fetch rows in a cursor
1804 : : */
1805 : : void
7044 bruce@momjian.us 1806 : 22114 : SPI_cursor_fetch(Portal portal, bool forward, long count)
1807 : : {
6819 tgl@sss.pgh.pa.us 1808 : 22114 : _SPI_cursor_operation(portal,
1809 : 22114 : forward ? FETCH_FORWARD : FETCH_BACKWARD, count,
1810 : : CreateDestReceiver(DestSPI));
1811 : : /* we know that the DestSPI receiver doesn't need a destroy call */
8975 JanWieck@Yahoo.com 1812 : 22110 : }
1813 : :
1814 : :
1815 : : /*
1816 : : * SPI_cursor_move()
1817 : : *
1818 : : * Move in a cursor
1819 : : */
1820 : : void
7044 bruce@momjian.us 1821 :UBC 0 : SPI_cursor_move(Portal portal, bool forward, long count)
1822 : : {
6819 tgl@sss.pgh.pa.us 1823 : 0 : _SPI_cursor_operation(portal,
1824 : 0 : forward ? FETCH_FORWARD : FETCH_BACKWARD, count,
1825 : : None_Receiver);
1826 : 0 : }
1827 : :
1828 : :
1829 : : /*
1830 : : * SPI_scroll_cursor_fetch()
1831 : : *
1832 : : * Fetch rows in a scrollable cursor
1833 : : */
1834 : : void
6819 tgl@sss.pgh.pa.us 1835 :CBC 151 : SPI_scroll_cursor_fetch(Portal portal, FetchDirection direction, long count)
1836 : : {
1837 : 151 : _SPI_cursor_operation(portal,
1838 : : direction, count,
1839 : : CreateDestReceiver(DestSPI));
1840 : : /* we know that the DestSPI receiver doesn't need a destroy call */
1841 : 148 : }
1842 : :
1843 : :
1844 : : /*
1845 : : * SPI_scroll_cursor_move()
1846 : : *
1847 : : * Move in a scrollable cursor
1848 : : */
1849 : : void
1850 : 21 : SPI_scroll_cursor_move(Portal portal, FetchDirection direction, long count)
1851 : : {
1852 : 21 : _SPI_cursor_operation(portal, direction, count, None_Receiver);
8975 JanWieck@Yahoo.com 1853 : 21 : }
1854 : :
1855 : :
1856 : : /*
1857 : : * SPI_cursor_close()
1858 : : *
1859 : : * Close a cursor
1860 : : */
1861 : : void
1862 : 6132 : SPI_cursor_close(Portal portal)
1863 : : {
8838 tgl@sss.pgh.pa.us 1864 [ - + ]: 6132 : if (!PortalIsValid(portal))
8975 JanWieck@Yahoo.com 1865 [ # # ]:UBC 0 : elog(ERROR, "invalid portal in SPI cursor operation");
1866 : :
8300 bruce@momjian.us 1867 :CBC 6132 : PortalDrop(portal, false);
8975 JanWieck@Yahoo.com 1868 : 6132 : }
1869 : :
1870 : : /*
1871 : : * Returns the Oid representing the type id for argument at argIndex. First
1872 : : * parameter is at index zero.
1873 : : */
1874 : : Oid
6851 tgl@sss.pgh.pa.us 1875 :UBC 0 : SPI_getargtypeid(SPIPlanPtr plan, int argIndex)
1876 : : {
1877 [ # # # # : 0 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC ||
# # ]
1878 [ # # ]: 0 : argIndex < 0 || argIndex >= plan->nargs)
1879 : : {
7940 1880 : 0 : SPI_result = SPI_ERROR_ARGUMENT;
1881 : 0 : return InvalidOid;
1882 : : }
6851 1883 : 0 : return plan->argtypes[argIndex];
1884 : : }
1885 : :
1886 : : /*
1887 : : * Returns the number of arguments for the prepared plan.
1888 : : */
1889 : : int
1890 : 0 : SPI_getargcount(SPIPlanPtr plan)
1891 : : {
1892 [ # # # # ]: 0 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC)
1893 : : {
7940 1894 : 0 : SPI_result = SPI_ERROR_ARGUMENT;
1895 : 0 : return -1;
1896 : : }
6851 1897 : 0 : return plan->nargs;
1898 : : }
1899 : :
1900 : : /*
1901 : : * Returns true if the plan contains exactly one command
1902 : : * and that command returns tuples to the caller (eg, SELECT or
1903 : : * INSERT ... RETURNING, but not SELECT ... INTO). In essence,
1904 : : * the result indicates if the command can be used with SPI_cursor_open
1905 : : *
1906 : : * Parameters
1907 : : * plan: A plan previously prepared using SPI_prepare
1908 : : */
1909 : : bool
6851 tgl@sss.pgh.pa.us 1910 :CBC 6187 : SPI_is_cursor_plan(SPIPlanPtr plan)
1911 : : {
1912 : : CachedPlanSource *plansource;
1913 : :
1914 [ + - - + ]: 6187 : if (plan == NULL || plan->magic != _SPI_PLAN_MAGIC)
1915 : : {
7940 tgl@sss.pgh.pa.us 1916 :UBC 0 : SPI_result = SPI_ERROR_ARGUMENT;
1917 : 0 : return false;
1918 : : }
1919 : :
6851 tgl@sss.pgh.pa.us 1920 [ - + ]:CBC 6187 : if (list_length(plan->plancache_list) != 1)
1921 : : {
6517 tgl@sss.pgh.pa.us 1922 :UBC 0 : SPI_result = 0;
7064 1923 : 0 : return false; /* not exactly 1 pre-rewrite command */
1924 : : }
6851 tgl@sss.pgh.pa.us 1925 :CBC 6187 : plansource = (CachedPlanSource *) linitial(plan->plancache_list);
1926 : :
1927 : : /*
1928 : : * We used to force revalidation of the cached plan here, but that seems
1929 : : * unnecessary: invalidation could mean a change in the rowtype of the
1930 : : * tuples returned by a plan, but not whether it returns tuples at all.
1931 : : */
6517 1932 : 6187 : SPI_result = 0;
1933 : :
1934 : : /* Does it return tuples? */
6851 1935 [ + - ]: 6187 : if (plansource->resultDesc)
1936 : 6187 : return true;
1937 : :
7940 tgl@sss.pgh.pa.us 1938 :UBC 0 : return false;
1939 : : }
1940 : :
1941 : : /*
1942 : : * SPI_plan_is_valid --- test whether a SPI plan is currently valid
1943 : : * (that is, not marked as being in need of revalidation).
1944 : : *
1945 : : * See notes for CachedPlanIsValid before using this.
1946 : : */
1947 : : bool
6301 tgl@sss.pgh.pa.us 1948 :CBC 1850 : SPI_plan_is_valid(SPIPlanPtr plan)
1949 : : {
1950 : : ListCell *lc;
1951 : :
5205 1952 [ - + ]: 1850 : Assert(plan->magic == _SPI_PLAN_MAGIC);
1953 : :
1954 [ + - + + : 3523 : foreach(lc, plan->plancache_list)
+ + ]
1955 : : {
1956 : 1850 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
1957 : :
1958 [ + + ]: 1850 : if (!CachedPlanIsValid(plansource))
1959 : 177 : return false;
1960 : : }
1961 : 1673 : return true;
1962 : : }
1963 : :
1964 : : /*
1965 : : * SPI_result_code_string --- convert any SPI return code to a string
1966 : : *
1967 : : * This is often useful in error messages. Most callers will probably
1968 : : * only pass negative (error-case) codes, but for generality we recognize
1969 : : * the success codes too.
1970 : : */
1971 : : const char *
7808 1972 : 60 : SPI_result_code_string(int code)
1973 : : {
1974 : : static char buf[64];
1975 : :
1976 [ - - - - : 60 : switch (code)
- - + - -
- - - - -
- + + - +
- - - - -
- - - - -
- - - ]
1977 : : {
7808 tgl@sss.pgh.pa.us 1978 :UBC 0 : case SPI_ERROR_CONNECT:
1979 : 0 : return "SPI_ERROR_CONNECT";
1980 : 0 : case SPI_ERROR_COPY:
1981 : 0 : return "SPI_ERROR_COPY";
1982 : 0 : case SPI_ERROR_OPUNKNOWN:
1983 : 0 : return "SPI_ERROR_OPUNKNOWN";
1984 : 0 : case SPI_ERROR_UNCONNECTED:
1985 : 0 : return "SPI_ERROR_UNCONNECTED";
1986 : 0 : case SPI_ERROR_ARGUMENT:
1987 : 0 : return "SPI_ERROR_ARGUMENT";
1988 : 0 : case SPI_ERROR_PARAM:
1989 : 0 : return "SPI_ERROR_PARAM";
7808 tgl@sss.pgh.pa.us 1990 :CBC 3 : case SPI_ERROR_TRANSACTION:
1991 : 3 : return "SPI_ERROR_TRANSACTION";
7808 tgl@sss.pgh.pa.us 1992 :UBC 0 : case SPI_ERROR_NOATTRIBUTE:
1993 : 0 : return "SPI_ERROR_NOATTRIBUTE";
1994 : 0 : case SPI_ERROR_NOOUTFUNC:
1995 : 0 : return "SPI_ERROR_NOOUTFUNC";
1996 : 0 : case SPI_ERROR_TYPUNKNOWN:
1997 : 0 : return "SPI_ERROR_TYPUNKNOWN";
3182 kgrittn@postgresql.o 1998 : 0 : case SPI_ERROR_REL_DUPLICATE:
1999 : 0 : return "SPI_ERROR_REL_DUPLICATE";
2000 : 0 : case SPI_ERROR_REL_NOT_FOUND:
2001 : 0 : return "SPI_ERROR_REL_NOT_FOUND";
7808 tgl@sss.pgh.pa.us 2002 : 0 : case SPI_OK_CONNECT:
2003 : 0 : return "SPI_OK_CONNECT";
2004 : 0 : case SPI_OK_FINISH:
2005 : 0 : return "SPI_OK_FINISH";
2006 : 0 : case SPI_OK_FETCH:
2007 : 0 : return "SPI_OK_FETCH";
7808 tgl@sss.pgh.pa.us 2008 :CBC 1 : case SPI_OK_UTILITY:
2009 : 1 : return "SPI_OK_UTILITY";
2010 : 11 : case SPI_OK_SELECT:
2011 : 11 : return "SPI_OK_SELECT";
7808 tgl@sss.pgh.pa.us 2012 :UBC 0 : case SPI_OK_SELINTO:
2013 : 0 : return "SPI_OK_SELINTO";
7808 tgl@sss.pgh.pa.us 2014 :CBC 45 : case SPI_OK_INSERT:
2015 : 45 : return "SPI_OK_INSERT";
7808 tgl@sss.pgh.pa.us 2016 :UBC 0 : case SPI_OK_DELETE:
2017 : 0 : return "SPI_OK_DELETE";
2018 : 0 : case SPI_OK_UPDATE:
2019 : 0 : return "SPI_OK_UPDATE";
2020 : 0 : case SPI_OK_CURSOR:
2021 : 0 : return "SPI_OK_CURSOR";
7051 2022 : 0 : case SPI_OK_INSERT_RETURNING:
2023 : 0 : return "SPI_OK_INSERT_RETURNING";
2024 : 0 : case SPI_OK_DELETE_RETURNING:
2025 : 0 : return "SPI_OK_DELETE_RETURNING";
2026 : 0 : case SPI_OK_UPDATE_RETURNING:
2027 : 0 : return "SPI_OK_UPDATE_RETURNING";
6173 heikki.linnakangas@i 2028 : 0 : case SPI_OK_REWRITTEN:
2029 : 0 : return "SPI_OK_REWRITTEN";
3182 kgrittn@postgresql.o 2030 : 0 : case SPI_OK_REL_REGISTER:
2031 : 0 : return "SPI_OK_REL_REGISTER";
2032 : 0 : case SPI_OK_REL_UNREGISTER:
2033 : 0 : return "SPI_OK_REL_UNREGISTER";
1028 dean.a.rasheed@gmail 2034 : 0 : case SPI_OK_TD_REGISTER:
2035 : 0 : return "SPI_OK_TD_REGISTER";
2036 : 0 : case SPI_OK_MERGE:
2037 : 0 : return "SPI_OK_MERGE";
639 2038 : 0 : case SPI_OK_MERGE_RETURNING:
2039 : 0 : return "SPI_OK_MERGE_RETURNING";
2040 : : }
2041 : : /* Unrecognized code ... return something useful ... */
7808 tgl@sss.pgh.pa.us 2042 : 0 : sprintf(buf, "Unrecognized SPI code %d", code);
2043 : 0 : return buf;
2044 : : }
2045 : :
2046 : : /*
2047 : : * SPI_plan_get_plan_sources --- get a SPI plan's underlying list of
2048 : : * CachedPlanSources.
2049 : : *
2050 : : * CAUTION: there is no check on whether the CachedPlanSources are up-to-date.
2051 : : *
2052 : : * This is exported so that PL/pgSQL can use it (this beats letting PL/pgSQL
2053 : : * look directly into the SPIPlan for itself). It's not documented in
2054 : : * spi.sgml because we'd just as soon not have too many places using this.
2055 : : */
2056 : : List *
4703 tgl@sss.pgh.pa.us 2057 :CBC 31326 : SPI_plan_get_plan_sources(SPIPlanPtr plan)
2058 : : {
2059 [ - + ]: 31326 : Assert(plan->magic == _SPI_PLAN_MAGIC);
2060 : 31326 : return plan->plancache_list;
2061 : : }
2062 : :
2063 : : /*
2064 : : * SPI_plan_get_cached_plan --- get a SPI plan's generic CachedPlan,
2065 : : * if the SPI plan contains exactly one CachedPlanSource. If not,
2066 : : * return NULL.
2067 : : *
2068 : : * The plan's refcount is incremented (and logged in CurrentResourceOwner,
2069 : : * if it's a saved plan). Caller is responsible for doing ReleaseCachedPlan.
2070 : : *
2071 : : * This is exported so that PL/pgSQL can use it (this beats letting PL/pgSQL
2072 : : * look directly into the SPIPlan for itself). It's not documented in
2073 : : * spi.sgml because we'd just as soon not have too many places using this.
2074 : : */
2075 : : CachedPlan *
2076 : 15241 : SPI_plan_get_cached_plan(SPIPlanPtr plan)
2077 : : {
2078 : : CachedPlanSource *plansource;
2079 : : CachedPlan *cplan;
2080 : : SPICallbackArg spicallbackarg;
2081 : : ErrorContextCallback spierrcontext;
2082 : :
2083 [ - + ]: 15241 : Assert(plan->magic == _SPI_PLAN_MAGIC);
2084 : :
2085 : : /* Can't support one-shot plans here */
2086 [ - + ]: 15241 : if (plan->oneshot)
4703 tgl@sss.pgh.pa.us 2087 :UBC 0 : return NULL;
2088 : :
2089 : : /* Must have exactly one CachedPlanSource */
4703 tgl@sss.pgh.pa.us 2090 [ - + ]:CBC 15241 : if (list_length(plan->plancache_list) != 1)
4703 tgl@sss.pgh.pa.us 2091 :UBC 0 : return NULL;
4703 tgl@sss.pgh.pa.us 2092 :CBC 15241 : plansource = (CachedPlanSource *) linitial(plan->plancache_list);
2093 : :
2094 : : /* Setup error traceback support for ereport() */
1807 2095 : 15241 : spicallbackarg.query = plansource->query_string;
2096 : 15241 : spicallbackarg.mode = plan->parse_mode;
4703 2097 : 15241 : spierrcontext.callback = _SPI_error_callback;
1807 2098 : 15241 : spierrcontext.arg = &spicallbackarg;
4703 2099 : 15241 : spierrcontext.previous = error_context_stack;
2100 : 15241 : error_context_stack = &spierrcontext;
2101 : :
2102 : : /* Get the generic plan for the query */
1786 2103 : 15241 : cplan = GetCachedPlan(plansource, NULL,
2104 : 15241 : plan->saved ? CurrentResourceOwner : NULL,
3182 kgrittn@postgresql.o 2105 [ + + ]: 15241 : _SPI_current->queryEnv);
4703 tgl@sss.pgh.pa.us 2106 [ - + ]: 15222 : Assert(cplan == plansource->gplan);
2107 : :
2108 : : /* Pop the error context stack */
2109 : 15222 : error_context_stack = spierrcontext.previous;
2110 : :
2111 : 15222 : return cplan;
2112 : : }
2113 : :
2114 : :
2115 : : /* =================== private functions =================== */
2116 : :
2117 : : /*
2118 : : * spi_dest_startup
2119 : : * Initialize to receive tuples from Executor into SPITupleTable
2120 : : * of current SPI procedure
2121 : : */
2122 : : void
8258 2123 : 48132 : spi_dest_startup(DestReceiver *self, int operation, TupleDesc typeinfo)
2124 : : {
2125 : : SPITupleTable *tuptable;
2126 : : MemoryContext oldcxt;
2127 : : MemoryContext tuptabcxt;
2128 : :
3325 2129 [ - + ]: 48132 : if (_SPI_current == NULL)
3325 tgl@sss.pgh.pa.us 2130 [ # # ]:UBC 0 : elog(ERROR, "spi_dest_startup called while not connected to SPI");
2131 : :
8365 tgl@sss.pgh.pa.us 2132 [ - + ]:CBC 48132 : if (_SPI_current->tuptable != NULL)
8184 tgl@sss.pgh.pa.us 2133 [ # # ]:UBC 0 : elog(ERROR, "improper call to spi_dest_startup");
2134 : :
2135 : : /* We create the tuple table context as a child of procCxt */
2136 : :
8365 tgl@sss.pgh.pa.us 2137 :CBC 48132 : oldcxt = _SPI_procmem(); /* switch to procedure memory context */
2138 : :
2139 : 48132 : tuptabcxt = AllocSetContextCreate(CurrentMemoryContext,
2140 : : "SPI TupTable",
2141 : : ALLOCSET_DEFAULT_SIZES);
2142 : 48132 : MemoryContextSwitchTo(tuptabcxt);
2143 : :
6 michael@paquier.xyz 2144 :GNC 48132 : _SPI_current->tuptable = tuptable = palloc0_object(SPITupleTable);
8365 tgl@sss.pgh.pa.us 2145 :CBC 48132 : tuptable->tuptabcxt = tuptabcxt;
4527 2146 : 48132 : tuptable->subid = GetCurrentSubTransactionId();
2147 : :
2148 : : /*
2149 : : * The tuptable is now valid enough to be freed by AtEOSubXact_SPI, so put
2150 : : * it onto the SPI context's tuptables list. This will ensure it's not
2151 : : * leaked even in the unlikely event the following few lines fail.
2152 : : */
2153 : 48132 : slist_push_head(&_SPI_current->tuptables, &tuptable->next);
2154 : :
2155 : : /* set up initial allocations */
2343 2156 : 48132 : tuptable->alloced = 128;
6 michael@paquier.xyz 2157 :GNC 48132 : tuptable->vals = palloc_array(HeapTuple, tuptable->alloced);
2343 tgl@sss.pgh.pa.us 2158 :CBC 48132 : tuptable->numvals = 0;
8365 2159 : 48132 : tuptable->tupdesc = CreateTupleDescCopy(typeinfo);
2160 : :
2161 : 48132 : MemoryContextSwitchTo(oldcxt);
2162 : 48132 : }
2163 : :
2164 : : /*
2165 : : * spi_printtup
2166 : : * store tuple retrieved by Executor into SPITupleTable
2167 : : * of current SPI procedure
2168 : : */
2169 : : bool
7580 2170 : 59490 : spi_printtup(TupleTableSlot *slot, DestReceiver *self)
2171 : : {
2172 : : SPITupleTable *tuptable;
2173 : : MemoryContext oldcxt;
2174 : :
3325 2175 [ - + ]: 59490 : if (_SPI_current == NULL)
3325 tgl@sss.pgh.pa.us 2176 [ # # ]:UBC 0 : elog(ERROR, "spi_printtup called while not connected to SPI");
2177 : :
10327 bruce@momjian.us 2178 :CBC 59490 : tuptable = _SPI_current->tuptable;
2179 [ - + ]: 59490 : if (tuptable == NULL)
8184 tgl@sss.pgh.pa.us 2180 [ # # ]:UBC 0 : elog(ERROR, "improper call to spi_printtup");
2181 : :
8365 tgl@sss.pgh.pa.us 2182 :CBC 59490 : oldcxt = MemoryContextSwitchTo(tuptable->tuptabcxt);
2183 : :
2343 2184 [ + + ]: 59490 : if (tuptable->numvals >= tuptable->alloced)
2185 : : {
2186 : : /* Double the size of the pointer array */
2187 : 2 : uint64 newalloced = tuptable->alloced * 2;
2188 : :
3564 2189 : 2 : tuptable->vals = (HeapTuple *) repalloc_huge(tuptable->vals,
2190 : : newalloced * sizeof(HeapTuple));
2343 2191 : 2 : tuptable->alloced = newalloced;
2192 : : }
2193 : :
2194 : 59490 : tuptable->vals[tuptable->numvals] = ExecCopySlotHeapTuple(slot);
2195 : 59490 : (tuptable->numvals)++;
2196 : :
10327 bruce@momjian.us 2197 : 59490 : MemoryContextSwitchTo(oldcxt);
2198 : :
3480 rhaas@postgresql.org 2199 : 59490 : return true;
2200 : : }
2201 : :
2202 : : /*
2203 : : * Static functions
2204 : : */
2205 : :
2206 : : /*
2207 : : * Parse and analyze a querystring.
2208 : : *
2209 : : * At entry, plan->argtypes and plan->nargs (or alternatively plan->parserSetup
2210 : : * and plan->parserSetupArg) must be valid, as must plan->parse_mode and
2211 : : * plan->cursor_options.
2212 : : *
2213 : : * Results are stored into *plan (specifically, plan->plancache_list).
2214 : : * Note that the result data is all in CurrentMemoryContext or child contexts
2215 : : * thereof; in practice this means it is in the SPI executor context, and
2216 : : * what we are creating is a "temporary" SPIPlan. Cruft generated during
2217 : : * parsing is also left in CurrentMemoryContext.
2218 : : */
2219 : : static void
4729 tgl@sss.pgh.pa.us 2220 : 22250 : _SPI_prepare_plan(const char *src, SPIPlanPtr plan)
2221 : : {
2222 : : List *raw_parsetree_list;
2223 : : List *plancache_list;
2224 : : ListCell *list_item;
2225 : : SPICallbackArg spicallbackarg;
2226 : : ErrorContextCallback spierrcontext;
2227 : :
2228 : : /*
2229 : : * Setup error traceback support for ereport()
2230 : : */
1807 2231 : 22250 : spicallbackarg.query = src;
2232 : 22250 : spicallbackarg.mode = plan->parse_mode;
7940 2233 : 22250 : spierrcontext.callback = _SPI_error_callback;
1807 2234 : 22250 : spierrcontext.arg = &spicallbackarg;
7940 2235 : 22250 : spierrcontext.previous = error_context_stack;
2236 : 22250 : error_context_stack = &spierrcontext;
2237 : :
2238 : : /*
2239 : : * Parse the request string into a list of raw parse trees.
2240 : : */
1807 2241 : 22250 : raw_parsetree_list = raw_parser(src, plan->parse_mode);
2242 : :
2243 : : /*
2244 : : * Do parse analysis and rule rewrite for each raw parsetree, storing the
2245 : : * results into unsaved plancache entries.
2246 : : */
6851 2247 : 22250 : plancache_list = NIL;
2248 : :
8464 2249 [ + - + + : 44448 : foreach(list_item, raw_parsetree_list)
+ + ]
2250 : : {
3172 2251 : 22250 : RawStmt *parsetree = lfirst_node(RawStmt, list_item);
2252 : : List *stmt_list;
2253 : : CachedPlanSource *plansource;
2254 : :
2255 : : /*
2256 : : * Create the CachedPlanSource before we do parse analysis, since it
2257 : : * needs to see the unmodified raw parse tree.
2258 : : */
5205 2259 : 22250 : plansource = CreateCachedPlan(parsetree,
2260 : : src,
2261 : : CreateCommandTag(parsetree->stmt));
2262 : :
2263 : : /*
2264 : : * Parameter datatypes are driven by parserSetup hook if provided,
2265 : : * otherwise we use the fixed parameter list.
2266 : : */
5886 2267 [ + + ]: 22250 : if (plan->parserSetup != NULL)
2268 : : {
2269 [ - + ]: 14843 : Assert(plan->nargs == 0);
1383 peter@eisentraut.org 2270 : 14843 : stmt_list = pg_analyze_and_rewrite_withcb(parsetree,
2271 : : src,
2272 : : plan->parserSetup,
2273 : : plan->parserSetupArg,
3182 kgrittn@postgresql.o 2274 : 14843 : _SPI_current->queryEnv);
2275 : : }
2276 : : else
2277 : : {
1383 peter@eisentraut.org 2278 : 7407 : stmt_list = pg_analyze_and_rewrite_fixedparams(parsetree,
2279 : : src,
1314 tgl@sss.pgh.pa.us 2280 : 7407 : plan->argtypes,
2281 : : plan->nargs,
2282 : 7407 : _SPI_current->queryEnv);
2283 : : }
2284 : :
2285 : : /* Finish filling in the CachedPlanSource */
5205 2286 : 22198 : CompleteCachedPlan(plansource,
2287 : : stmt_list,
2288 : : NULL,
2289 : : plan->argtypes,
2290 : : plan->nargs,
2291 : : plan->parserSetup,
2292 : : plan->parserSetupArg,
2293 : : plan->cursor_options,
2294 : : false); /* not fixed result */
2295 : :
6851 2296 : 22198 : plancache_list = lappend(plancache_list, plansource);
2297 : : }
2298 : :
2299 : 22198 : plan->plancache_list = plancache_list;
4729 2300 : 22198 : plan->oneshot = false;
2301 : :
2302 : : /*
2303 : : * Pop the error context stack
2304 : : */
2305 : 22198 : error_context_stack = spierrcontext.previous;
2306 : 22198 : }
2307 : :
2308 : : /*
2309 : : * Parse, but don't analyze, a querystring.
2310 : : *
2311 : : * This is a stripped-down version of _SPI_prepare_plan that only does the
2312 : : * initial raw parsing. It creates "one shot" CachedPlanSources
2313 : : * that still require parse analysis before execution is possible.
2314 : : *
2315 : : * The advantage of using the "one shot" form of CachedPlanSource is that
2316 : : * we eliminate data copying and invalidation overhead. Postponing parse
2317 : : * analysis also prevents issues if some of the raw parsetrees are DDL
2318 : : * commands that affect validity of later parsetrees. Both of these
2319 : : * attributes are good things for SPI_execute() and similar cases.
2320 : : *
2321 : : * Results are stored into *plan (specifically, plan->plancache_list).
2322 : : * Note that the result data is all in CurrentMemoryContext or child contexts
2323 : : * thereof; in practice this means it is in the SPI executor context, and
2324 : : * what we are creating is a "temporary" SPIPlan. Cruft generated during
2325 : : * parsing is also left in CurrentMemoryContext.
2326 : : */
2327 : : static void
2328 : 8011 : _SPI_prepare_oneshot_plan(const char *src, SPIPlanPtr plan)
2329 : : {
2330 : : List *raw_parsetree_list;
2331 : : List *plancache_list;
2332 : : ListCell *list_item;
2333 : : SPICallbackArg spicallbackarg;
2334 : : ErrorContextCallback spierrcontext;
2335 : :
2336 : : /*
2337 : : * Setup error traceback support for ereport()
2338 : : */
1807 2339 : 8011 : spicallbackarg.query = src;
2340 : 8011 : spicallbackarg.mode = plan->parse_mode;
4729 2341 : 8011 : spierrcontext.callback = _SPI_error_callback;
1807 2342 : 8011 : spierrcontext.arg = &spicallbackarg;
4729 2343 : 8011 : spierrcontext.previous = error_context_stack;
2344 : 8011 : error_context_stack = &spierrcontext;
2345 : :
2346 : : /*
2347 : : * Parse the request string into a list of raw parse trees.
2348 : : */
1807 2349 : 8011 : raw_parsetree_list = raw_parser(src, plan->parse_mode);
2350 : :
2351 : : /*
2352 : : * Construct plancache entries, but don't do parse analysis yet.
2353 : : */
4729 2354 : 8005 : plancache_list = NIL;
2355 : :
2356 [ + - + + : 16015 : foreach(list_item, raw_parsetree_list)
+ + ]
2357 : : {
3172 2358 : 8010 : RawStmt *parsetree = lfirst_node(RawStmt, list_item);
2359 : : CachedPlanSource *plansource;
2360 : :
4729 2361 : 8010 : plansource = CreateOneShotCachedPlan(parsetree,
2362 : : src,
2363 : : CreateCommandTag(parsetree->stmt));
2364 : :
2365 : 8010 : plancache_list = lappend(plancache_list, plansource);
2366 : : }
2367 : :
2368 : 8005 : plan->plancache_list = plancache_list;
2369 : 8005 : plan->oneshot = true;
2370 : :
2371 : : /*
2372 : : * Pop the error context stack
2373 : : */
7940 2374 : 8005 : error_context_stack = spierrcontext.previous;
10336 vadim4o@yahoo.com 2375 : 8005 : }
2376 : :
2377 : : /*
2378 : : * _SPI_execute_plan: execute the given plan with the given options
2379 : : *
2380 : : * options contains options accessible from outside SPI:
2381 : : * params: parameter values to pass to query
2382 : : * read_only: true for read-only execution (no CommandCounterIncrement)
2383 : : * allow_nonatomic: true to allow nonatomic CALL/DO execution
2384 : : * must_return_tuples: throw error if query doesn't return tuples
2385 : : * tcount: execution tuple-count limit, or 0 for none
2386 : : * dest: DestReceiver to receive output, or NULL for normal SPI output
2387 : : * owner: ResourceOwner that will be used to hold refcount on plan;
2388 : : * if NULL, CurrentResourceOwner is used (ignored for non-saved plan)
2389 : : *
2390 : : * Additional, only-internally-accessible options:
2391 : : * snapshot: query snapshot to use, or InvalidSnapshot for the normal
2392 : : * behavior of taking a new snapshot for each query.
2393 : : * crosscheck_snapshot: for RI use, all others pass InvalidSnapshot
2394 : : * fire_triggers: true to fire AFTER triggers at end of query (normal case);
2395 : : * false means any AFTER triggers are postponed to end of outer query
2396 : : */
2397 : : static int
1535 tgl@sss.pgh.pa.us 2398 : 52161 : _SPI_execute_plan(SPIPlanPtr plan, const SPIExecuteOptions *options,
2399 : : Snapshot snapshot, Snapshot crosscheck_snapshot,
2400 : : bool fire_triggers)
2401 : : {
6427 alvherre@alvh.no-ip. 2402 : 52161 : int my_res = 0;
3566 tgl@sss.pgh.pa.us 2403 : 52161 : uint64 my_processed = 0;
6427 alvherre@alvh.no-ip. 2404 : 52161 : SPITupleTable *my_tuptable = NULL;
2405 : 52161 : int res = 0;
2406 : : bool allow_nonatomic;
5405 tgl@sss.pgh.pa.us 2407 : 52161 : bool pushed_active_snap = false;
1535 2408 : 52161 : ResourceOwner plan_owner = options->owner;
2409 : : SPICallbackArg spicallbackarg;
2410 : : ErrorContextCallback spierrcontext;
6427 alvherre@alvh.no-ip. 2411 : 52161 : CachedPlan *cplan = NULL;
2412 : : ListCell *lc1;
2413 : :
2414 : : /*
2415 : : * We allow nonatomic behavior only if options->allow_nonatomic is set
2416 : : * *and* the SPI_OPT_NONATOMIC flag was given when connecting and we are
2417 : : * not inside a subtransaction. The latter two tests match whether
2418 : : * _SPI_commit() would allow a commit; see there for more commentary.
2419 : : */
426 tgl@sss.pgh.pa.us 2420 : 104381 : allow_nonatomic = options->allow_nonatomic &&
2421 [ + + + + : 52161 : !_SPI_current->atomic && !IsSubTransaction();
+ + ]
2422 : :
2423 : : /*
2424 : : * Setup error traceback support for ereport()
2425 : : */
1807 2426 : 52161 : spicallbackarg.query = NULL; /* we'll fill this below */
2427 : 52161 : spicallbackarg.mode = plan->parse_mode;
6427 alvherre@alvh.no-ip. 2428 : 52161 : spierrcontext.callback = _SPI_error_callback;
1807 tgl@sss.pgh.pa.us 2429 : 52161 : spierrcontext.arg = &spicallbackarg;
6427 alvherre@alvh.no-ip. 2430 : 52161 : spierrcontext.previous = error_context_stack;
2431 : 52161 : error_context_stack = &spierrcontext;
2432 : :
2433 : : /*
2434 : : * We support four distinct snapshot management behaviors:
2435 : : *
2436 : : * snapshot != InvalidSnapshot, read_only = true: use exactly the given
2437 : : * snapshot.
2438 : : *
2439 : : * snapshot != InvalidSnapshot, read_only = false: use the given snapshot,
2440 : : * modified by advancing its command ID before each querytree.
2441 : : *
2442 : : * snapshot == InvalidSnapshot, read_only = true: do nothing for queries
2443 : : * that require no snapshot. For those that do, ensure that a Portal
2444 : : * snapshot exists; then use that, or use the entry-time ActiveSnapshot if
2445 : : * that exists and is different.
2446 : : *
2447 : : * snapshot == InvalidSnapshot, read_only = false: do nothing for queries
2448 : : * that require no snapshot. For those that do, ensure that a Portal
2449 : : * snapshot exists; then, in atomic execution (!allow_nonatomic) take a
2450 : : * full new snapshot for each user command, and advance its command ID
2451 : : * before each querytree within the command. In allow_nonatomic mode we
2452 : : * just use the Portal snapshot unmodified.
2453 : : *
2454 : : * In the first two cases, we can just push the snap onto the stack once
2455 : : * for the whole plan list.
2456 : : *
2457 : : * Note that snapshot != InvalidSnapshot implies an atomic execution
2458 : : * context.
2459 : : */
1670 tgl@sss.pgh.pa.us 2460 [ + + ]: 52161 : if (snapshot != InvalidSnapshot)
2461 : : {
2462 : : /* this intentionally tests the options field not the derived value */
1535 2463 [ - + ]: 626 : Assert(!options->allow_nonatomic);
2464 [ + + ]: 626 : if (options->read_only)
2465 : : {
5405 2466 : 590 : PushActiveSnapshot(snapshot);
2467 : 590 : pushed_active_snap = true;
2468 : : }
2469 : : else
2470 : : {
2471 : : /* Make sure we have a private copy of the snapshot to modify */
2472 : 36 : PushCopiedSnapshot(snapshot);
2473 : 36 : pushed_active_snap = true;
2474 : : }
2475 : : }
2476 : :
2477 : : /*
2478 : : * Ensure that we have a resource owner if plan is saved, and not if it
2479 : : * isn't.
2480 : : */
1786 2481 [ + + ]: 52161 : if (!plan->saved)
2482 : 8669 : plan_owner = NULL;
2483 [ + + ]: 43492 : else if (plan_owner == NULL)
2484 : 43437 : plan_owner = CurrentResourceOwner;
2485 : :
2486 : : /*
2487 : : * We interpret must_return_tuples as "there must be at least one query,
2488 : : * and all of them must return tuples". This is a bit laxer than
2489 : : * SPI_is_cursor_plan's check, but there seems no reason to enforce that
2490 : : * there be only one query.
2491 : : */
1535 2492 [ + + - + ]: 52161 : if (options->must_return_tuples && plan->plancache_list == NIL)
1535 tgl@sss.pgh.pa.us 2493 [ # # ]:UBC 0 : ereport(ERROR,
2494 : : (errcode(ERRCODE_SYNTAX_ERROR),
2495 : : errmsg("empty query does not return tuples")));
2496 : :
6427 alvherre@alvh.no-ip. 2497 [ + - + + :CBC 101460 : foreach(lc1, plan->plancache_list)
+ + ]
2498 : : {
2499 : 52165 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc1);
2500 : : List *stmt_list;
2501 : : ListCell *lc2;
2502 : :
1807 tgl@sss.pgh.pa.us 2503 : 52165 : spicallbackarg.query = plansource->query_string;
2504 : :
2505 : : /*
2506 : : * If this is a one-shot plan, we still need to do parse analysis.
2507 : : */
4729 2508 [ + + ]: 52165 : if (plan->oneshot)
2509 : : {
3258 2510 : 8009 : RawStmt *parsetree = plansource->raw_parse_tree;
4729 2511 : 8009 : const char *src = plansource->query_string;
2512 : : List *querytree_list;
2513 : :
2514 : : /*
2515 : : * Parameter datatypes are driven by parserSetup hook if provided,
2516 : : * otherwise we use the fixed parameter list.
2517 : : */
4052 2518 [ - + ]: 8009 : if (parsetree == NULL)
1168 drowley@postgresql.o 2519 :UBC 0 : querytree_list = NIL;
4052 tgl@sss.pgh.pa.us 2520 [ + + ]:CBC 8009 : else if (plan->parserSetup != NULL)
2521 : : {
4729 2522 [ - + ]: 285 : Assert(plan->nargs == 0);
1168 drowley@postgresql.o 2523 : 285 : querytree_list = pg_analyze_and_rewrite_withcb(parsetree,
2524 : : src,
2525 : : plan->parserSetup,
2526 : : plan->parserSetupArg,
2527 : 285 : _SPI_current->queryEnv);
2528 : : }
2529 : : else
2530 : : {
2531 : 7724 : querytree_list = pg_analyze_and_rewrite_fixedparams(parsetree,
2532 : : src,
2533 : 7724 : plan->argtypes,
2534 : : plan->nargs,
2535 : 7724 : _SPI_current->queryEnv);
2536 : : }
2537 : :
2538 : : /* Finish filling in the CachedPlanSource */
4729 tgl@sss.pgh.pa.us 2539 : 8002 : CompleteCachedPlan(plansource,
2540 : : querytree_list,
2541 : : NULL,
2542 : : plan->argtypes,
2543 : : plan->nargs,
2544 : : plan->parserSetup,
2545 : : plan->parserSetupArg,
2546 : : plan->cursor_options,
2547 : : false); /* not fixed result */
2548 : : }
2549 : :
2550 : : /*
2551 : : * If asked to, complain when query does not return tuples.
2552 : : * (Replanning can't change this, so we can check it before that.
2553 : : * However, we can't check it till after parse analysis, so in the
2554 : : * case of a one-shot plan this is the earliest we could check.)
2555 : : */
1535 2556 [ + + + + ]: 52158 : if (options->must_return_tuples && !plansource->resultDesc)
2557 : : {
2558 : : /* try to give a good error message */
2559 : : const char *cmdtag;
2560 : :
2561 : : /* A SELECT without resultDesc must be SELECT INTO */
2562 [ + - ]: 6 : if (plansource->commandTag == CMDTAG_SELECT)
2563 : 6 : cmdtag = "SELECT INTO";
2564 : : else
1535 tgl@sss.pgh.pa.us 2565 :UBC 0 : cmdtag = GetCommandTagName(plansource->commandTag);
1535 tgl@sss.pgh.pa.us 2566 [ + - ]:CBC 6 : ereport(ERROR,
2567 : : (errcode(ERRCODE_SYNTAX_ERROR),
2568 : : /* translator: %s is name of a SQL command, eg INSERT */
2569 : : errmsg("%s query does not return tuples", cmdtag)));
2570 : : }
2571 : :
2572 : : /*
2573 : : * Replan if needed, and increment plan refcount. If it's a saved
2574 : : * plan, the refcount must be backed by the plan_owner.
2575 : : */
2576 : 52152 : cplan = GetCachedPlan(plansource, options->params,
1786 2577 : 52152 : plan_owner, _SPI_current->queryEnv);
2578 : :
5205 2579 : 52081 : stmt_list = cplan->stmt_list;
2580 : :
2581 : : /*
2582 : : * If we weren't given a specific snapshot to use, and the statement
2583 : : * list requires a snapshot, set that up.
2584 : : */
1670 2585 [ + + + - ]: 103536 : if (snapshot == InvalidSnapshot &&
2586 [ + - ]: 102910 : (list_length(stmt_list) > 1 ||
2587 [ + + ]: 102910 : (list_length(stmt_list) == 1 &&
2588 : 51455 : PlannedStmtRequiresSnapshot(linitial_node(PlannedStmt,
2589 : : stmt_list)))))
2590 : : {
2591 : : /*
2592 : : * First, ensure there's a Portal-level snapshot. This back-fills
2593 : : * the snapshot stack in case the previous operation was a COMMIT
2594 : : * or ROLLBACK inside a procedure or DO block. (We can't put back
2595 : : * the Portal snapshot any sooner, or we'd break cases like doing
2596 : : * SET or LOCK just after COMMIT.) It's enough to check once per
2597 : : * statement list, since COMMIT/ROLLBACK/CALL/DO can't appear
2598 : : * within a multi-statement list.
2599 : : */
2600 : 44885 : EnsurePortalSnapshotExists();
2601 : :
2602 : : /*
2603 : : * In the default non-read-only case, get a new per-statement-list
2604 : : * snapshot, replacing any that we pushed in a previous cycle.
2605 : : * Skip it when doing non-atomic execution, though (we rely
2606 : : * entirely on the Portal snapshot in that case).
2607 : : */
557 2608 [ + + + + ]: 44885 : if (!options->read_only && !allow_nonatomic)
2609 : : {
1670 2610 [ + + ]: 42396 : if (pushed_active_snap)
2611 : 4 : PopActiveSnapshot();
2612 : 42396 : PushActiveSnapshot(GetTransactionSnapshot());
2613 : 42396 : pushed_active_snap = true;
2614 : : }
2615 : : }
2616 : :
6427 alvherre@alvh.no-ip. 2617 [ + - + + : 101380 : foreach(lc2, stmt_list)
+ + ]
2618 : : {
3172 tgl@sss.pgh.pa.us 2619 : 52081 : PlannedStmt *stmt = lfirst_node(PlannedStmt, lc2);
3258 2620 : 52081 : bool canSetTag = stmt->canSetTag;
2621 : : DestReceiver *dest;
2622 : :
2623 : : /*
2624 : : * Reset output state. (Note that if a non-SPI receiver is used,
2625 : : * _SPI_current->processed will stay zero, and that's what we'll
2626 : : * report to the caller. It's the receiver's job to count tuples
2627 : : * in that case.)
2628 : : */
6427 alvherre@alvh.no-ip. 2629 : 52081 : _SPI_current->processed = 0;
2630 : 52081 : _SPI_current->tuptable = NULL;
2631 : :
2632 : : /* Check for unsupported cases. */
3258 tgl@sss.pgh.pa.us 2633 [ + + ]: 52081 : if (stmt->utilityStmt)
2634 : : {
2635 [ + + ]: 12491 : if (IsA(stmt->utilityStmt, CopyStmt))
2636 : : {
2637 : 9 : CopyStmt *cstmt = (CopyStmt *) stmt->utilityStmt;
2638 : :
6427 alvherre@alvh.no-ip. 2639 [ + + ]: 9 : if (cstmt->filename == NULL)
2640 : : {
2641 : 4 : my_res = SPI_ERROR_COPY;
7764 tgl@sss.pgh.pa.us 2642 : 9 : goto fail;
2643 : : }
2644 : : }
3258 2645 [ + + ]: 12482 : else if (IsA(stmt->utilityStmt, TransactionStmt))
2646 : : {
6427 alvherre@alvh.no-ip. 2647 : 5 : my_res = SPI_ERROR_TRANSACTION;
2648 : 5 : goto fail;
2649 : : }
2650 : : }
2651 : :
1535 tgl@sss.pgh.pa.us 2652 [ + + - + ]: 52072 : if (options->read_only && !CommandIsReadOnly(stmt))
6427 alvherre@alvh.no-ip. 2653 [ # # ]:UBC 0 : ereport(ERROR,
2654 : : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
2655 : : /* translator: %s is a SQL statement name */
2656 : : errmsg("%s is not allowed in a non-volatile function",
2657 : : CreateCommandName((Node *) stmt))));
2658 : :
2659 : : /*
2660 : : * If not read-only mode, advance the command counter before each
2661 : : * command and update the snapshot. (But skip it if the snapshot
2662 : : * isn't under our control.)
2663 : : */
1535 tgl@sss.pgh.pa.us 2664 [ + + + + ]:CBC 52072 : if (!options->read_only && pushed_active_snap)
2665 : : {
6427 alvherre@alvh.no-ip. 2666 : 42428 : CommandCounterIncrement();
5405 tgl@sss.pgh.pa.us 2667 : 42428 : UpdateActiveSnapshotCommandId();
2668 : : }
2669 : :
2670 : : /*
2671 : : * Select appropriate tuple receiver. Output from non-canSetTag
2672 : : * subqueries always goes to the bit bucket.
2673 : : */
2013 2674 [ - + ]: 52072 : if (!canSetTag)
2013 tgl@sss.pgh.pa.us 2675 :UBC 0 : dest = CreateDestReceiver(DestNone);
1535 tgl@sss.pgh.pa.us 2676 [ + + ]:CBC 52072 : else if (options->dest)
2677 : 1341 : dest = options->dest;
2678 : : else
2013 2679 : 50731 : dest = CreateDestReceiver(DestSPI);
2680 : :
3258 2681 [ + + ]: 52072 : if (stmt->utilityStmt == NULL)
2682 : : {
2683 : : QueryDesc *qdesc;
2684 : : Snapshot snap;
2685 : :
6427 alvherre@alvh.no-ip. 2686 [ + - ]: 39590 : if (ActiveSnapshotSet())
2687 : 39590 : snap = GetActiveSnapshot();
2688 : : else
6427 alvherre@alvh.no-ip. 2689 :UBC 0 : snap = InvalidSnapshot;
2690 : :
3258 tgl@sss.pgh.pa.us 2691 :CBC 39590 : qdesc = CreateQueryDesc(stmt,
2692 : : plansource->query_string,
2693 : : snap, crosscheck_snapshot,
2694 : : dest,
1535 2695 : 39590 : options->params,
2696 : 39590 : _SPI_current->queryEnv,
2697 : : 0);
208 amitlan@postgresql.o 2698 [ + - ]: 39590 : res = _SPI_pquery(qdesc, fire_triggers,
2699 : : canSetTag ? options->tcount : 0);
6427 alvherre@alvh.no-ip. 2700 : 36886 : FreeQueryDesc(qdesc);
2701 : : }
2702 : : else
2703 : : {
2704 : : ProcessUtilityContext context;
2705 : : QueryCompletion qc;
2706 : :
2707 : : /*
2708 : : * If we're not allowing nonatomic operations, tell
2709 : : * ProcessUtility this is an atomic execution context.
2710 : : */
557 tgl@sss.pgh.pa.us 2711 [ + + ]: 12482 : if (allow_nonatomic)
2824 peter_e@gmx.net 2712 : 51 : context = PROCESS_UTILITY_QUERY_NONATOMIC;
2713 : : else
557 tgl@sss.pgh.pa.us 2714 : 12431 : context = PROCESS_UTILITY_QUERY;
2715 : :
2115 alvherre@alvh.no-ip. 2716 : 12482 : InitializeQueryCompletion(&qc);
6427 2717 : 12482 : ProcessUtility(stmt,
2718 : : plansource->query_string,
2719 : : true, /* protect plancache's node tree */
2720 : : context,
1535 tgl@sss.pgh.pa.us 2721 : 12482 : options->params,
3182 kgrittn@postgresql.o 2722 : 12482 : _SPI_current->queryEnv,
2723 : : dest,
2724 : : &qc);
2725 : :
2726 : : /* Update "processed" if stmt returned tuples */
6427 alvherre@alvh.no-ip. 2727 [ + + ]: 12413 : if (_SPI_current->tuptable)
2343 tgl@sss.pgh.pa.us 2728 : 937 : _SPI_current->processed = _SPI_current->tuptable->numvals;
2729 : :
4822 heikki.linnakangas@i 2730 : 12413 : res = SPI_OK_UTILITY;
2731 : :
2732 : : /*
2733 : : * Some utility statements return a row count, even though the
2734 : : * tuples are not returned to the caller.
2735 : : */
3258 tgl@sss.pgh.pa.us 2736 [ + + ]: 12413 : if (IsA(stmt->utilityStmt, CreateTableAsStmt))
2737 : : {
2738 : 25 : CreateTableAsStmt *ctastmt = (CreateTableAsStmt *) stmt->utilityStmt;
2739 : :
2115 alvherre@alvh.no-ip. 2740 [ + + ]: 25 : if (qc.commandTag == CMDTAG_SELECT)
2741 : 22 : _SPI_current->processed = qc.nprocessed;
2742 : : else
2743 : : {
2744 : : /*
2745 : : * Must be an IF NOT EXISTS that did nothing, or a
2746 : : * CREATE ... WITH NO DATA.
2747 : : */
3414 tgl@sss.pgh.pa.us 2748 [ + - - + ]: 3 : Assert(ctastmt->if_not_exists ||
2749 : : ctastmt->into->skipData);
3536 2750 : 3 : _SPI_current->processed = 0;
2751 : : }
2752 : :
2753 : : /*
2754 : : * For historical reasons, if CREATE TABLE AS was spelled
2755 : : * as SELECT INTO, return a special return code.
2756 : : */
2757 [ - + ]: 25 : if (ctastmt->is_select_into)
5020 tgl@sss.pgh.pa.us 2758 :UBC 0 : res = SPI_OK_SELINTO;
2759 : : }
3258 tgl@sss.pgh.pa.us 2760 [ + + ]:CBC 12388 : else if (IsA(stmt->utilityStmt, CopyStmt))
2761 : : {
2115 alvherre@alvh.no-ip. 2762 [ - + ]: 5 : Assert(qc.commandTag == CMDTAG_COPY);
2763 : 5 : _SPI_current->processed = qc.nprocessed;
2764 : : }
2765 : : }
2766 : :
2767 : : /*
2768 : : * The last canSetTag query sets the status values returned to the
2769 : : * caller. Be careful to free any tuptables not returned, to
2770 : : * avoid intra-transaction memory leak.
2771 : : */
6427 2772 [ + - ]: 49299 : if (canSetTag)
2773 : : {
2774 : 49299 : my_processed = _SPI_current->processed;
2775 : 49299 : SPI_freetuptable(my_tuptable);
2776 : 49299 : my_tuptable = _SPI_current->tuptable;
2777 : 49299 : my_res = res;
2778 : : }
2779 : : else
2780 : : {
6427 alvherre@alvh.no-ip. 2781 :UBC 0 : SPI_freetuptable(_SPI_current->tuptable);
2782 : 0 : _SPI_current->tuptable = NULL;
2783 : : }
2784 : :
2785 : : /*
2786 : : * We don't issue a destroy call to the receiver. The SPI and
2787 : : * None receivers would ignore it anyway, while if the caller
2788 : : * supplied a receiver, it's not our job to destroy it.
2789 : : */
2790 : :
6427 alvherre@alvh.no-ip. 2791 [ - + ]:CBC 49299 : if (res < 0)
2792 : : {
6427 alvherre@alvh.no-ip. 2793 :UBC 0 : my_res = res;
2794 : 0 : goto fail;
2795 : : }
2796 : : }
2797 : :
2798 : : /* Done with this plan, so release refcount */
1786 tgl@sss.pgh.pa.us 2799 :CBC 49299 : ReleaseCachedPlan(cplan, plan_owner);
6427 alvherre@alvh.no-ip. 2800 : 49299 : cplan = NULL;
2801 : :
2802 : : /*
2803 : : * If not read-only mode, advance the command counter after the last
2804 : : * command. This ensures that its effects are visible, in case it was
2805 : : * DDL that would affect the next CachedPlanSource.
2806 : : */
1535 tgl@sss.pgh.pa.us 2807 [ + + ]: 49299 : if (!options->read_only)
6427 alvherre@alvh.no-ip. 2808 : 46298 : CommandCounterIncrement();
2809 : : }
2810 : :
2811 : 49304 : fail:
2812 : :
2813 : : /* Pop the snapshot off the stack if we pushed one */
5405 tgl@sss.pgh.pa.us 2814 [ + + ]: 49304 : if (pushed_active_snap)
2815 : 40281 : PopActiveSnapshot();
2816 : :
2817 : : /* We no longer need the cached plan refcount, if any */
6427 alvherre@alvh.no-ip. 2818 [ + + ]: 49304 : if (cplan)
1786 tgl@sss.pgh.pa.us 2819 : 9 : ReleaseCachedPlan(cplan, plan_owner);
2820 : :
2821 : : /*
2822 : : * Pop the error context stack
2823 : : */
6427 alvherre@alvh.no-ip. 2824 : 49304 : error_context_stack = spierrcontext.previous;
2825 : :
2826 : : /* Save results for caller */
7381 tgl@sss.pgh.pa.us 2827 : 49304 : SPI_processed = my_processed;
2828 : 49304 : SPI_tuptable = my_tuptable;
2829 : :
2830 : : /* tuptable now is caller's responsibility, not SPI's */
6948 2831 : 49304 : _SPI_current->tuptable = NULL;
2832 : :
2833 : : /*
2834 : : * If none of the queries had canSetTag, return SPI_OK_REWRITTEN. Prior to
2835 : : * 8.4, we used return the last query's result code, but not its auxiliary
2836 : : * results, but that's confusing.
2837 : : */
6930 2838 [ - + ]: 49304 : if (my_res == 0)
6173 heikki.linnakangas@i 2839 :UBC 0 : my_res = SPI_OK_REWRITTEN;
2840 : :
7064 tgl@sss.pgh.pa.us 2841 :CBC 49304 : return my_res;
2842 : : }
2843 : :
2844 : : /*
2845 : : * Convert arrays of query parameters to form wanted by planner and executor
2846 : : */
2847 : : static ParamListInfo
6468 2848 : 6322 : _SPI_convert_params(int nargs, Oid *argtypes,
2849 : : const Datum *Values, const char *Nulls)
2850 : : {
2851 : : ParamListInfo paramLI;
2852 : :
2853 [ + + ]: 6322 : if (nargs > 0)
2854 : : {
2469 peter@eisentraut.org 2855 : 5542 : paramLI = makeParamList(nargs);
2856 : :
2857 [ + + ]: 14197 : for (int i = 0; i < nargs; i++)
2858 : : {
6468 tgl@sss.pgh.pa.us 2859 : 8655 : ParamExternData *prm = ¶mLI->params[i];
2860 : :
2861 : 8655 : prm->value = Values[i];
2862 [ + + - + ]: 8655 : prm->isnull = (Nulls && Nulls[i] == 'n');
5205 2863 : 8655 : prm->pflags = PARAM_FLAG_CONST;
6468 2864 : 8655 : prm->ptype = argtypes[i];
2865 : : }
2866 : : }
2867 : : else
2868 : 780 : paramLI = NULL;
2869 : 6322 : return paramLI;
2870 : : }
2871 : :
2872 : : static int
208 amitlan@postgresql.o 2873 : 39590 : _SPI_pquery(QueryDesc *queryDesc, bool fire_triggers, uint64 tcount)
2874 : : {
10309 vadim4o@yahoo.com 2875 : 39590 : int operation = queryDesc->operation;
2876 : : int eflags;
2877 : : int res;
2878 : :
10327 bruce@momjian.us 2879 [ + + + + : 39590 : switch (operation)
+ - ]
2880 : : {
10326 2881 : 25754 : case CMD_SELECT:
2013 tgl@sss.pgh.pa.us 2882 [ - + ]: 25754 : if (queryDesc->dest->mydest == DestNone)
2883 : : {
2884 : : /* Don't return SPI_OK_SELECT if we're discarding result */
7381 tgl@sss.pgh.pa.us 2885 :UBC 0 : res = SPI_OK_UTILITY;
2886 : : }
2887 : : else
7066 tgl@sss.pgh.pa.us 2888 :CBC 25754 : res = SPI_OK_SELECT;
10326 bruce@momjian.us 2889 : 25754 : break;
2890 : 8765 : case CMD_INSERT:
5911 tgl@sss.pgh.pa.us 2891 [ + + ]: 8765 : if (queryDesc->plannedstmt->hasReturning)
7051 2892 : 473 : res = SPI_OK_INSERT_RETURNING;
2893 : : else
2894 : 8292 : res = SPI_OK_INSERT;
10326 bruce@momjian.us 2895 : 8765 : break;
2896 : 4220 : case CMD_DELETE:
5911 tgl@sss.pgh.pa.us 2897 [ - + ]: 4220 : if (queryDesc->plannedstmt->hasReturning)
7051 tgl@sss.pgh.pa.us 2898 :UBC 0 : res = SPI_OK_DELETE_RETURNING;
2899 : : else
7051 tgl@sss.pgh.pa.us 2900 :CBC 4220 : res = SPI_OK_DELETE;
10326 bruce@momjian.us 2901 : 4220 : break;
2902 : 818 : case CMD_UPDATE:
5911 tgl@sss.pgh.pa.us 2903 [ + + ]: 818 : if (queryDesc->plannedstmt->hasReturning)
7051 2904 : 5 : res = SPI_OK_UPDATE_RETURNING;
2905 : : else
2906 : 813 : res = SPI_OK_UPDATE;
10326 bruce@momjian.us 2907 : 818 : break;
1359 alvherre@alvh.no-ip. 2908 : 33 : case CMD_MERGE:
639 dean.a.rasheed@gmail 2909 [ + + ]: 33 : if (queryDesc->plannedstmt->hasReturning)
2910 : 9 : res = SPI_OK_MERGE_RETURNING;
2911 : : else
2912 : 24 : res = SPI_OK_MERGE;
1359 alvherre@alvh.no-ip. 2913 : 33 : break;
10326 bruce@momjian.us 2914 :UBC 0 : default:
9968 2915 : 0 : return SPI_ERROR_OPUNKNOWN;
2916 : : }
2917 : :
2918 : : #ifdef SPI_EXECUTOR_STATS
2919 : : if (ShowExecutorStats)
2920 : : ResetUsage();
2921 : : #endif
2922 : :
2923 : : /* Select execution options */
6698 tgl@sss.pgh.pa.us 2924 [ + + ]:CBC 39590 : if (fire_triggers)
5406 2925 : 35595 : eflags = 0; /* default run-to-completion flags */
2926 : : else
2927 : 3995 : eflags = EXEC_FLAG_SKIP_TRIGGERS;
2928 : :
208 amitlan@postgresql.o 2929 : 39590 : ExecutorStart(queryDesc, eflags);
2930 : :
372 tgl@sss.pgh.pa.us 2931 : 39590 : ExecutorRun(queryDesc, ForwardScanDirection, tcount);
2932 : :
8412 2933 : 36888 : _SPI_current->processed = queryDesc->estate->es_processed;
2934 : :
5911 2935 [ + + + + ]: 36888 : if ((res == SPI_OK_SELECT || queryDesc->plannedstmt->hasReturning) &&
7051 2936 [ + + ]: 23557 : queryDesc->dest->mydest == DestSPI)
2937 : : {
10309 vadim4o@yahoo.com 2938 [ - + ]: 22247 : if (_SPI_checktuples())
8184 tgl@sss.pgh.pa.us 2939 [ # # ]:UBC 0 : elog(ERROR, "consistency check on SPI tuple count failed");
2940 : : }
2941 : :
5406 tgl@sss.pgh.pa.us 2942 :CBC 36888 : ExecutorFinish(queryDesc);
7571 2943 : 36886 : ExecutorEnd(queryDesc);
2944 : : /* FreeQueryDesc is done by the caller */
2945 : :
2946 : : #ifdef SPI_EXECUTOR_STATS
2947 : : if (ShowExecutorStats)
2948 : : ShowUsage("SPI EXECUTOR STATS");
2949 : : #endif
2950 : :
8402 2951 : 36886 : return res;
2952 : : }
2953 : :
2954 : : /*
2955 : : * _SPI_error_callback
2956 : : *
2957 : : * Add context information when a query invoked via SPI fails
2958 : : */
2959 : : static void
7940 2960 : 3270 : _SPI_error_callback(void *arg)
2961 : : {
1807 2962 : 3270 : SPICallbackArg *carg = (SPICallbackArg *) arg;
2963 : 3270 : const char *query = carg->query;
2964 : : int syntaxerrposition;
2965 : :
2927 2966 [ - + ]: 3270 : if (query == NULL) /* in case arg wasn't set yet */
2927 tgl@sss.pgh.pa.us 2967 :UBC 0 : return;
2968 : :
2969 : : /*
2970 : : * If there is a syntax error position, convert to internal syntax error;
2971 : : * otherwise treat the query as an item of context stack
2972 : : */
7940 tgl@sss.pgh.pa.us 2973 :CBC 3270 : syntaxerrposition = geterrposition();
2974 [ + + ]: 3270 : if (syntaxerrposition > 0)
2975 : : {
2976 : 53 : errposition(0);
2977 : 53 : internalerrposition(syntaxerrposition);
2978 : 53 : internalerrquery(query);
2979 : : }
2980 : : else
2981 : : {
2982 : : /* Use the parse mode to decide how to describe the query */
1807 2983 [ + + + ]: 3217 : switch (carg->mode)
2984 : : {
2985 : 41 : case RAW_PARSE_PLPGSQL_EXPR:
467 peter@eisentraut.org 2986 : 41 : errcontext("PL/pgSQL expression \"%s\"", query);
1807 tgl@sss.pgh.pa.us 2987 : 41 : break;
2988 : 8 : case RAW_PARSE_PLPGSQL_ASSIGN1:
2989 : : case RAW_PARSE_PLPGSQL_ASSIGN2:
2990 : : case RAW_PARSE_PLPGSQL_ASSIGN3:
2991 : 8 : errcontext("PL/pgSQL assignment \"%s\"", query);
2992 : 8 : break;
2993 : 3168 : default:
2994 : 3168 : errcontext("SQL statement \"%s\"", query);
2995 : 3168 : break;
2996 : : }
2997 : : }
2998 : : }
2999 : :
3000 : : /*
3001 : : * _SPI_cursor_operation()
3002 : : *
3003 : : * Do a FETCH or MOVE in a cursor
3004 : : */
3005 : : static void
6819 3006 : 22286 : _SPI_cursor_operation(Portal portal, FetchDirection direction, long count,
3007 : : DestReceiver *dest)
3008 : : {
3009 : : uint64 nfetched;
3010 : :
3011 : : /* Check that the portal is valid */
8975 JanWieck@Yahoo.com 3012 [ - + ]: 22286 : if (!PortalIsValid(portal))
8975 JanWieck@Yahoo.com 3013 [ # # ]:UBC 0 : elog(ERROR, "invalid portal in SPI cursor operation");
3014 : :
3015 : : /* Push the SPI stack */
8120 tgl@sss.pgh.pa.us 3016 [ - + ]:CBC 22286 : if (_SPI_begin_call(true) < 0)
8120 tgl@sss.pgh.pa.us 3017 [ # # ]:UBC 0 : elog(ERROR, "SPI cursor operation called while not connected");
3018 : :
3019 : : /* Reset the SPI result (note we deliberately don't touch lastoid) */
8975 JanWieck@Yahoo.com 3020 :CBC 22286 : SPI_processed = 0;
3021 : 22286 : SPI_tuptable = NULL;
3022 : 22286 : _SPI_current->processed = 0;
3023 : 22286 : _SPI_current->tuptable = NULL;
3024 : :
3025 : : /* Run the cursor */
8166 tgl@sss.pgh.pa.us 3026 : 22286 : nfetched = PortalRunFetch(portal,
3027 : : direction,
3028 : : count,
3029 : : dest);
3030 : :
3031 : : /*
3032 : : * Think not to combine this store with the preceding function call. If
3033 : : * the portal contains calls to functions that use SPI, then _SPI_stack is
3034 : : * likely to move around while the portal runs. When control returns,
3035 : : * _SPI_current will point to the correct stack entry... but the pointer
3036 : : * may be different than it was beforehand. So we must be sure to re-fetch
3037 : : * the pointer after the function call completes.
3038 : : */
3039 : 22279 : _SPI_current->processed = nfetched;
3040 : :
7348 alvherre@alvh.no-ip. 3041 [ + + - + ]: 22279 : if (dest->mydest == DestSPI && _SPI_checktuples())
8184 tgl@sss.pgh.pa.us 3042 [ # # ]:UBC 0 : elog(ERROR, "consistency check on SPI tuple count failed");
3043 : :
3044 : : /* Put the result into place for access by caller */
8975 JanWieck@Yahoo.com 3045 :CBC 22279 : SPI_processed = _SPI_current->processed;
8818 bruce@momjian.us 3046 : 22279 : SPI_tuptable = _SPI_current->tuptable;
3047 : :
3048 : : /* tuptable now is caller's responsibility, not SPI's */
6948 tgl@sss.pgh.pa.us 3049 : 22279 : _SPI_current->tuptable = NULL;
3050 : :
3051 : : /* Pop the SPI stack */
8975 JanWieck@Yahoo.com 3052 : 22279 : _SPI_end_call(true);
3053 : 22279 : }
3054 : :
3055 : :
3056 : : static MemoryContext
7734 neilc@samurai.com 3057 : 102890 : _SPI_execmem(void)
3058 : : {
9302 tgl@sss.pgh.pa.us 3059 : 102890 : return MemoryContextSwitchTo(_SPI_current->execCxt);
3060 : : }
3061 : :
3062 : : static MemoryContext
7734 neilc@samurai.com 3063 : 148100 : _SPI_procmem(void)
3064 : : {
9302 tgl@sss.pgh.pa.us 3065 : 148100 : return MemoryContextSwitchTo(_SPI_current->procCxt);
3066 : : }
3067 : :
3068 : : /*
3069 : : * _SPI_begin_call: begin a SPI operation within a connected procedure
3070 : : *
3071 : : * use_exec is true if we intend to make use of the procedure's execCxt
3072 : : * during this SPI operation. We'll switch into that context, and arrange
3073 : : * for it to be cleaned up at _SPI_end_call or if an error occurs.
3074 : : */
3075 : : static int
2993 3076 : 154070 : _SPI_begin_call(bool use_exec)
3077 : : {
3325 3078 [ - + ]: 154070 : if (_SPI_current == NULL)
9968 bruce@momjian.us 3079 :UBC 0 : return SPI_ERROR_UNCONNECTED;
3080 : :
2993 tgl@sss.pgh.pa.us 3081 [ + + ]:CBC 154070 : if (use_exec)
3082 : : {
3083 : : /* remember when the Executor operation started */
3084 : 102890 : _SPI_current->execSubid = GetCurrentSubTransactionId();
3085 : : /* switch to the Executor memory context */
10327 bruce@momjian.us 3086 : 102890 : _SPI_execmem();
3087 : : }
3088 : :
9968 3089 : 154070 : return 0;
3090 : : }
3091 : :
3092 : : /*
3093 : : * _SPI_end_call: end a SPI operation within a connected procedure
3094 : : *
3095 : : * use_exec must be the same as in the previous _SPI_begin_call
3096 : : *
3097 : : * Note: this currently has no failure return cases, so callers don't check
3098 : : */
3099 : : static int
2993 tgl@sss.pgh.pa.us 3100 : 100325 : _SPI_end_call(bool use_exec)
3101 : : {
3102 [ + + ]: 100325 : if (use_exec)
3103 : : {
3104 : : /* switch to the procedure memory context */
10327 bruce@momjian.us 3105 : 99968 : _SPI_procmem();
3106 : : /* mark Executor context no longer in use */
2993 tgl@sss.pgh.pa.us 3107 : 99968 : _SPI_current->execSubid = InvalidSubTransactionId;
3108 : : /* and free Executor memory */
762 nathan@postgresql.or 3109 : 99968 : MemoryContextReset(_SPI_current->execCxt);
3110 : : }
3111 : :
9968 bruce@momjian.us 3112 : 100325 : return 0;
3113 : : }
3114 : :
3115 : : static bool
8902 tgl@sss.pgh.pa.us 3116 : 44505 : _SPI_checktuples(void)
3117 : : {
3566 3118 : 44505 : uint64 processed = _SPI_current->processed;
10326 bruce@momjian.us 3119 : 44505 : SPITupleTable *tuptable = _SPI_current->tuptable;
3120 : 44505 : bool failed = false;
3121 : :
8170 3122 [ - + ]: 44505 : if (tuptable == NULL) /* spi_dest_startup was not called */
8365 tgl@sss.pgh.pa.us 3123 :UBC 0 : failed = true;
2343 tgl@sss.pgh.pa.us 3124 [ - + ]:CBC 44505 : else if (processed != tuptable->numvals)
8365 tgl@sss.pgh.pa.us 3125 :UBC 0 : failed = true;
3126 : :
9968 bruce@momjian.us 3127 :CBC 44505 : return failed;
3128 : : }
3129 : :
3130 : : /*
3131 : : * Convert a "temporary" SPIPlan into an "unsaved" plan.
3132 : : *
3133 : : * The passed _SPI_plan struct is on the stack, and all its subsidiary data
3134 : : * is in or under the current SPI executor context. Copy the plan into the
3135 : : * SPI procedure context so it will survive _SPI_end_call(). To minimize
3136 : : * data copying, this destructively modifies the input plan, by taking the
3137 : : * plancache entries away from it and reparenting them to the new SPIPlan.
3138 : : */
3139 : : static SPIPlanPtr
5205 tgl@sss.pgh.pa.us 3140 : 17482 : _SPI_make_plan_non_temp(SPIPlanPtr plan)
3141 : : {
3142 : : SPIPlanPtr newplan;
3143 : 17482 : MemoryContext parentcxt = _SPI_current->procCxt;
3144 : : MemoryContext plancxt;
3145 : : MemoryContext oldcxt;
3146 : : ListCell *lc;
3147 : :
3148 : : /* Assert the input is a temporary SPIPlan */
3149 [ - + ]: 17482 : Assert(plan->magic == _SPI_PLAN_MAGIC);
3150 [ - + ]: 17482 : Assert(plan->plancxt == NULL);
3151 : : /* One-shot plans can't be saved */
4729 3152 [ - + ]: 17482 : Assert(!plan->oneshot);
3153 : :
3154 : : /*
3155 : : * Create a memory context for the plan, underneath the procedure context.
3156 : : * We don't expect the plan to be very large.
3157 : : */
8975 JanWieck@Yahoo.com 3158 : 17482 : plancxt = AllocSetContextCreate(parentcxt,
3159 : : "SPI Plan",
3160 : : ALLOCSET_SMALL_SIZES);
3161 : 17482 : oldcxt = MemoryContextSwitchTo(plancxt);
3162 : :
3163 : : /* Copy the _SPI_plan struct and subsidiary data into the new context */
6 michael@paquier.xyz 3164 :GNC 17482 : newplan = palloc0_object(_SPI_plan);
6851 tgl@sss.pgh.pa.us 3165 :CBC 17482 : newplan->magic = _SPI_PLAN_MAGIC;
8975 JanWieck@Yahoo.com 3166 : 17482 : newplan->plancxt = plancxt;
1807 tgl@sss.pgh.pa.us 3167 : 17482 : newplan->parse_mode = plan->parse_mode;
6819 3168 : 17482 : newplan->cursor_options = plan->cursor_options;
10327 bruce@momjian.us 3169 : 17482 : newplan->nargs = plan->nargs;
3170 [ + + ]: 17482 : if (plan->nargs > 0)
3171 : : {
6 michael@paquier.xyz 3172 :GNC 1916 : newplan->argtypes = palloc_array(Oid, plan->nargs);
10327 bruce@momjian.us 3173 :CBC 1916 : memcpy(newplan->argtypes, plan->argtypes, plan->nargs * sizeof(Oid));
3174 : : }
3175 : : else
3176 : 15566 : newplan->argtypes = NULL;
5886 tgl@sss.pgh.pa.us 3177 : 17482 : newplan->parserSetup = plan->parserSetup;
3178 : 17482 : newplan->parserSetupArg = plan->parserSetupArg;
3179 : :
3180 : : /*
3181 : : * Reparent all the CachedPlanSources into the procedure context. In
3182 : : * theory this could fail partway through due to the pallocs, but we don't
3183 : : * care too much since both the procedure context and the executor context
3184 : : * would go away on error.
3185 : : */
6851 3186 [ + - + + : 34964 : foreach(lc, plan->plancache_list)
+ + ]
3187 : : {
3188 : 17482 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
3189 : :
5205 3190 : 17482 : CachedPlanSetParentContext(plansource, parentcxt);
3191 : :
3192 : : /* Build new list, with list cells in plancxt */
3193 : 17482 : newplan->plancache_list = lappend(newplan->plancache_list, plansource);
3194 : : }
3195 : :
6851 3196 : 17482 : MemoryContextSwitchTo(oldcxt);
3197 : :
3198 : : /* For safety, unlink the CachedPlanSources from the temporary plan */
5205 3199 : 17482 : plan->plancache_list = NIL;
3200 : :
6851 3201 : 17482 : return newplan;
3202 : : }
3203 : :
3204 : : /*
3205 : : * Make a "saved" copy of the given plan.
3206 : : */
3207 : : static SPIPlanPtr
6851 tgl@sss.pgh.pa.us 3208 :UBC 0 : _SPI_save_plan(SPIPlanPtr plan)
3209 : : {
3210 : : SPIPlanPtr newplan;
3211 : : MemoryContext plancxt;
3212 : : MemoryContext oldcxt;
3213 : : ListCell *lc;
3214 : :
3215 : : /* One-shot plans can't be saved */
4729 3216 [ # # ]: 0 : Assert(!plan->oneshot);
3217 : :
3218 : : /*
3219 : : * Create a memory context for the plan. We don't expect the plan to be
3220 : : * very large, so use smaller-than-default alloc parameters. It's a
3221 : : * transient context until we finish copying everything.
3222 : : */
5205 3223 : 0 : plancxt = AllocSetContextCreate(CurrentMemoryContext,
3224 : : "SPI Plan",
3225 : : ALLOCSET_SMALL_SIZES);
6851 3226 : 0 : oldcxt = MemoryContextSwitchTo(plancxt);
3227 : :
3228 : : /* Copy the SPI plan into its own context */
6 michael@paquier.xyz 3229 :UNC 0 : newplan = palloc0_object(_SPI_plan);
6851 tgl@sss.pgh.pa.us 3230 :UBC 0 : newplan->magic = _SPI_PLAN_MAGIC;
3231 : 0 : newplan->plancxt = plancxt;
1807 3232 : 0 : newplan->parse_mode = plan->parse_mode;
6819 3233 : 0 : newplan->cursor_options = plan->cursor_options;
6851 3234 : 0 : newplan->nargs = plan->nargs;
3235 [ # # ]: 0 : if (plan->nargs > 0)
3236 : : {
6 michael@paquier.xyz 3237 :UNC 0 : newplan->argtypes = palloc_array(Oid, plan->nargs);
6851 tgl@sss.pgh.pa.us 3238 :UBC 0 : memcpy(newplan->argtypes, plan->argtypes, plan->nargs * sizeof(Oid));
3239 : : }
3240 : : else
3241 : 0 : newplan->argtypes = NULL;
5886 3242 : 0 : newplan->parserSetup = plan->parserSetup;
3243 : 0 : newplan->parserSetupArg = plan->parserSetupArg;
3244 : :
3245 : : /* Copy all the plancache entries */
6851 3246 [ # # # # : 0 : foreach(lc, plan->plancache_list)
# # ]
3247 : : {
3248 : 0 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
3249 : : CachedPlanSource *newsource;
3250 : :
5205 3251 : 0 : newsource = CopyCachedPlan(plansource);
6851 3252 : 0 : newplan->plancache_list = lappend(newplan->plancache_list, newsource);
3253 : : }
3254 : :
8975 JanWieck@Yahoo.com 3255 : 0 : MemoryContextSwitchTo(oldcxt);
3256 : :
3257 : : /*
3258 : : * Mark it saved, reparent it under CacheMemoryContext, and mark all the
3259 : : * component CachedPlanSources as saved. This sequence cannot fail
3260 : : * partway through, so there's no risk of long-term memory leakage.
3261 : : */
5205 tgl@sss.pgh.pa.us 3262 : 0 : newplan->saved = true;
3263 : 0 : MemoryContextSetParent(newplan->plancxt, CacheMemoryContext);
3264 : :
3265 [ # # # # : 0 : foreach(lc, newplan->plancache_list)
# # ]
3266 : : {
3267 : 0 : CachedPlanSource *plansource = (CachedPlanSource *) lfirst(lc);
3268 : :
3269 : 0 : SaveCachedPlan(plansource);
3270 : : }
3271 : :
9968 bruce@momjian.us 3272 : 0 : return newplan;
3273 : : }
3274 : :
3275 : : /*
3276 : : * Internal lookup of ephemeral named relation by name.
3277 : : */
3278 : : static EphemeralNamedRelation
3182 kgrittn@postgresql.o 3279 :CBC 357 : _SPI_find_ENR_by_name(const char *name)
3280 : : {
3281 : : /* internal static function; any error is bug in SPI itself */
3282 [ - + ]: 357 : Assert(name != NULL);
3283 : :
3284 : : /* fast exit if no tuplestores have been added */
3285 [ + + ]: 357 : if (_SPI_current->queryEnv == NULL)
3286 : 282 : return NULL;
3287 : :
3288 : 75 : return get_ENR(_SPI_current->queryEnv, name);
3289 : : }
3290 : :
3291 : : /*
3292 : : * Register an ephemeral named relation for use by the planner and executor on
3293 : : * subsequent calls using this SPI connection.
3294 : : */
3295 : : int
3296 : 357 : SPI_register_relation(EphemeralNamedRelation enr)
3297 : : {
3298 : : EphemeralNamedRelation match;
3299 : : int res;
3300 : :
3301 [ + - - + ]: 357 : if (enr == NULL || enr->md.name == NULL)
3182 kgrittn@postgresql.o 3302 :UBC 0 : return SPI_ERROR_ARGUMENT;
3303 : :
3100 tgl@sss.pgh.pa.us 3304 :CBC 357 : res = _SPI_begin_call(false); /* keep current memory context */
3182 kgrittn@postgresql.o 3305 [ - + ]: 357 : if (res < 0)
3182 kgrittn@postgresql.o 3306 :UBC 0 : return res;
3307 : :
3182 kgrittn@postgresql.o 3308 :CBC 357 : match = _SPI_find_ENR_by_name(enr->md.name);
3309 [ - + ]: 357 : if (match)
3182 kgrittn@postgresql.o 3310 :UBC 0 : res = SPI_ERROR_REL_DUPLICATE;
3311 : : else
3312 : : {
3182 kgrittn@postgresql.o 3313 [ + + ]:CBC 357 : if (_SPI_current->queryEnv == NULL)
3314 : 282 : _SPI_current->queryEnv = create_queryEnv();
3315 : :
3316 : 357 : register_ENR(_SPI_current->queryEnv, enr);
3317 : 357 : res = SPI_OK_REL_REGISTER;
3318 : : }
3319 : :
3320 : 357 : _SPI_end_call(false);
3321 : :
3322 : 357 : return res;
3323 : : }
3324 : :
3325 : : /*
3326 : : * Unregister an ephemeral named relation by name. This will probably be a
3327 : : * rarely used function, since SPI_finish will clear it automatically.
3328 : : */
3329 : : int
3182 kgrittn@postgresql.o 3330 :UBC 0 : SPI_unregister_relation(const char *name)
3331 : : {
3332 : : EphemeralNamedRelation match;
3333 : : int res;
3334 : :
3335 [ # # ]: 0 : if (name == NULL)
3336 : 0 : return SPI_ERROR_ARGUMENT;
3337 : :
3100 tgl@sss.pgh.pa.us 3338 : 0 : res = _SPI_begin_call(false); /* keep current memory context */
3182 kgrittn@postgresql.o 3339 [ # # ]: 0 : if (res < 0)
3340 : 0 : return res;
3341 : :
3342 : 0 : match = _SPI_find_ENR_by_name(name);
3343 [ # # ]: 0 : if (match)
3344 : : {
3345 : 0 : unregister_ENR(_SPI_current->queryEnv, match->md.name);
3346 : 0 : res = SPI_OK_REL_UNREGISTER;
3347 : : }
3348 : : else
3349 : 0 : res = SPI_ERROR_REL_NOT_FOUND;
3350 : :
3351 : 0 : _SPI_end_call(false);
3352 : :
3353 : 0 : return res;
3354 : : }
3355 : :
3356 : : /*
3357 : : * Register the transient relations from 'tdata' using this SPI connection.
3358 : : * This should be called by PL implementations' trigger handlers after
3359 : : * connecting, in order to make transition tables visible to any queries run
3360 : : * in this connection.
3361 : : */
3362 : : int
3178 kgrittn@postgresql.o 3363 :CBC 7835 : SPI_register_trigger_data(TriggerData *tdata)
3364 : : {
3365 [ - + ]: 7835 : if (tdata == NULL)
3178 kgrittn@postgresql.o 3366 :UBC 0 : return SPI_ERROR_ARGUMENT;
3367 : :
3178 kgrittn@postgresql.o 3368 [ + + ]:CBC 7835 : if (tdata->tg_newtable)
3369 : : {
3370 : : EphemeralNamedRelation enr =
6 michael@paquier.xyz 3371 :GNC 201 : palloc_object(EphemeralNamedRelationData);
3372 : : int rc;
3373 : :
3178 kgrittn@postgresql.o 3374 :CBC 201 : enr->md.name = tdata->tg_trigger->tgnewtable;
3375 : 201 : enr->md.reliddesc = tdata->tg_relation->rd_id;
3376 : 201 : enr->md.tupdesc = NULL;
3377 : 201 : enr->md.enrtype = ENR_NAMED_TUPLESTORE;
3378 : 201 : enr->md.enrtuples = tuplestore_tuple_count(tdata->tg_newtable);
3379 : 201 : enr->reldata = tdata->tg_newtable;
3380 : 201 : rc = SPI_register_relation(enr);
3381 [ - + ]: 201 : if (rc != SPI_OK_REL_REGISTER)
3178 kgrittn@postgresql.o 3382 :UBC 0 : return rc;
3383 : : }
3384 : :
3178 kgrittn@postgresql.o 3385 [ + + ]:CBC 7835 : if (tdata->tg_oldtable)
3386 : : {
3387 : : EphemeralNamedRelation enr =
6 michael@paquier.xyz 3388 :GNC 156 : palloc_object(EphemeralNamedRelationData);
3389 : : int rc;
3390 : :
3178 kgrittn@postgresql.o 3391 :CBC 156 : enr->md.name = tdata->tg_trigger->tgoldtable;
3392 : 156 : enr->md.reliddesc = tdata->tg_relation->rd_id;
3393 : 156 : enr->md.tupdesc = NULL;
3394 : 156 : enr->md.enrtype = ENR_NAMED_TUPLESTORE;
3395 : 156 : enr->md.enrtuples = tuplestore_tuple_count(tdata->tg_oldtable);
3396 : 156 : enr->reldata = tdata->tg_oldtable;
3397 : 156 : rc = SPI_register_relation(enr);
3398 [ - + ]: 156 : if (rc != SPI_OK_REL_REGISTER)
3178 kgrittn@postgresql.o 3399 :UBC 0 : return rc;
3400 : : }
3401 : :
3178 kgrittn@postgresql.o 3402 :CBC 7835 : return SPI_OK_TD_REGISTER;
3403 : : }
|