Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * fmgr.c
4 : : * The Postgres function manager.
5 : : *
6 : : * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
7 : : * Portions Copyright (c) 1994, Regents of the University of California
8 : : *
9 : : *
10 : : * IDENTIFICATION
11 : : * src/backend/utils/fmgr/fmgr.c
12 : : *
13 : : *-------------------------------------------------------------------------
14 : : */
15 : :
16 : : #include "postgres.h"
17 : :
18 : : #include "access/detoast.h"
19 : : #include "access/htup_details.h"
20 : : #include "catalog/pg_language.h"
21 : : #include "catalog/pg_proc.h"
22 : : #include "catalog/pg_type.h"
23 : : #include "executor/functions.h"
24 : : #include "lib/stringinfo.h"
25 : : #include "miscadmin.h"
26 : : #include "nodes/makefuncs.h"
27 : : #include "nodes/miscnodes.h"
28 : : #include "nodes/nodeFuncs.h"
29 : : #include "pgstat.h"
30 : : #include "utils/acl.h"
31 : : #include "utils/builtins.h"
32 : : #include "utils/fmgrtab.h"
33 : : #include "utils/guc.h"
34 : : #include "utils/hsearch.h"
35 : : #include "utils/lsyscache.h"
36 : : #include "utils/syscache.h"
37 : :
38 : : /*
39 : : * Hooks for function calls
40 : : */
41 : : PGDLLIMPORT needs_fmgr_hook_type needs_fmgr_hook = NULL;
42 : : PGDLLIMPORT fmgr_hook_type fmgr_hook = NULL;
43 : :
44 : : /*
45 : : * Hashtable for fast lookup of external C functions
46 : : */
47 : : typedef struct
48 : : {
49 : : /* fn_oid is the hash key and so must be first! */
50 : : Oid fn_oid; /* OID of an external C function */
51 : : TransactionId fn_xmin; /* for checking up-to-dateness */
52 : : ItemPointerData fn_tid;
53 : : PGFunction user_fn; /* the function's address */
54 : : const Pg_finfo_record *inforec; /* address of its info record */
55 : : } CFuncHashTabEntry;
56 : :
57 : : static HTAB *CFuncHash = NULL;
58 : :
59 : :
60 : : static void fmgr_info_cxt_security(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt,
61 : : bool ignore_security);
62 : : static void fmgr_info_C_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple);
63 : : static void fmgr_info_other_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple);
64 : : static CFuncHashTabEntry *lookup_C_func(HeapTuple procedureTuple);
65 : : static void record_C_func(HeapTuple procedureTuple,
66 : : PGFunction user_fn, const Pg_finfo_record *inforec);
67 : :
68 : : /* extern so it's callable via JIT */
69 : : extern Datum fmgr_security_definer(PG_FUNCTION_ARGS);
70 : :
71 : :
72 : : /*
73 : : * Lookup routines for builtin-function table. We can search by either Oid
74 : : * or name, but search by Oid is much faster.
75 : : */
76 : :
77 : : static const FmgrBuiltin *
9498 tgl@sss.pgh.pa.us 78 :CBC 25539525 : fmgr_isbuiltin(Oid id)
79 : : {
80 : : uint16 index;
81 : :
82 : : /* fast lookup only possible if original oid still assigned */
2698 83 [ + + ]: 25539525 : if (id > fmgr_last_builtin_oid)
3160 andres@anarazel.de 84 : 87601 : return NULL;
85 : :
86 : : /*
87 : : * Lookup function data. If there's a miss in that range it's likely a
88 : : * nonexistent function, returning NULL here will trigger an ERROR later.
89 : : */
90 : 25451924 : index = fmgr_builtin_oid_index[id];
91 [ + + ]: 25451924 : if (index == InvalidOidBuiltinMapping)
92 : 5638 : return NULL;
93 : :
94 : 25446286 : return &fmgr_builtins[index];
95 : : }
96 : :
97 : : /*
98 : : * Lookup a builtin by name. Note there can be more than one entry in
99 : : * the array with the same name, but they should all point to the same
100 : : * routine.
101 : : */
102 : : static const FmgrBuiltin *
9200 bruce@momjian.us 103 : 2876 : fmgr_lookupByName(const char *name)
104 : : {
105 : : int i;
106 : :
9498 tgl@sss.pgh.pa.us 107 [ + + ]: 3073940 : for (i = 0; i < fmgr_nbuiltins; i++)
108 : : {
109 [ + + ]: 3073936 : if (strcmp(name, fmgr_builtins[i].funcName) == 0)
110 : 2872 : return fmgr_builtins + i;
111 : : }
8179 neilc@samurai.com 112 : 4 : return NULL;
113 : : }
114 : :
115 : : /*
116 : : * This routine fills a FmgrInfo struct, given the OID
117 : : * of the function to be called.
118 : : *
119 : : * The caller's CurrentMemoryContext is used as the fn_mcxt of the info
120 : : * struct; this means that any subsidiary data attached to the info struct
121 : : * (either by fmgr_info itself, or later on by a function call handler)
122 : : * will be allocated in that context. The caller must ensure that this
123 : : * context is at least as long-lived as the info struct itself. This is
124 : : * not a problem in typical cases where the info struct is on the stack or
125 : : * in freshly-palloc'd space. However, if one intends to store an info
126 : : * struct in a long-lived table, it's better to use fmgr_info_cxt.
127 : : */
128 : : void
9498 tgl@sss.pgh.pa.us 129 : 23923105 : fmgr_info(Oid functionId, FmgrInfo *finfo)
130 : : {
5113 131 : 23923105 : fmgr_info_cxt_security(functionId, finfo, CurrentMemoryContext, false);
9002 132 : 23923105 : }
133 : :
134 : : /*
135 : : * Fill a FmgrInfo struct, specifying a memory context in which its
136 : : * subsidiary data should go.
137 : : */
138 : : void
139 : 1594289 : fmgr_info_cxt(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt)
140 : : {
8778 peter_e@gmx.net 141 : 1594289 : fmgr_info_cxt_security(functionId, finfo, mcxt, false);
142 : 1594289 : }
143 : :
144 : : /*
145 : : * This one does the actual work. ignore_security is ordinarily false
146 : : * but is set to true when we need to avoid recursion.
147 : : */
148 : : static void
149 : 25539525 : fmgr_info_cxt_security(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt,
150 : : bool ignore_security)
151 : : {
152 : : const FmgrBuiltin *fbp;
153 : : HeapTuple procedureTuple;
154 : : Form_pg_proc procedureStruct;
155 : : Datum prosrcdatum;
156 : : char *prosrc;
157 : :
158 : : /*
159 : : * fn_oid *must* be filled in last. Some code assumes that if fn_oid is
160 : : * valid, the whole struct is valid. Some FmgrInfo struct's do survive
161 : : * elogs.
162 : : */
9142 163 : 25539525 : finfo->fn_oid = InvalidOid;
9498 tgl@sss.pgh.pa.us 164 : 25539525 : finfo->fn_extra = NULL;
9002 165 : 25539525 : finfo->fn_mcxt = mcxt;
8453 166 : 25539525 : finfo->fn_expr = NULL; /* caller may set this later */
167 : :
9498 168 [ + + ]: 25539525 : if ((fbp = fmgr_isbuiltin(functionId)) != NULL)
169 : : {
170 : : /*
171 : : * Fast path for builtin functions: don't bother consulting pg_proc
172 : : */
173 : 25446286 : finfo->fn_nargs = fbp->nargs;
174 : 25446286 : finfo->fn_strict = fbp->strict;
9322 175 : 25446286 : finfo->fn_retset = fbp->retset;
3265 176 : 25446286 : finfo->fn_stats = TRACK_FUNC_ALL; /* ie, never track */
9322 177 : 25446286 : finfo->fn_addr = fbp->func;
9142 peter_e@gmx.net 178 : 25446286 : finfo->fn_oid = functionId;
9498 tgl@sss.pgh.pa.us 179 : 25446286 : return;
180 : : }
181 : :
182 : : /* Otherwise we need the pg_proc entry */
5949 rhaas@postgresql.org 183 : 93239 : procedureTuple = SearchSysCache1(PROCOID, ObjectIdGetDatum(functionId));
9498 tgl@sss.pgh.pa.us 184 [ - + ]: 93239 : if (!HeapTupleIsValid(procedureTuple))
8345 tgl@sss.pgh.pa.us 185 [ # # ]:UBC 0 : elog(ERROR, "cache lookup failed for function %u", functionId);
9498 tgl@sss.pgh.pa.us 186 :CBC 93239 : procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
187 : :
188 : 93239 : finfo->fn_nargs = procedureStruct->pronargs;
189 : 93239 : finfo->fn_strict = procedureStruct->proisstrict;
9410 190 : 93239 : finfo->fn_retset = procedureStruct->proretset;
191 : :
192 : : /*
193 : : * If it has prosecdef set, non-null proconfig, or if a plugin wants to
194 : : * hook function entry/exit, use fmgr_security_definer call handler ---
195 : : * unless we are being called again by fmgr_security_definer or
196 : : * fmgr_info_other_lang.
197 : : *
198 : : * When using fmgr_security_definer, function stats tracking is always
199 : : * disabled at the outer level, and instead we set the flag properly in
200 : : * fmgr_security_definer's private flinfo and implement the tracking
201 : : * inside fmgr_security_definer. This loses the ability to charge the
202 : : * overhead of fmgr_security_definer to the function, but gains the
203 : : * ability to set the track_functions GUC as a local GUC parameter of an
204 : : * interesting function and have the right things happen.
205 : : */
6844 206 [ + + ]: 93239 : if (!ignore_security &&
207 [ + + ]: 71108 : (procedureStruct->prosecdef ||
2985 andrew@dunslane.net 208 [ + + - - ]: 70960 : !heap_attisnull(procedureTuple, Anum_pg_proc_proconfig, NULL) ||
5647 rhaas@postgresql.org 209 [ - + ]: 70878 : FmgrHookIsNeeded(functionId)))
210 : : {
8778 peter_e@gmx.net 211 : 230 : finfo->fn_addr = fmgr_security_definer;
3265 tgl@sss.pgh.pa.us 212 : 230 : finfo->fn_stats = TRACK_FUNC_ALL; /* ie, never track */
9142 peter_e@gmx.net 213 : 230 : finfo->fn_oid = functionId;
9326 tgl@sss.pgh.pa.us 214 : 230 : ReleaseSysCache(procedureTuple);
9498 215 : 230 : return;
216 : : }
217 : :
9322 218 [ + + + + ]: 93009 : switch (procedureStruct->prolang)
219 : : {
9498 220 : 1789 : case INTERNALlanguageId:
221 : :
222 : : /*
223 : : * For an ordinary builtin function, we should never get here
224 : : * because the fmgr_isbuiltin() search above will have succeeded.
225 : : * However, if the user has done a CREATE FUNCTION to create an
226 : : * alias for a builtin function, we can end up here. In that case
227 : : * we have to look up the function by name. The name of the
228 : : * internal function is stored in prosrc (it doesn't have to be
229 : : * the same as the name of the alias!)
230 : : */
1162 dgustafsson@postgres 231 : 1789 : prosrcdatum = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
232 : : Anum_pg_proc_prosrc);
6640 tgl@sss.pgh.pa.us 233 : 1789 : prosrc = TextDatumGetCString(prosrcdatum);
9498 234 : 1789 : fbp = fmgr_lookupByName(prosrc);
235 [ - + ]: 1789 : if (fbp == NULL)
8345 tgl@sss.pgh.pa.us 236 [ # # ]:UBC 0 : ereport(ERROR,
237 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
238 : : errmsg("internal function \"%s\" is not in internal lookup table",
239 : : prosrc)));
9498 tgl@sss.pgh.pa.us 240 :CBC 1789 : pfree(prosrc);
241 : : /* Should we check that nargs, strict, retset match the table? */
9322 242 : 1789 : finfo->fn_addr = fbp->func;
243 : : /* note this policy is also assumed in fast path above */
6589 244 : 1789 : finfo->fn_stats = TRACK_FUNC_ALL; /* ie, never track */
9498 245 : 1789 : break;
246 : :
247 : 57319 : case ClanguageId:
9142 peter_e@gmx.net 248 : 57319 : fmgr_info_C_lang(functionId, finfo, procedureTuple);
6589 tgl@sss.pgh.pa.us 249 : 57319 : finfo->fn_stats = TRACK_FUNC_PL; /* ie, track if ALL */
9498 250 : 57319 : break;
251 : :
252 : 11992 : case SQLlanguageId:
253 : 11992 : finfo->fn_addr = fmgr_sql;
6589 254 : 11992 : finfo->fn_stats = TRACK_FUNC_PL; /* ie, track if ALL */
9498 255 : 11992 : break;
256 : :
257 : 21909 : default:
9142 peter_e@gmx.net 258 : 21909 : fmgr_info_other_lang(functionId, finfo, procedureTuple);
6589 tgl@sss.pgh.pa.us 259 : 21909 : finfo->fn_stats = TRACK_FUNC_OFF; /* ie, track if not OFF */
9322 260 : 21909 : break;
261 : : }
262 : :
9142 peter_e@gmx.net 263 : 93009 : finfo->fn_oid = functionId;
9322 tgl@sss.pgh.pa.us 264 : 93009 : ReleaseSysCache(procedureTuple);
265 : : }
266 : :
267 : : /*
268 : : * Return module and C function name providing implementation of functionId.
269 : : *
270 : : * If *mod == NULL and *fn == NULL, no C symbol is known to implement
271 : : * function.
272 : : *
273 : : * If *mod == NULL and *fn != NULL, the function is implemented by a symbol in
274 : : * the main binary.
275 : : *
276 : : * If *mod != NULL and *fn != NULL the function is implemented in an extension
277 : : * shared object.
278 : : *
279 : : * The returned module and function names are pstrdup'ed into the current
280 : : * memory context.
281 : : */
282 : : void
2993 andres@anarazel.de 283 :UBC 0 : fmgr_symbol(Oid functionId, char **mod, char **fn)
284 : : {
285 : : HeapTuple procedureTuple;
286 : : Form_pg_proc procedureStruct;
287 : : Datum prosrcattr;
288 : : Datum probinattr;
289 : :
290 : 0 : procedureTuple = SearchSysCache1(PROCOID, ObjectIdGetDatum(functionId));
291 [ # # ]: 0 : if (!HeapTupleIsValid(procedureTuple))
292 [ # # ]: 0 : elog(ERROR, "cache lookup failed for function %u", functionId);
293 : 0 : procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
294 : :
295 [ # # ]: 0 : if (procedureStruct->prosecdef ||
2985 andrew@dunslane.net 296 [ # # # # ]: 0 : !heap_attisnull(procedureTuple, Anum_pg_proc_proconfig, NULL) ||
2993 andres@anarazel.de 297 [ # # ]: 0 : FmgrHookIsNeeded(functionId))
298 : : {
2956 tgl@sss.pgh.pa.us 299 : 0 : *mod = NULL; /* core binary */
2993 andres@anarazel.de 300 : 0 : *fn = pstrdup("fmgr_security_definer");
301 : 0 : ReleaseSysCache(procedureTuple);
302 : 0 : return;
303 : : }
304 : :
305 : : /* see fmgr_info_cxt_security for the individual cases */
306 [ # # # # ]: 0 : switch (procedureStruct->prolang)
307 : : {
308 : 0 : case INTERNALlanguageId:
1162 dgustafsson@postgres 309 : 0 : prosrcattr = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
310 : : Anum_pg_proc_prosrc);
311 : :
2956 tgl@sss.pgh.pa.us 312 : 0 : *mod = NULL; /* core binary */
2993 andres@anarazel.de 313 : 0 : *fn = TextDatumGetCString(prosrcattr);
314 : 0 : break;
315 : :
316 : 0 : case ClanguageId:
1162 dgustafsson@postgres 317 : 0 : prosrcattr = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
318 : : Anum_pg_proc_prosrc);
319 : :
320 : 0 : probinattr = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
321 : : Anum_pg_proc_probin);
322 : :
323 : : /*
324 : : * No need to check symbol presence / API version here, already
325 : : * checked in fmgr_info_cxt_security.
326 : : */
2993 andres@anarazel.de 327 : 0 : *mod = TextDatumGetCString(probinattr);
328 : 0 : *fn = TextDatumGetCString(prosrcattr);
329 : 0 : break;
330 : :
331 : 0 : case SQLlanguageId:
2956 tgl@sss.pgh.pa.us 332 : 0 : *mod = NULL; /* core binary */
2993 andres@anarazel.de 333 : 0 : *fn = pstrdup("fmgr_sql");
334 : 0 : break;
335 : :
336 : 0 : default:
337 : 0 : *mod = NULL;
2956 tgl@sss.pgh.pa.us 338 : 0 : *fn = NULL; /* unknown, pass pointer */
2993 andres@anarazel.de 339 : 0 : break;
340 : : }
341 : :
342 : 0 : ReleaseSysCache(procedureTuple);
343 : : }
344 : :
345 : :
346 : : /*
347 : : * Special fmgr_info processing for C-language functions. Note that
348 : : * finfo->fn_oid is not valid yet.
349 : : */
350 : : static void
9142 peter_e@gmx.net 351 :CBC 57319 : fmgr_info_C_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple)
352 : : {
353 : : CFuncHashTabEntry *hashentry;
354 : : PGFunction user_fn;
355 : : const Pg_finfo_record *inforec;
356 : :
357 : : /*
358 : : * See if we have the function address cached already
359 : : */
8167 tgl@sss.pgh.pa.us 360 : 57319 : hashentry = lookup_C_func(procedureTuple);
361 [ + + ]: 57319 : if (hashentry)
362 : : {
363 : 52848 : user_fn = hashentry->user_fn;
364 : 52848 : inforec = hashentry->inforec;
365 : : }
366 : : else
367 : : {
368 : : Datum prosrcattr,
369 : : probinattr;
370 : : char *prosrcstring,
371 : : *probinstring;
372 : : void *libraryhandle;
373 : :
374 : : /*
375 : : * Get prosrc and probin strings (link symbol and library filename).
376 : : * While in general these columns might be null, that's not allowed
377 : : * for C-language functions.
378 : : */
1162 dgustafsson@postgres 379 : 4471 : prosrcattr = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
380 : : Anum_pg_proc_prosrc);
6640 tgl@sss.pgh.pa.us 381 : 4471 : prosrcstring = TextDatumGetCString(prosrcattr);
382 : :
1162 dgustafsson@postgres 383 : 4471 : probinattr = SysCacheGetAttrNotNull(PROCOID, procedureTuple,
384 : : Anum_pg_proc_probin);
6640 tgl@sss.pgh.pa.us 385 : 4471 : probinstring = TextDatumGetCString(probinattr);
386 : :
387 : : /* Look up the function itself */
8167 388 : 4471 : user_fn = load_external_function(probinstring, prosrcstring, true,
389 : : &libraryhandle);
390 : :
391 : : /* Get the function information record (real or default) */
392 : 4471 : inforec = fetch_finfo_record(libraryhandle, prosrcstring);
393 : :
394 : : /* Cache the addresses for later calls */
395 : 4471 : record_C_func(procedureTuple, user_fn, inforec);
396 : :
397 : 4471 : pfree(prosrcstring);
398 : 4471 : pfree(probinstring);
399 : : }
400 : :
9322 401 [ + - ]: 57319 : switch (inforec->api_version)
402 : : {
403 : 57319 : case 1:
404 : : /* New style: call directly */
405 : 57319 : finfo->fn_addr = user_fn;
406 : 57319 : break;
9322 tgl@sss.pgh.pa.us 407 :UBC 0 : default:
408 : : /* Shouldn't get here if fetch_finfo_record did its job */
8345 409 [ # # ]: 0 : elog(ERROR, "unrecognized function API version: %d",
410 : : inforec->api_version);
411 : : break;
412 : : }
9919 tgl@sss.pgh.pa.us 413 :CBC 57319 : }
414 : :
415 : : /*
416 : : * Special fmgr_info processing for other-language functions. Note
417 : : * that finfo->fn_oid is not valid yet.
418 : : */
419 : : static void
9142 peter_e@gmx.net 420 : 21909 : fmgr_info_other_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple)
421 : : {
9322 tgl@sss.pgh.pa.us 422 : 21909 : Form_pg_proc procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
423 : 21909 : Oid language = procedureStruct->prolang;
424 : : HeapTuple languageTuple;
425 : : Form_pg_language languageStruct;
426 : : FmgrInfo plfinfo;
427 : :
5949 rhaas@postgresql.org 428 : 21909 : languageTuple = SearchSysCache1(LANGOID, ObjectIdGetDatum(language));
9322 tgl@sss.pgh.pa.us 429 [ - + ]: 21909 : if (!HeapTupleIsValid(languageTuple))
8345 tgl@sss.pgh.pa.us 430 [ # # ]:UBC 0 : elog(ERROR, "cache lookup failed for language %u", language);
9322 tgl@sss.pgh.pa.us 431 :CBC 21909 : languageStruct = (Form_pg_language) GETSTRUCT(languageTuple);
432 : :
433 : : /*
434 : : * Look up the language's call handler function, ignoring any attributes
435 : : * that would normally cause insertion of fmgr_security_definer. We need
436 : : * to get back a bare pointer to the actual C-language function.
437 : : */
5113 438 : 21909 : fmgr_info_cxt_security(languageStruct->lanplcallfoid, &plfinfo,
439 : : CurrentMemoryContext, true);
8691 peter_e@gmx.net 440 : 21909 : finfo->fn_addr = plfinfo.fn_addr;
441 : :
9322 tgl@sss.pgh.pa.us 442 : 21909 : ReleaseSysCache(languageTuple);
443 : 21909 : }
444 : :
445 : : /*
446 : : * Fetch and validate the information record for the given external function.
447 : : * The function is specified by a handle for the containing library
448 : : * (obtained from load_external_function) as well as the function name.
449 : : *
450 : : * If no info function exists for the given name an error is raised.
451 : : *
452 : : * This function is broken out of fmgr_info_C_lang so that fmgr_c_validator
453 : : * can validate the information record for a function not yet entered into
454 : : * pg_proc.
455 : : */
456 : : const Pg_finfo_record *
3333 457 : 8413 : fetch_finfo_record(void *filehandle, const char *funcname)
458 : : {
459 : : char *infofuncname;
460 : : PGFInfoFunction infofunc;
461 : : const Pg_finfo_record *inforec;
462 : :
4612 peter_e@gmx.net 463 : 8413 : infofuncname = psprintf("pg_finfo_%s", funcname);
464 : :
465 : : /* Try to look up the info function */
9002 tgl@sss.pgh.pa.us 466 : 8413 : infofunc = (PGFInfoFunction) lookup_external_function(filehandle,
467 : : infofuncname);
8179 neilc@samurai.com 468 [ - + ]: 8413 : if (infofunc == NULL)
469 : : {
3349 andres@anarazel.de 470 [ # # ]:UBC 0 : ereport(ERROR,
471 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
472 : : errmsg("could not find function information for function \"%s\"",
473 : : funcname),
474 : : errhint("SQL-callable functions need an accompanying PG_FUNCTION_INFO_V1(funcname).")));
475 : : return NULL; /* silence compiler */
476 : : }
477 : :
478 : : /* Found, so call it */
9200 bruce@momjian.us 479 :CBC 8413 : inforec = (*infofunc) ();
480 : :
481 : : /* Validate result as best we can */
9322 tgl@sss.pgh.pa.us 482 [ - + ]: 8413 : if (inforec == NULL)
8345 tgl@sss.pgh.pa.us 483 [ # # ]:UBC 0 : elog(ERROR, "null result from info function \"%s\"", infofuncname);
9322 tgl@sss.pgh.pa.us 484 [ + - ]:CBC 8413 : switch (inforec->api_version)
485 : : {
486 : 8413 : case 1:
487 : : /* OK, no additional fields to validate */
488 : 8413 : break;
9322 tgl@sss.pgh.pa.us 489 :UBC 0 : default:
8345 490 [ # # ]: 0 : ereport(ERROR,
491 : : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
492 : : errmsg("unrecognized API version %d reported by info function \"%s\"",
493 : : inforec->api_version, infofuncname)));
494 : : break;
495 : : }
496 : :
9322 tgl@sss.pgh.pa.us 497 :CBC 8413 : pfree(infofuncname);
498 : 8413 : return inforec;
499 : : }
500 : :
501 : :
502 : : /*-------------------------------------------------------------------------
503 : : * Routines for caching lookup information for external C functions.
504 : : *
505 : : * The routines in dfmgr.c are relatively slow, so we try to avoid running
506 : : * them more than once per external function per session. We use a hash table
507 : : * with the function OID as the lookup key.
508 : : *-------------------------------------------------------------------------
509 : : */
510 : :
511 : : /*
512 : : * lookup_C_func: try to find a C function in the hash table
513 : : *
514 : : * If an entry exists and is up to date, return it; else return NULL
515 : : */
516 : : static CFuncHashTabEntry *
8167 517 : 57319 : lookup_C_func(HeapTuple procedureTuple)
518 : : {
2748 andres@anarazel.de 519 : 57319 : Oid fn_oid = ((Form_pg_proc) GETSTRUCT(procedureTuple))->oid;
520 : : CFuncHashTabEntry *entry;
521 : :
8167 tgl@sss.pgh.pa.us 522 [ + + ]: 57319 : if (CFuncHash == NULL)
523 : 2129 : return NULL; /* no table yet */
524 : : entry = (CFuncHashTabEntry *)
525 : 55190 : hash_search(CFuncHash,
526 : : &fn_oid,
527 : : HASH_FIND,
528 : : NULL);
529 [ + + ]: 55190 : if (entry == NULL)
530 : 2340 : return NULL; /* no such entry */
4542 rhaas@postgresql.org 531 [ + + + - ]: 105698 : if (entry->fn_xmin == HeapTupleHeaderGetRawXmin(procedureTuple->t_data) &&
7050 tgl@sss.pgh.pa.us 532 : 52848 : ItemPointerEquals(&entry->fn_tid, &procedureTuple->t_self))
8167 533 : 52848 : return entry; /* OK */
534 : 2 : return NULL; /* entry is out of date */
535 : : }
536 : :
537 : : /*
538 : : * record_C_func: enter (or update) info about a C function in the hash table
539 : : */
540 : : static void
541 : 4471 : record_C_func(HeapTuple procedureTuple,
542 : : PGFunction user_fn, const Pg_finfo_record *inforec)
543 : : {
2565 544 : 4471 : Oid fn_oid = ((Form_pg_proc) GETSTRUCT(procedureTuple))->oid;
545 : : CFuncHashTabEntry *entry;
546 : : bool found;
547 : :
548 : : /* Create the hash table if it doesn't exist yet */
8167 549 [ + + ]: 4471 : if (CFuncHash == NULL)
550 : : {
551 : : HASHCTL hash_ctl;
552 : :
553 : 2129 : hash_ctl.keysize = sizeof(Oid);
554 : 2129 : hash_ctl.entrysize = sizeof(CFuncHashTabEntry);
555 : 2129 : CFuncHash = hash_create("CFuncHash",
556 : : 100,
557 : : &hash_ctl,
558 : : HASH_ELEM | HASH_BLOBS);
559 : : }
560 : :
561 : : entry = (CFuncHashTabEntry *)
562 : 4471 : hash_search(CFuncHash,
563 : : &fn_oid,
564 : : HASH_ENTER,
565 : : &found);
566 : : /* OID is already filled in */
4542 rhaas@postgresql.org 567 : 4471 : entry->fn_xmin = HeapTupleHeaderGetRawXmin(procedureTuple->t_data);
7050 tgl@sss.pgh.pa.us 568 : 4471 : entry->fn_tid = procedureTuple->t_self;
8167 569 : 4471 : entry->user_fn = user_fn;
570 : 4471 : entry->inforec = inforec;
571 : 4471 : }
572 : :
573 : :
574 : : /*
575 : : * Copy an FmgrInfo struct
576 : : *
577 : : * This is inherently somewhat bogus since we can't reliably duplicate
578 : : * language-dependent subsidiary info. We cheat by zeroing fn_extra,
579 : : * instead, meaning that subsidiary info will have to be recomputed.
580 : : */
581 : : void
9002 582 : 30447126 : fmgr_info_copy(FmgrInfo *dstinfo, FmgrInfo *srcinfo,
583 : : MemoryContext destcxt)
584 : : {
585 : 30447126 : memcpy(dstinfo, srcinfo, sizeof(FmgrInfo));
586 : 30447126 : dstinfo->fn_mcxt = destcxt;
3349 andres@anarazel.de 587 : 30447126 : dstinfo->fn_extra = NULL;
9002 tgl@sss.pgh.pa.us 588 : 30447126 : }
589 : :
590 : :
591 : : /*
592 : : * Specialized lookup routine for fmgr_internal_validator: given the alleged
593 : : * name of an internal function, return the OID of the function.
594 : : * If the name is not recognized, return InvalidOid.
595 : : */
596 : : Oid
9322 597 : 1087 : fmgr_internal_function(const char *proname)
598 : : {
9498 599 : 1087 : const FmgrBuiltin *fbp = fmgr_lookupByName(proname);
600 : :
601 [ + + ]: 1087 : if (fbp == NULL)
602 : 4 : return InvalidOid;
9322 603 : 1083 : return fbp->foid;
604 : : }
605 : :
606 : :
607 : : /*
608 : : * Support for security-definer and proconfig-using functions. We support
609 : : * both of these features using the same call handler, because they are
610 : : * often used together and it would be inefficient (as well as notationally
611 : : * messy) to have two levels of call handler involved.
612 : : */
613 : : struct fmgr_security_definer_cache
614 : : {
615 : : FmgrInfo flinfo; /* lookup info for target function */
616 : : Oid userid; /* userid to set, or InvalidOid */
617 : : List *configNames; /* GUC names to set, or NIL */
618 : : List *configHandles; /* GUC handles to set, or NIL */
619 : : List *configValues; /* GUC values to set, or NIL */
620 : : Datum arg; /* passthrough argument for plugin modules */
621 : : };
622 : :
623 : : /*
624 : : * Function handler for security-definer/proconfig/plugin-hooked functions.
625 : : * We extract the OID of the actual function and do a fmgr lookup again.
626 : : * Then we fetch the pg_proc row and copy the owner ID and proconfig fields.
627 : : * (All this info is cached for the duration of the current query.)
628 : : * To execute a call, we temporarily replace the flinfo with the cached
629 : : * and looked-up one, while keeping the outer fcinfo (which contains all
630 : : * the actual arguments, etc.) intact. This is not re-entrant, but then
631 : : * the fcinfo itself can't be used reentrantly anyway.
632 : : */
633 : : extern Datum
8778 peter_e@gmx.net 634 : 252 : fmgr_security_definer(PG_FUNCTION_ARGS)
635 : : {
636 : : Datum result;
637 : : struct fmgr_security_definer_cache *volatile fcache;
638 : : FmgrInfo *save_flinfo;
639 : : Oid save_userid;
640 : : int save_sec_context;
641 : : ListCell *lc1,
642 : : *lc2,
643 : : *lc3;
644 : : int save_nestlevel;
645 : : PgStat_FunctionCallUsage fcusage;
646 : :
647 [ + + ]: 252 : if (!fcinfo->flinfo->fn_extra)
648 : : {
649 : : HeapTuple tuple;
650 : : Form_pg_proc procedureStruct;
651 : : Datum datum;
652 : : bool isnull;
653 : : MemoryContext oldcxt;
654 : :
7732 tgl@sss.pgh.pa.us 655 : 222 : fcache = MemoryContextAllocZero(fcinfo->flinfo->fn_mcxt,
656 : : sizeof(*fcache));
657 : :
8778 peter_e@gmx.net 658 : 222 : fmgr_info_cxt_security(fcinfo->flinfo->fn_oid, &fcache->flinfo,
659 : 222 : fcinfo->flinfo->fn_mcxt, true);
6878 tgl@sss.pgh.pa.us 660 : 222 : fcache->flinfo.fn_expr = fcinfo->flinfo->fn_expr;
661 : :
5949 rhaas@postgresql.org 662 : 222 : tuple = SearchSysCache1(PROCOID,
663 : 222 : ObjectIdGetDatum(fcinfo->flinfo->fn_oid));
8778 peter_e@gmx.net 664 [ - + ]: 222 : if (!HeapTupleIsValid(tuple))
8345 tgl@sss.pgh.pa.us 665 [ # # ]:UBC 0 : elog(ERROR, "cache lookup failed for function %u",
666 : : fcinfo->flinfo->fn_oid);
6844 tgl@sss.pgh.pa.us 667 :CBC 222 : procedureStruct = (Form_pg_proc) GETSTRUCT(tuple);
668 : :
669 [ + + ]: 222 : if (procedureStruct->prosecdef)
670 : 148 : fcache->userid = procedureStruct->proowner;
671 : :
672 : 222 : datum = SysCacheGetAttr(PROCOID, tuple, Anum_pg_proc_proconfig,
673 : : &isnull);
674 [ + + ]: 222 : if (!isnull)
675 : : {
676 : : ArrayType *array;
677 : : ListCell *lc;
678 : :
679 : 78 : oldcxt = MemoryContextSwitchTo(fcinfo->flinfo->fn_mcxt);
1024 jdavis@postgresql.or 680 : 78 : array = DatumGetArrayTypeP(datum);
681 : 78 : TransformGUCArray(array, &fcache->configNames,
682 : 78 : &fcache->configValues);
683 : :
684 : : /* transform names to config handles to avoid lookup cost */
904 685 : 78 : fcache->configHandles = NIL;
686 [ + - + + : 156 : foreach(lc, fcache->configNames)
+ + ]
687 : : {
688 : 78 : char *name = (char *) lfirst(lc);
689 : :
690 : 78 : fcache->configHandles = lappend(fcache->configHandles,
691 : 78 : get_config_handle(name));
692 : : }
693 : :
6844 tgl@sss.pgh.pa.us 694 : 78 : MemoryContextSwitchTo(oldcxt);
695 : : }
696 : :
8778 peter_e@gmx.net 697 : 222 : ReleaseSysCache(tuple);
698 : :
699 : 222 : fcinfo->flinfo->fn_extra = fcache;
700 : : }
701 : : else
702 : 30 : fcache = fcinfo->flinfo->fn_extra;
703 : :
704 : : /* GetUserIdAndSecContext is cheap enough that no harm in a wasted call */
6016 tgl@sss.pgh.pa.us 705 : 252 : GetUserIdAndSecContext(&save_userid, &save_sec_context);
1023 michael@paquier.xyz 706 [ + + ]: 252 : if (fcache->configNames != NIL) /* Need a new GUC nesting level */
6844 tgl@sss.pgh.pa.us 707 : 108 : save_nestlevel = NewGUCNestLevel();
708 : : else
709 : 144 : save_nestlevel = 0; /* keep compiler quiet */
710 : :
6722 711 [ + + ]: 252 : if (OidIsValid(fcache->userid))
6016 712 : 178 : SetUserIdAndSecContext(fcache->userid,
713 : : save_sec_context | SECURITY_LOCAL_USERID_CHANGE);
714 : :
904 jdavis@postgresql.or 715 : 356 : forthree(lc1, fcache->configNames,
[ + + + +
+ + + + +
+ + + + +
+ - + - +
+ ]
716 : : lc2, fcache->configHandles,
717 : : lc3, fcache->configValues)
718 : : {
1023 michael@paquier.xyz 719 [ + + ]: 108 : GucContext context = superuser() ? PGC_SUSET : PGC_USERSET;
720 : 108 : GucSource source = PGC_S_SESSION;
721 : 108 : GucAction action = GUC_ACTION_SAVE;
722 : 108 : char *name = lfirst(lc1);
904 jdavis@postgresql.or 723 : 108 : config_handle *handle = lfirst(lc2);
724 : 108 : char *value = lfirst(lc3);
725 : :
726 : 108 : (void) set_config_with_handle(name, handle, value,
727 : : context, source, GetUserId(),
728 : : action, true, 0, false);
729 : : }
730 : :
731 : : /* function manager hook */
5647 rhaas@postgresql.org 732 [ - + ]: 248 : if (fmgr_hook)
5529 bruce@momjian.us 733 :UBC 0 : (*fmgr_hook) (FHET_START, &fcache->flinfo, &fcache->arg);
734 : :
735 : : /*
736 : : * We don't need to restore GUC or userid settings on error, because the
737 : : * ensuing xact or subxact abort will do that. The PG_TRY block is only
738 : : * needed to clean up the flinfo link.
739 : : */
6722 tgl@sss.pgh.pa.us 740 :CBC 248 : save_flinfo = fcinfo->flinfo;
741 : :
742 [ + + ]: 248 : PG_TRY();
743 : : {
744 : 248 : fcinfo->flinfo = &fcache->flinfo;
745 : :
746 : : /* See notes in fmgr_info_cxt_security */
6589 747 : 248 : pgstat_init_function_usage(fcinfo, &fcusage);
748 : :
7911 749 : 248 : result = FunctionCallInvoke(fcinfo);
750 : :
751 : : /*
752 : : * We could be calling either a regular or a set-returning function,
753 : : * so we have to test to see what finalize flag to use.
754 : : */
6589 755 : 234 : pgstat_end_function_usage(&fcusage,
756 : 242 : (fcinfo->resultinfo == NULL ||
757 [ + + + - ]: 242 : !IsA(fcinfo->resultinfo, ReturnSetInfo) ||
758 [ + - ]: 8 : ((ReturnSetInfo *) fcinfo->resultinfo)->isDone != ExprMultipleResult));
759 : : }
7911 760 : 14 : PG_CATCH();
761 : : {
762 : 14 : fcinfo->flinfo = save_flinfo;
5647 rhaas@postgresql.org 763 [ - + ]: 14 : if (fmgr_hook)
5529 bruce@momjian.us 764 :UBC 0 : (*fmgr_hook) (FHET_ABORT, &fcache->flinfo, &fcache->arg);
7911 tgl@sss.pgh.pa.us 765 :CBC 14 : PG_RE_THROW();
766 : : }
767 [ - + ]: 234 : PG_END_TRY();
768 : :
8778 peter_e@gmx.net 769 : 234 : fcinfo->flinfo = save_flinfo;
770 : :
1024 jdavis@postgresql.or 771 [ + + ]: 234 : if (fcache->configNames != NIL)
6844 tgl@sss.pgh.pa.us 772 : 91 : AtEOXact_GUC(true, save_nestlevel);
773 [ + + ]: 234 : if (OidIsValid(fcache->userid))
6016 774 : 177 : SetUserIdAndSecContext(save_userid, save_sec_context);
5647 rhaas@postgresql.org 775 [ - + ]: 234 : if (fmgr_hook)
5529 bruce@momjian.us 776 :UBC 0 : (*fmgr_hook) (FHET_END, &fcache->flinfo, &fcache->arg);
777 : :
8778 peter_e@gmx.net 778 :CBC 234 : return result;
779 : : }
780 : :
781 : :
782 : : /*-------------------------------------------------------------------------
783 : : * Support routines for callers of fmgr-compatible functions
784 : : *-------------------------------------------------------------------------
785 : : */
786 : :
787 : : /*
788 : : * These are for invocation of a specifically named function with a
789 : : * directly-computed parameter list. Note that neither arguments nor result
790 : : * are allowed to be NULL. Also, the function cannot be one that needs to
791 : : * look at FmgrInfo, since there won't be any.
792 : : */
793 : : Datum
5527 tgl@sss.pgh.pa.us 794 : 1533451 : DirectFunctionCall1Coll(PGFunction func, Oid collation, Datum arg1)
795 : : {
2681 andres@anarazel.de 796 : 1533451 : LOCAL_FCINFO(fcinfo, 1);
797 : : Datum result;
798 : :
799 : 1533451 : InitFunctionCallInfoData(*fcinfo, NULL, 1, collation, NULL, NULL);
800 : :
801 : 1533451 : fcinfo->args[0].value = arg1;
802 : 1533451 : fcinfo->args[0].isnull = false;
803 : :
804 : 1533451 : result = (*func) (fcinfo);
805 : :
806 : : /* Check for null result, since caller is clearly not expecting one */
807 [ - + ]: 1533414 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 808 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
809 : :
9498 tgl@sss.pgh.pa.us 810 :CBC 1533414 : return result;
811 : : }
812 : :
813 : : Datum
5527 814 : 26340955 : DirectFunctionCall2Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2)
815 : : {
2681 andres@anarazel.de 816 : 26340955 : LOCAL_FCINFO(fcinfo, 2);
817 : : Datum result;
818 : :
819 : 26340955 : InitFunctionCallInfoData(*fcinfo, NULL, 2, collation, NULL, NULL);
820 : :
821 : 26340955 : fcinfo->args[0].value = arg1;
822 : 26340955 : fcinfo->args[0].isnull = false;
823 : 26340955 : fcinfo->args[1].value = arg2;
824 : 26340955 : fcinfo->args[1].isnull = false;
825 : :
826 : 26340955 : result = (*func) (fcinfo);
827 : :
828 : : /* Check for null result, since caller is clearly not expecting one */
829 [ - + ]: 26340895 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 830 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
831 : :
9498 tgl@sss.pgh.pa.us 832 :CBC 26340895 : return result;
833 : : }
834 : :
835 : : Datum
5527 836 : 84678 : DirectFunctionCall3Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
837 : : Datum arg3)
838 : : {
2681 andres@anarazel.de 839 : 84678 : LOCAL_FCINFO(fcinfo, 3);
840 : : Datum result;
841 : :
842 : 84678 : InitFunctionCallInfoData(*fcinfo, NULL, 3, collation, NULL, NULL);
843 : :
844 : 84678 : fcinfo->args[0].value = arg1;
845 : 84678 : fcinfo->args[0].isnull = false;
846 : 84678 : fcinfo->args[1].value = arg2;
847 : 84678 : fcinfo->args[1].isnull = false;
848 : 84678 : fcinfo->args[2].value = arg3;
849 : 84678 : fcinfo->args[2].isnull = false;
850 : :
851 : 84678 : result = (*func) (fcinfo);
852 : :
853 : : /* Check for null result, since caller is clearly not expecting one */
854 [ - + ]: 84645 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 855 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
856 : :
9498 tgl@sss.pgh.pa.us 857 :CBC 84645 : return result;
858 : : }
859 : :
860 : : Datum
5527 861 : 680 : DirectFunctionCall4Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
862 : : Datum arg3, Datum arg4)
863 : : {
2681 andres@anarazel.de 864 : 680 : LOCAL_FCINFO(fcinfo, 4);
865 : : Datum result;
866 : :
867 : 680 : InitFunctionCallInfoData(*fcinfo, NULL, 4, collation, NULL, NULL);
868 : :
869 : 680 : fcinfo->args[0].value = arg1;
870 : 680 : fcinfo->args[0].isnull = false;
871 : 680 : fcinfo->args[1].value = arg2;
872 : 680 : fcinfo->args[1].isnull = false;
873 : 680 : fcinfo->args[2].value = arg3;
874 : 680 : fcinfo->args[2].isnull = false;
875 : 680 : fcinfo->args[3].value = arg4;
876 : 680 : fcinfo->args[3].isnull = false;
877 : :
878 : 680 : result = (*func) (fcinfo);
879 : :
880 : : /* Check for null result, since caller is clearly not expecting one */
881 [ - + ]: 680 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 882 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
883 : :
9498 tgl@sss.pgh.pa.us 884 :CBC 680 : return result;
885 : : }
886 : :
887 : : Datum
5527 888 : 3514 : DirectFunctionCall5Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
889 : : Datum arg3, Datum arg4, Datum arg5)
890 : : {
2681 andres@anarazel.de 891 : 3514 : LOCAL_FCINFO(fcinfo, 5);
892 : : Datum result;
893 : :
894 : 3514 : InitFunctionCallInfoData(*fcinfo, NULL, 5, collation, NULL, NULL);
895 : :
896 : 3514 : fcinfo->args[0].value = arg1;
897 : 3514 : fcinfo->args[0].isnull = false;
898 : 3514 : fcinfo->args[1].value = arg2;
899 : 3514 : fcinfo->args[1].isnull = false;
900 : 3514 : fcinfo->args[2].value = arg3;
901 : 3514 : fcinfo->args[2].isnull = false;
902 : 3514 : fcinfo->args[3].value = arg4;
903 : 3514 : fcinfo->args[3].isnull = false;
904 : 3514 : fcinfo->args[4].value = arg5;
905 : 3514 : fcinfo->args[4].isnull = false;
906 : :
907 : 3514 : result = (*func) (fcinfo);
908 : :
909 : : /* Check for null result, since caller is clearly not expecting one */
910 [ - + ]: 3510 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 911 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
912 : :
9498 tgl@sss.pgh.pa.us 913 :CBC 3510 : return result;
914 : : }
915 : :
916 : : Datum
5527 tgl@sss.pgh.pa.us 917 :UBC 0 : DirectFunctionCall6Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
918 : : Datum arg3, Datum arg4, Datum arg5,
919 : : Datum arg6)
920 : : {
2681 andres@anarazel.de 921 : 0 : LOCAL_FCINFO(fcinfo, 6);
922 : : Datum result;
923 : :
924 : 0 : InitFunctionCallInfoData(*fcinfo, NULL, 6, collation, NULL, NULL);
925 : :
926 : 0 : fcinfo->args[0].value = arg1;
927 : 0 : fcinfo->args[0].isnull = false;
928 : 0 : fcinfo->args[1].value = arg2;
929 : 0 : fcinfo->args[1].isnull = false;
930 : 0 : fcinfo->args[2].value = arg3;
931 : 0 : fcinfo->args[2].isnull = false;
932 : 0 : fcinfo->args[3].value = arg4;
933 : 0 : fcinfo->args[3].isnull = false;
934 : 0 : fcinfo->args[4].value = arg5;
935 : 0 : fcinfo->args[4].isnull = false;
936 : 0 : fcinfo->args[5].value = arg6;
937 : 0 : fcinfo->args[5].isnull = false;
938 : :
939 : 0 : result = (*func) (fcinfo);
940 : :
941 : : /* Check for null result, since caller is clearly not expecting one */
942 [ # # ]: 0 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 943 [ # # ]: 0 : elog(ERROR, "function %p returned NULL", (void *) func);
944 : :
9498 945 : 0 : return result;
946 : : }
947 : :
948 : : Datum
5527 949 : 0 : DirectFunctionCall7Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
950 : : Datum arg3, Datum arg4, Datum arg5,
951 : : Datum arg6, Datum arg7)
952 : : {
2681 andres@anarazel.de 953 : 0 : LOCAL_FCINFO(fcinfo, 7);
954 : : Datum result;
955 : :
956 : 0 : InitFunctionCallInfoData(*fcinfo, NULL, 7, collation, NULL, NULL);
957 : :
958 : 0 : fcinfo->args[0].value = arg1;
959 : 0 : fcinfo->args[0].isnull = false;
960 : 0 : fcinfo->args[1].value = arg2;
961 : 0 : fcinfo->args[1].isnull = false;
962 : 0 : fcinfo->args[2].value = arg3;
963 : 0 : fcinfo->args[2].isnull = false;
964 : 0 : fcinfo->args[3].value = arg4;
965 : 0 : fcinfo->args[3].isnull = false;
966 : 0 : fcinfo->args[4].value = arg5;
967 : 0 : fcinfo->args[4].isnull = false;
968 : 0 : fcinfo->args[5].value = arg6;
969 : 0 : fcinfo->args[5].isnull = false;
970 : 0 : fcinfo->args[6].value = arg7;
971 : 0 : fcinfo->args[6].isnull = false;
972 : :
973 : 0 : result = (*func) (fcinfo);
974 : :
975 : : /* Check for null result, since caller is clearly not expecting one */
976 [ # # ]: 0 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 977 [ # # ]: 0 : elog(ERROR, "function %p returned NULL", (void *) func);
978 : :
9498 979 : 0 : return result;
980 : : }
981 : :
982 : : Datum
5527 983 : 0 : DirectFunctionCall8Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
984 : : Datum arg3, Datum arg4, Datum arg5,
985 : : Datum arg6, Datum arg7, Datum arg8)
986 : : {
2681 andres@anarazel.de 987 : 0 : LOCAL_FCINFO(fcinfo, 8);
988 : : Datum result;
989 : :
990 : 0 : InitFunctionCallInfoData(*fcinfo, NULL, 8, collation, NULL, NULL);
991 : :
992 : 0 : fcinfo->args[0].value = arg1;
993 : 0 : fcinfo->args[0].isnull = false;
994 : 0 : fcinfo->args[1].value = arg2;
995 : 0 : fcinfo->args[1].isnull = false;
996 : 0 : fcinfo->args[2].value = arg3;
997 : 0 : fcinfo->args[2].isnull = false;
998 : 0 : fcinfo->args[3].value = arg4;
999 : 0 : fcinfo->args[3].isnull = false;
1000 : 0 : fcinfo->args[4].value = arg5;
1001 : 0 : fcinfo->args[4].isnull = false;
1002 : 0 : fcinfo->args[5].value = arg6;
1003 : 0 : fcinfo->args[5].isnull = false;
1004 : 0 : fcinfo->args[6].value = arg7;
1005 : 0 : fcinfo->args[6].isnull = false;
1006 : 0 : fcinfo->args[7].value = arg8;
1007 : 0 : fcinfo->args[7].isnull = false;
1008 : :
1009 : 0 : result = (*func) (fcinfo);
1010 : :
1011 : : /* Check for null result, since caller is clearly not expecting one */
1012 [ # # ]: 0 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 1013 [ # # ]: 0 : elog(ERROR, "function %p returned NULL", (void *) func);
1014 : :
9498 1015 : 0 : return result;
1016 : : }
1017 : :
1018 : : Datum
5527 1019 : 0 : DirectFunctionCall9Coll(PGFunction func, Oid collation, Datum arg1, Datum arg2,
1020 : : Datum arg3, Datum arg4, Datum arg5,
1021 : : Datum arg6, Datum arg7, Datum arg8,
1022 : : Datum arg9)
1023 : : {
2681 andres@anarazel.de 1024 : 0 : LOCAL_FCINFO(fcinfo, 9);
1025 : : Datum result;
1026 : :
1027 : 0 : InitFunctionCallInfoData(*fcinfo, NULL, 9, collation, NULL, NULL);
1028 : :
1029 : 0 : fcinfo->args[0].value = arg1;
1030 : 0 : fcinfo->args[0].isnull = false;
1031 : 0 : fcinfo->args[1].value = arg2;
1032 : 0 : fcinfo->args[1].isnull = false;
1033 : 0 : fcinfo->args[2].value = arg3;
1034 : 0 : fcinfo->args[2].isnull = false;
1035 : 0 : fcinfo->args[3].value = arg4;
1036 : 0 : fcinfo->args[3].isnull = false;
1037 : 0 : fcinfo->args[4].value = arg5;
1038 : 0 : fcinfo->args[4].isnull = false;
1039 : 0 : fcinfo->args[5].value = arg6;
1040 : 0 : fcinfo->args[5].isnull = false;
1041 : 0 : fcinfo->args[6].value = arg7;
1042 : 0 : fcinfo->args[6].isnull = false;
1043 : 0 : fcinfo->args[7].value = arg8;
1044 : 0 : fcinfo->args[7].isnull = false;
1045 : 0 : fcinfo->args[8].value = arg9;
1046 : 0 : fcinfo->args[8].isnull = false;
1047 : :
1048 : 0 : result = (*func) (fcinfo);
1049 : :
1050 : : /* Check for null result, since caller is clearly not expecting one */
1051 [ # # ]: 0 : if (fcinfo->isnull)
8345 tgl@sss.pgh.pa.us 1052 [ # # ]: 0 : elog(ERROR, "function %p returned NULL", (void *) func);
1053 : :
9498 1054 : 0 : return result;
1055 : : }
1056 : :
1057 : : /*
1058 : : * These functions work like the DirectFunctionCall functions except that
1059 : : * they use the flinfo parameter to initialise the fcinfo for the call.
1060 : : * It's recommended that the callee only use the fn_extra and fn_mcxt
1061 : : * fields, as other fields will typically describe the calling function
1062 : : * not the callee. Conversely, the calling function should not have
1063 : : * used fn_extra, unless its use is known to be compatible with the callee's.
1064 : : */
1065 : :
1066 : : Datum
3357 andrew@dunslane.net 1067 : 0 : CallerFInfoFunctionCall1(PGFunction func, FmgrInfo *flinfo, Oid collation, Datum arg1)
1068 : : {
2681 andres@anarazel.de 1069 : 0 : LOCAL_FCINFO(fcinfo, 1);
1070 : : Datum result;
1071 : :
1072 : 0 : InitFunctionCallInfoData(*fcinfo, flinfo, 1, collation, NULL, NULL);
1073 : :
1074 : 0 : fcinfo->args[0].value = arg1;
1075 : 0 : fcinfo->args[0].isnull = false;
1076 : :
1077 : 0 : result = (*func) (fcinfo);
1078 : :
1079 : : /* Check for null result, since caller is clearly not expecting one */
1080 [ # # ]: 0 : if (fcinfo->isnull)
3357 andrew@dunslane.net 1081 [ # # ]: 0 : elog(ERROR, "function %p returned NULL", (void *) func);
1082 : :
1083 : 0 : return result;
1084 : : }
1085 : :
1086 : : Datum
3357 andrew@dunslane.net 1087 :CBC 210336 : CallerFInfoFunctionCall2(PGFunction func, FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2)
1088 : : {
2681 andres@anarazel.de 1089 : 210336 : LOCAL_FCINFO(fcinfo, 2);
1090 : : Datum result;
1091 : :
1092 : 210336 : InitFunctionCallInfoData(*fcinfo, flinfo, 2, collation, NULL, NULL);
1093 : :
1094 : 210336 : fcinfo->args[0].value = arg1;
1095 : 210336 : fcinfo->args[0].isnull = false;
1096 : 210336 : fcinfo->args[1].value = arg2;
1097 : 210336 : fcinfo->args[1].isnull = false;
1098 : :
1099 : 210336 : result = (*func) (fcinfo);
1100 : :
1101 : : /* Check for null result, since caller is clearly not expecting one */
1102 [ - + ]: 210336 : if (fcinfo->isnull)
3357 andrew@dunslane.net 1103 [ # # ]:UBC 0 : elog(ERROR, "function %p returned NULL", (void *) func);
1104 : :
3357 andrew@dunslane.net 1105 :CBC 210336 : return result;
1106 : : }
1107 : :
1108 : : /*
1109 : : * These are for invocation of a previously-looked-up function with a
1110 : : * directly-computed parameter list. Note that neither arguments nor result
1111 : : * are allowed to be NULL.
1112 : : */
1113 : : Datum
2681 andres@anarazel.de 1114 : 3354091 : FunctionCall0Coll(FmgrInfo *flinfo, Oid collation)
1115 : : {
1116 : 3354091 : LOCAL_FCINFO(fcinfo, 0);
1117 : : Datum result;
1118 : :
1119 : 3354091 : InitFunctionCallInfoData(*fcinfo, flinfo, 0, collation, NULL, NULL);
1120 : :
1121 : 3354091 : result = FunctionCallInvoke(fcinfo);
1122 : :
1123 : : /* Check for null result, since caller is clearly not expecting one */
1124 [ - + ]: 3354091 : if (fcinfo->isnull)
2681 andres@anarazel.de 1125 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1126 : :
2681 andres@anarazel.de 1127 :CBC 3354091 : return result;
1128 : : }
1129 : :
1130 : : Datum
5527 tgl@sss.pgh.pa.us 1131 : 39888143 : FunctionCall1Coll(FmgrInfo *flinfo, Oid collation, Datum arg1)
1132 : : {
2681 andres@anarazel.de 1133 : 39888143 : LOCAL_FCINFO(fcinfo, 1);
1134 : : Datum result;
1135 : :
1136 : 39888143 : InitFunctionCallInfoData(*fcinfo, flinfo, 1, collation, NULL, NULL);
1137 : :
1138 : 39888143 : fcinfo->args[0].value = arg1;
1139 : 39888143 : fcinfo->args[0].isnull = false;
1140 : :
1141 : 39888143 : result = FunctionCallInvoke(fcinfo);
1142 : :
1143 : : /* Check for null result, since caller is clearly not expecting one */
1144 [ - + ]: 39887737 : if (fcinfo->isnull)
2681 andres@anarazel.de 1145 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1146 : :
9498 tgl@sss.pgh.pa.us 1147 :CBC 39887737 : return result;
1148 : : }
1149 : :
1150 : : Datum
5527 1151 : 555380611 : FunctionCall2Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2)
1152 : : {
2681 andres@anarazel.de 1153 : 555380611 : LOCAL_FCINFO(fcinfo, 2);
1154 : : Datum result;
1155 : :
1156 : 555380611 : InitFunctionCallInfoData(*fcinfo, flinfo, 2, collation, NULL, NULL);
1157 : :
1158 : 555380611 : fcinfo->args[0].value = arg1;
1159 : 555380611 : fcinfo->args[0].isnull = false;
1160 : 555380611 : fcinfo->args[1].value = arg2;
1161 : 555380611 : fcinfo->args[1].isnull = false;
1162 : :
1163 : 555380611 : result = FunctionCallInvoke(fcinfo);
1164 : :
1165 : : /* Check for null result, since caller is clearly not expecting one */
1166 [ - + ]: 555380519 : if (fcinfo->isnull)
2681 andres@anarazel.de 1167 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1168 : :
9498 tgl@sss.pgh.pa.us 1169 :CBC 555380519 : return result;
1170 : : }
1171 : :
1172 : : Datum
5527 1173 : 30927283 : FunctionCall3Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1174 : : Datum arg3)
1175 : : {
2681 andres@anarazel.de 1176 : 30927283 : LOCAL_FCINFO(fcinfo, 3);
1177 : : Datum result;
1178 : :
1179 : 30927283 : InitFunctionCallInfoData(*fcinfo, flinfo, 3, collation, NULL, NULL);
1180 : :
1181 : 30927283 : fcinfo->args[0].value = arg1;
1182 : 30927283 : fcinfo->args[0].isnull = false;
1183 : 30927283 : fcinfo->args[1].value = arg2;
1184 : 30927283 : fcinfo->args[1].isnull = false;
1185 : 30927283 : fcinfo->args[2].value = arg3;
1186 : 30927283 : fcinfo->args[2].isnull = false;
1187 : :
1188 : 30927283 : result = FunctionCallInvoke(fcinfo);
1189 : :
1190 : : /* Check for null result, since caller is clearly not expecting one */
1191 [ - + ]: 30927271 : if (fcinfo->isnull)
2681 andres@anarazel.de 1192 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1193 : :
9498 tgl@sss.pgh.pa.us 1194 :CBC 30927271 : return result;
1195 : : }
1196 : :
1197 : : Datum
5527 1198 : 1416746 : FunctionCall4Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1199 : : Datum arg3, Datum arg4)
1200 : : {
2681 andres@anarazel.de 1201 : 1416746 : LOCAL_FCINFO(fcinfo, 4);
1202 : : Datum result;
1203 : :
1204 : 1416746 : InitFunctionCallInfoData(*fcinfo, flinfo, 4, collation, NULL, NULL);
1205 : :
1206 : 1416746 : fcinfo->args[0].value = arg1;
1207 : 1416746 : fcinfo->args[0].isnull = false;
1208 : 1416746 : fcinfo->args[1].value = arg2;
1209 : 1416746 : fcinfo->args[1].isnull = false;
1210 : 1416746 : fcinfo->args[2].value = arg3;
1211 : 1416746 : fcinfo->args[2].isnull = false;
1212 : 1416746 : fcinfo->args[3].value = arg4;
1213 : 1416746 : fcinfo->args[3].isnull = false;
1214 : :
1215 : 1416746 : result = FunctionCallInvoke(fcinfo);
1216 : :
1217 : : /* Check for null result, since caller is clearly not expecting one */
1218 [ - + ]: 1416726 : if (fcinfo->isnull)
2681 andres@anarazel.de 1219 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1220 : :
9498 tgl@sss.pgh.pa.us 1221 :CBC 1416726 : return result;
1222 : : }
1223 : :
1224 : : Datum
5527 1225 : 1490831 : FunctionCall5Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1226 : : Datum arg3, Datum arg4, Datum arg5)
1227 : : {
2681 andres@anarazel.de 1228 : 1490831 : LOCAL_FCINFO(fcinfo, 5);
1229 : : Datum result;
1230 : :
1231 : 1490831 : InitFunctionCallInfoData(*fcinfo, flinfo, 5, collation, NULL, NULL);
1232 : :
1233 : 1490831 : fcinfo->args[0].value = arg1;
1234 : 1490831 : fcinfo->args[0].isnull = false;
1235 : 1490831 : fcinfo->args[1].value = arg2;
1236 : 1490831 : fcinfo->args[1].isnull = false;
1237 : 1490831 : fcinfo->args[2].value = arg3;
1238 : 1490831 : fcinfo->args[2].isnull = false;
1239 : 1490831 : fcinfo->args[3].value = arg4;
1240 : 1490831 : fcinfo->args[3].isnull = false;
1241 : 1490831 : fcinfo->args[4].value = arg5;
1242 : 1490831 : fcinfo->args[4].isnull = false;
1243 : :
1244 : 1490831 : result = FunctionCallInvoke(fcinfo);
1245 : :
1246 : : /* Check for null result, since caller is clearly not expecting one */
1247 [ - + ]: 1490767 : if (fcinfo->isnull)
2681 andres@anarazel.de 1248 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1249 : :
9498 tgl@sss.pgh.pa.us 1250 :CBC 1490767 : return result;
1251 : : }
1252 : :
1253 : : Datum
5527 1254 : 3346 : FunctionCall6Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1255 : : Datum arg3, Datum arg4, Datum arg5,
1256 : : Datum arg6)
1257 : : {
2681 andres@anarazel.de 1258 : 3346 : LOCAL_FCINFO(fcinfo, 6);
1259 : : Datum result;
1260 : :
1261 : 3346 : InitFunctionCallInfoData(*fcinfo, flinfo, 6, collation, NULL, NULL);
1262 : :
1263 : 3346 : fcinfo->args[0].value = arg1;
1264 : 3346 : fcinfo->args[0].isnull = false;
1265 : 3346 : fcinfo->args[1].value = arg2;
1266 : 3346 : fcinfo->args[1].isnull = false;
1267 : 3346 : fcinfo->args[2].value = arg3;
1268 : 3346 : fcinfo->args[2].isnull = false;
1269 : 3346 : fcinfo->args[3].value = arg4;
1270 : 3346 : fcinfo->args[3].isnull = false;
1271 : 3346 : fcinfo->args[4].value = arg5;
1272 : 3346 : fcinfo->args[4].isnull = false;
1273 : 3346 : fcinfo->args[5].value = arg6;
1274 : 3346 : fcinfo->args[5].isnull = false;
1275 : :
1276 : 3346 : result = FunctionCallInvoke(fcinfo);
1277 : :
1278 : : /* Check for null result, since caller is clearly not expecting one */
1279 [ - + ]: 2306 : if (fcinfo->isnull)
2681 andres@anarazel.de 1280 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1281 : :
9498 tgl@sss.pgh.pa.us 1282 :CBC 2306 : return result;
1283 : : }
1284 : :
1285 : : Datum
5527 1286 : 615657 : FunctionCall7Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1287 : : Datum arg3, Datum arg4, Datum arg5,
1288 : : Datum arg6, Datum arg7)
1289 : : {
2681 andres@anarazel.de 1290 : 615657 : LOCAL_FCINFO(fcinfo, 7);
1291 : : Datum result;
1292 : :
1293 : 615657 : InitFunctionCallInfoData(*fcinfo, flinfo, 7, collation, NULL, NULL);
1294 : :
1295 : 615657 : fcinfo->args[0].value = arg1;
1296 : 615657 : fcinfo->args[0].isnull = false;
1297 : 615657 : fcinfo->args[1].value = arg2;
1298 : 615657 : fcinfo->args[1].isnull = false;
1299 : 615657 : fcinfo->args[2].value = arg3;
1300 : 615657 : fcinfo->args[2].isnull = false;
1301 : 615657 : fcinfo->args[3].value = arg4;
1302 : 615657 : fcinfo->args[3].isnull = false;
1303 : 615657 : fcinfo->args[4].value = arg5;
1304 : 615657 : fcinfo->args[4].isnull = false;
1305 : 615657 : fcinfo->args[5].value = arg6;
1306 : 615657 : fcinfo->args[5].isnull = false;
1307 : 615657 : fcinfo->args[6].value = arg7;
1308 : 615657 : fcinfo->args[6].isnull = false;
1309 : :
1310 : 615657 : result = FunctionCallInvoke(fcinfo);
1311 : :
1312 : : /* Check for null result, since caller is clearly not expecting one */
1313 [ - + ]: 615657 : if (fcinfo->isnull)
2681 andres@anarazel.de 1314 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1315 : :
9498 tgl@sss.pgh.pa.us 1316 :CBC 615657 : return result;
1317 : : }
1318 : :
1319 : : Datum
5527 1320 : 19417 : FunctionCall8Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1321 : : Datum arg3, Datum arg4, Datum arg5,
1322 : : Datum arg6, Datum arg7, Datum arg8)
1323 : : {
2681 andres@anarazel.de 1324 : 19417 : LOCAL_FCINFO(fcinfo, 8);
1325 : : Datum result;
1326 : :
1327 : 19417 : InitFunctionCallInfoData(*fcinfo, flinfo, 8, collation, NULL, NULL);
1328 : :
1329 : 19417 : fcinfo->args[0].value = arg1;
1330 : 19417 : fcinfo->args[0].isnull = false;
1331 : 19417 : fcinfo->args[1].value = arg2;
1332 : 19417 : fcinfo->args[1].isnull = false;
1333 : 19417 : fcinfo->args[2].value = arg3;
1334 : 19417 : fcinfo->args[2].isnull = false;
1335 : 19417 : fcinfo->args[3].value = arg4;
1336 : 19417 : fcinfo->args[3].isnull = false;
1337 : 19417 : fcinfo->args[4].value = arg5;
1338 : 19417 : fcinfo->args[4].isnull = false;
1339 : 19417 : fcinfo->args[5].value = arg6;
1340 : 19417 : fcinfo->args[5].isnull = false;
1341 : 19417 : fcinfo->args[6].value = arg7;
1342 : 19417 : fcinfo->args[6].isnull = false;
1343 : 19417 : fcinfo->args[7].value = arg8;
1344 : 19417 : fcinfo->args[7].isnull = false;
1345 : :
1346 : 19417 : result = FunctionCallInvoke(fcinfo);
1347 : :
1348 : : /* Check for null result, since caller is clearly not expecting one */
1349 [ - + ]: 19417 : if (fcinfo->isnull)
2681 andres@anarazel.de 1350 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1351 : :
9498 tgl@sss.pgh.pa.us 1352 :CBC 19417 : return result;
1353 : : }
1354 : :
1355 : : Datum
5527 tgl@sss.pgh.pa.us 1356 :UBC 0 : FunctionCall9Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2,
1357 : : Datum arg3, Datum arg4, Datum arg5,
1358 : : Datum arg6, Datum arg7, Datum arg8,
1359 : : Datum arg9)
1360 : : {
2681 andres@anarazel.de 1361 : 0 : LOCAL_FCINFO(fcinfo, 9);
1362 : : Datum result;
1363 : :
1364 : 0 : InitFunctionCallInfoData(*fcinfo, flinfo, 9, collation, NULL, NULL);
1365 : :
1366 : 0 : fcinfo->args[0].value = arg1;
1367 : 0 : fcinfo->args[0].isnull = false;
1368 : 0 : fcinfo->args[1].value = arg2;
1369 : 0 : fcinfo->args[1].isnull = false;
1370 : 0 : fcinfo->args[2].value = arg3;
1371 : 0 : fcinfo->args[2].isnull = false;
1372 : 0 : fcinfo->args[3].value = arg4;
1373 : 0 : fcinfo->args[3].isnull = false;
1374 : 0 : fcinfo->args[4].value = arg5;
1375 : 0 : fcinfo->args[4].isnull = false;
1376 : 0 : fcinfo->args[5].value = arg6;
1377 : 0 : fcinfo->args[5].isnull = false;
1378 : 0 : fcinfo->args[6].value = arg7;
1379 : 0 : fcinfo->args[6].isnull = false;
1380 : 0 : fcinfo->args[7].value = arg8;
1381 : 0 : fcinfo->args[7].isnull = false;
1382 : 0 : fcinfo->args[8].value = arg9;
1383 : 0 : fcinfo->args[8].isnull = false;
1384 : :
1385 : 0 : result = FunctionCallInvoke(fcinfo);
1386 : :
1387 : : /* Check for null result, since caller is clearly not expecting one */
1388 [ # # ]: 0 : if (fcinfo->isnull)
1389 [ # # ]: 0 : elog(ERROR, "function %u returned NULL", flinfo->fn_oid);
1390 : :
9498 tgl@sss.pgh.pa.us 1391 : 0 : return result;
1392 : : }
1393 : :
1394 : :
1395 : : /*
1396 : : * These are for invocation of a function identified by OID with a
1397 : : * directly-computed parameter list. Note that neither arguments nor result
1398 : : * are allowed to be NULL. These are essentially fmgr_info() followed
1399 : : * by FunctionCallN(). If the same function is to be invoked repeatedly,
1400 : : * do the fmgr_info() once and then use FunctionCallN().
1401 : : */
1402 : : Datum
5527 tgl@sss.pgh.pa.us 1403 :CBC 3354091 : OidFunctionCall0Coll(Oid functionId, Oid collation)
1404 : : {
1405 : : FmgrInfo flinfo;
1406 : :
5578 1407 : 3354091 : fmgr_info(functionId, &flinfo);
1408 : :
2681 andres@anarazel.de 1409 : 3354091 : return FunctionCall0Coll(&flinfo, collation);
1410 : : }
1411 : :
1412 : : Datum
5527 tgl@sss.pgh.pa.us 1413 : 497619 : OidFunctionCall1Coll(Oid functionId, Oid collation, Datum arg1)
1414 : : {
1415 : : FmgrInfo flinfo;
1416 : :
9498 1417 : 497619 : fmgr_info(functionId, &flinfo);
1418 : :
2681 andres@anarazel.de 1419 : 497619 : return FunctionCall1Coll(&flinfo, collation, arg1);
1420 : : }
1421 : :
1422 : : Datum
5527 tgl@sss.pgh.pa.us 1423 : 728 : OidFunctionCall2Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2)
1424 : : {
1425 : : FmgrInfo flinfo;
1426 : :
9498 1427 : 728 : fmgr_info(functionId, &flinfo);
1428 : :
2681 andres@anarazel.de 1429 : 728 : return FunctionCall2Coll(&flinfo, collation, arg1, arg2);
1430 : : }
1431 : :
1432 : : Datum
5527 tgl@sss.pgh.pa.us 1433 : 2415 : OidFunctionCall3Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1434 : : Datum arg3)
1435 : : {
1436 : : FmgrInfo flinfo;
1437 : :
9498 1438 : 2415 : fmgr_info(functionId, &flinfo);
1439 : :
2681 andres@anarazel.de 1440 : 2415 : return FunctionCall3Coll(&flinfo, collation, arg1, arg2, arg3);
1441 : : }
1442 : :
1443 : : Datum
5527 tgl@sss.pgh.pa.us 1444 : 606152 : OidFunctionCall4Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1445 : : Datum arg3, Datum arg4)
1446 : : {
1447 : : FmgrInfo flinfo;
1448 : :
9498 1449 : 606152 : fmgr_info(functionId, &flinfo);
1450 : :
2681 andres@anarazel.de 1451 : 606152 : return FunctionCall4Coll(&flinfo, collation, arg1, arg2, arg3, arg4);
1452 : : }
1453 : :
1454 : : Datum
5527 tgl@sss.pgh.pa.us 1455 : 221303 : OidFunctionCall5Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1456 : : Datum arg3, Datum arg4, Datum arg5)
1457 : : {
1458 : : FmgrInfo flinfo;
1459 : :
9498 1460 : 221303 : fmgr_info(functionId, &flinfo);
1461 : :
2681 andres@anarazel.de 1462 : 221303 : return FunctionCall5Coll(&flinfo, collation, arg1, arg2, arg3, arg4, arg5);
1463 : : }
1464 : :
1465 : : Datum
5527 tgl@sss.pgh.pa.us 1466 : 3082 : OidFunctionCall6Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1467 : : Datum arg3, Datum arg4, Datum arg5,
1468 : : Datum arg6)
1469 : : {
1470 : : FmgrInfo flinfo;
1471 : :
9498 1472 : 3082 : fmgr_info(functionId, &flinfo);
1473 : :
2681 andres@anarazel.de 1474 : 3082 : return FunctionCall6Coll(&flinfo, collation, arg1, arg2, arg3, arg4, arg5,
1475 : : arg6);
1476 : : }
1477 : :
1478 : : Datum
5527 tgl@sss.pgh.pa.us 1479 :UBC 0 : OidFunctionCall7Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1480 : : Datum arg3, Datum arg4, Datum arg5,
1481 : : Datum arg6, Datum arg7)
1482 : : {
1483 : : FmgrInfo flinfo;
1484 : :
9498 1485 : 0 : fmgr_info(functionId, &flinfo);
1486 : :
2681 andres@anarazel.de 1487 : 0 : return FunctionCall7Coll(&flinfo, collation, arg1, arg2, arg3, arg4, arg5,
1488 : : arg6, arg7);
1489 : : }
1490 : :
1491 : : Datum
5527 tgl@sss.pgh.pa.us 1492 : 0 : OidFunctionCall8Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1493 : : Datum arg3, Datum arg4, Datum arg5,
1494 : : Datum arg6, Datum arg7, Datum arg8)
1495 : : {
1496 : : FmgrInfo flinfo;
1497 : :
9498 1498 : 0 : fmgr_info(functionId, &flinfo);
1499 : :
2681 andres@anarazel.de 1500 : 0 : return FunctionCall8Coll(&flinfo, collation, arg1, arg2, arg3, arg4, arg5,
1501 : : arg6, arg7, arg8);
1502 : : }
1503 : :
1504 : : Datum
5527 tgl@sss.pgh.pa.us 1505 : 0 : OidFunctionCall9Coll(Oid functionId, Oid collation, Datum arg1, Datum arg2,
1506 : : Datum arg3, Datum arg4, Datum arg5,
1507 : : Datum arg6, Datum arg7, Datum arg8,
1508 : : Datum arg9)
1509 : : {
1510 : : FmgrInfo flinfo;
1511 : :
9498 1512 : 0 : fmgr_info(functionId, &flinfo);
1513 : :
2681 andres@anarazel.de 1514 : 0 : return FunctionCall9Coll(&flinfo, collation, arg1, arg2, arg3, arg4, arg5,
1515 : : arg6, arg7, arg8, arg9);
1516 : : }
1517 : :
1518 : :
1519 : : /*
1520 : : * Special cases for convenient invocation of datatype I/O functions.
1521 : : */
1522 : :
1523 : : /*
1524 : : * Call a previously-looked-up datatype input function.
1525 : : *
1526 : : * "str" may be NULL to indicate we are reading a NULL. In this case
1527 : : * the caller should assume the result is NULL, but we'll call the input
1528 : : * function anyway if it's not strict. So this is almost but not quite
1529 : : * the same as FunctionCall3.
1530 : : */
1531 : : Datum
7361 tgl@sss.pgh.pa.us 1532 :CBC 22526517 : InputFunctionCall(FmgrInfo *flinfo, char *str, Oid typioparam, int32 typmod)
1533 : : {
2681 andres@anarazel.de 1534 : 22526517 : LOCAL_FCINFO(fcinfo, 3);
1535 : : Datum result;
1536 : :
7361 tgl@sss.pgh.pa.us 1537 [ + + + + ]: 22526517 : if (str == NULL && flinfo->fn_strict)
1538 : 4409908 : return (Datum) 0; /* just return null result */
1539 : :
2681 andres@anarazel.de 1540 : 18116609 : InitFunctionCallInfoData(*fcinfo, flinfo, 3, InvalidOid, NULL, NULL);
1541 : :
1542 : 18116609 : fcinfo->args[0].value = CStringGetDatum(str);
1543 : 18116609 : fcinfo->args[0].isnull = false;
1544 : 18116609 : fcinfo->args[1].value = ObjectIdGetDatum(typioparam);
1545 : 18116609 : fcinfo->args[1].isnull = false;
1546 : 18116609 : fcinfo->args[2].value = Int32GetDatum(typmod);
1547 : 18116609 : fcinfo->args[2].isnull = false;
1548 : :
1549 : 18116609 : result = FunctionCallInvoke(fcinfo);
1550 : :
1551 : : /* Should get null result if and only if str is NULL */
7361 tgl@sss.pgh.pa.us 1552 [ - + ]: 18113095 : if (str == NULL)
1553 : : {
2681 andres@anarazel.de 1554 [ # # ]:UBC 0 : if (!fcinfo->isnull)
7361 tgl@sss.pgh.pa.us 1555 [ # # ]: 0 : elog(ERROR, "input function %u returned non-NULL",
1556 : : flinfo->fn_oid);
1557 : : }
1558 : : else
1559 : : {
2681 andres@anarazel.de 1560 [ - + ]:CBC 18113095 : if (fcinfo->isnull)
7361 tgl@sss.pgh.pa.us 1561 [ # # ]:UBC 0 : elog(ERROR, "input function %u returned NULL",
1562 : : flinfo->fn_oid);
1563 : : }
1564 : :
7361 tgl@sss.pgh.pa.us 1565 :CBC 18113095 : return result;
1566 : : }
1567 : :
1568 : : /*
1569 : : * Call a previously-looked-up datatype input function, with non-exception
1570 : : * handling of "soft" errors.
1571 : : *
1572 : : * This is basically like InputFunctionCall, but the converted Datum is
1573 : : * returned into *result while the function result is true for success or
1574 : : * false for failure. Also, the caller may pass an ErrorSaveContext node.
1575 : : *
1576 : : * If escontext points to an ErrorSaveContext, any "soft" errors detected by
1577 : : * the input function will be reported by filling the escontext struct and
1578 : : * returning false. (The caller can choose to test SOFT_ERROR_OCCURRED(),
1579 : : * but checking the function result instead is usually cheaper.)
1580 : : *
1581 : : * If escontext does not point to an ErrorSaveContext, errors are reported
1582 : : * via ereport(ERROR), so that there is no functional difference from
1583 : : * InputFunctionCall; the result will always be true if control returns.
1584 : : */
1585 : : bool
1268 1586 : 3834715 : InputFunctionCallSafe(FmgrInfo *flinfo, char *str,
1587 : : Oid typioparam, int32 typmod,
1588 : : Node *escontext,
1589 : : Datum *result)
1590 : : {
1591 : 3834715 : LOCAL_FCINFO(fcinfo, 3);
1592 : :
1593 [ + + + + ]: 3834715 : if (str == NULL && flinfo->fn_strict)
1594 : : {
1595 : 3922 : *result = (Datum) 0; /* just return null result */
1596 : 3922 : return true;
1597 : : }
1598 : :
1599 : 3830793 : InitFunctionCallInfoData(*fcinfo, flinfo, 3, InvalidOid, escontext, NULL);
1600 : :
1601 : 3830793 : fcinfo->args[0].value = CStringGetDatum(str);
1602 : 3830793 : fcinfo->args[0].isnull = false;
1603 : 3830793 : fcinfo->args[1].value = ObjectIdGetDatum(typioparam);
1604 : 3830793 : fcinfo->args[1].isnull = false;
1605 : 3830793 : fcinfo->args[2].value = Int32GetDatum(typmod);
1606 : 3830793 : fcinfo->args[2].isnull = false;
1607 : :
1608 : 3830793 : *result = FunctionCallInvoke(fcinfo);
1609 : :
1610 : : /* Result value is garbage, and could be null, if an error was reported */
1611 [ + + + - : 3830570 : if (SOFT_ERROR_OCCURRED(escontext))
+ + ]
1612 : 1604 : return false;
1613 : :
1614 : : /* Otherwise, should get null result if and only if str is NULL */
1615 [ + + ]: 3828966 : if (str == NULL)
1616 : : {
1617 [ - + ]: 32 : if (!fcinfo->isnull)
1268 tgl@sss.pgh.pa.us 1618 [ # # ]:UBC 0 : elog(ERROR, "input function %u returned non-NULL",
1619 : : flinfo->fn_oid);
1620 : : }
1621 : : else
1622 : : {
1268 tgl@sss.pgh.pa.us 1623 [ - + ]:CBC 3828934 : if (fcinfo->isnull)
1268 tgl@sss.pgh.pa.us 1624 [ # # ]:UBC 0 : elog(ERROR, "input function %u returned NULL",
1625 : : flinfo->fn_oid);
1626 : : }
1627 : :
1268 tgl@sss.pgh.pa.us 1628 :CBC 3828966 : return true;
1629 : : }
1630 : :
1631 : : /*
1632 : : * Call a directly-named datatype input function, with non-exception
1633 : : * handling of "soft" errors.
1634 : : *
1635 : : * This is like InputFunctionCallSafe, except that it is given a direct
1636 : : * pointer to the C function to call. We assume that that function is
1637 : : * strict. Also, the function cannot be one that needs to
1638 : : * look at FmgrInfo, since there won't be any.
1639 : : */
1640 : : bool
1266 1641 : 830454 : DirectInputFunctionCallSafe(PGFunction func, char *str,
1642 : : Oid typioparam, int32 typmod,
1643 : : Node *escontext,
1644 : : Datum *result)
1645 : : {
1646 : 830454 : LOCAL_FCINFO(fcinfo, 3);
1647 : :
1648 [ - + ]: 830454 : if (str == NULL)
1649 : : {
1266 tgl@sss.pgh.pa.us 1650 :UBC 0 : *result = (Datum) 0; /* just return null result */
1651 : 0 : return true;
1652 : : }
1653 : :
1266 tgl@sss.pgh.pa.us 1654 :CBC 830454 : InitFunctionCallInfoData(*fcinfo, NULL, 3, InvalidOid, escontext, NULL);
1655 : :
1656 : 830454 : fcinfo->args[0].value = CStringGetDatum(str);
1657 : 830454 : fcinfo->args[0].isnull = false;
1658 : 830454 : fcinfo->args[1].value = ObjectIdGetDatum(typioparam);
1659 : 830454 : fcinfo->args[1].isnull = false;
1660 : 830454 : fcinfo->args[2].value = Int32GetDatum(typmod);
1661 : 830454 : fcinfo->args[2].isnull = false;
1662 : :
1663 : 830454 : *result = (*func) (fcinfo);
1664 : :
1665 : : /* Result value is garbage, and could be null, if an error was reported */
1666 [ + + + - : 830454 : if (SOFT_ERROR_OCCURRED(escontext))
+ + ]
1667 : 220 : return false;
1668 : :
1669 : : /* Otherwise, shouldn't get null result */
1670 [ - + ]: 830234 : if (fcinfo->isnull)
1266 tgl@sss.pgh.pa.us 1671 [ # # ]:UBC 0 : elog(ERROR, "input function %p returned NULL", (void *) func);
1672 : :
1266 tgl@sss.pgh.pa.us 1673 :CBC 830234 : return true;
1674 : : }
1675 : :
1676 : : /*
1677 : : * Call a previously-looked-up datatype output function.
1678 : : *
1679 : : * Do not call this on NULL datums.
1680 : : *
1681 : : * This is currently little more than window dressing for FunctionCall1.
1682 : : */
1683 : : char *
7361 1684 : 32365938 : OutputFunctionCall(FmgrInfo *flinfo, Datum val)
1685 : : {
3490 1686 : 32365938 : return DatumGetCString(FunctionCall1(flinfo, val));
1687 : : }
1688 : :
1689 : : /*
1690 : : * Call a previously-looked-up datatype binary-input function.
1691 : : *
1692 : : * "buf" may be NULL to indicate we are reading a NULL. In this case
1693 : : * the caller should assume the result is NULL, but we'll call the receive
1694 : : * function anyway if it's not strict. So this is almost but not quite
1695 : : * the same as FunctionCall3.
1696 : : */
1697 : : Datum
7361 1698 : 156141 : ReceiveFunctionCall(FmgrInfo *flinfo, StringInfo buf,
1699 : : Oid typioparam, int32 typmod)
1700 : : {
2681 andres@anarazel.de 1701 : 156141 : LOCAL_FCINFO(fcinfo, 3);
1702 : : Datum result;
1703 : :
7361 tgl@sss.pgh.pa.us 1704 [ + + + - ]: 156141 : if (buf == NULL && flinfo->fn_strict)
1705 : 20 : return (Datum) 0; /* just return null result */
1706 : :
2681 andres@anarazel.de 1707 : 156121 : InitFunctionCallInfoData(*fcinfo, flinfo, 3, InvalidOid, NULL, NULL);
1708 : :
1709 : 156121 : fcinfo->args[0].value = PointerGetDatum(buf);
1710 : 156121 : fcinfo->args[0].isnull = false;
1711 : 156121 : fcinfo->args[1].value = ObjectIdGetDatum(typioparam);
1712 : 156121 : fcinfo->args[1].isnull = false;
1713 : 156121 : fcinfo->args[2].value = Int32GetDatum(typmod);
1714 : 156121 : fcinfo->args[2].isnull = false;
1715 : :
1716 : 156121 : result = FunctionCallInvoke(fcinfo);
1717 : :
1718 : : /* Should get null result if and only if buf is NULL */
7361 tgl@sss.pgh.pa.us 1719 [ - + ]: 156121 : if (buf == NULL)
1720 : : {
2681 andres@anarazel.de 1721 [ # # ]:UBC 0 : if (!fcinfo->isnull)
7361 tgl@sss.pgh.pa.us 1722 [ # # ]: 0 : elog(ERROR, "receive function %u returned non-NULL",
1723 : : flinfo->fn_oid);
1724 : : }
1725 : : else
1726 : : {
2681 andres@anarazel.de 1727 [ - + ]:CBC 156121 : if (fcinfo->isnull)
7361 tgl@sss.pgh.pa.us 1728 [ # # ]:UBC 0 : elog(ERROR, "receive function %u returned NULL",
1729 : : flinfo->fn_oid);
1730 : : }
1731 : :
7361 tgl@sss.pgh.pa.us 1732 :CBC 156121 : return result;
1733 : : }
1734 : :
1735 : : /*
1736 : : * Call a previously-looked-up datatype binary-output function.
1737 : : *
1738 : : * Do not call this on NULL datums.
1739 : : *
1740 : : * This is little more than window dressing for FunctionCall1, but it does
1741 : : * guarantee a non-toasted result, which strictly speaking the underlying
1742 : : * function doesn't.
1743 : : */
1744 : : bytea *
1745 : 123689 : SendFunctionCall(FmgrInfo *flinfo, Datum val)
1746 : : {
3490 1747 : 123689 : return DatumGetByteaP(FunctionCall1(flinfo, val));
1748 : : }
1749 : :
1750 : : /*
1751 : : * As above, for I/O functions identified by OID. These are only to be used
1752 : : * in seldom-executed code paths. They are not only slow but leak memory.
1753 : : */
1754 : : Datum
7361 1755 : 8112947 : OidInputFunctionCall(Oid functionId, char *str, Oid typioparam, int32 typmod)
1756 : : {
1757 : : FmgrInfo flinfo;
1758 : :
1759 : 8112947 : fmgr_info(functionId, &flinfo);
1760 : 8112947 : return InputFunctionCall(&flinfo, str, typioparam, typmod);
1761 : : }
1762 : :
1763 : : char *
1764 : 604170 : OidOutputFunctionCall(Oid functionId, Datum val)
1765 : : {
1766 : : FmgrInfo flinfo;
1767 : :
1768 : 604170 : fmgr_info(functionId, &flinfo);
1769 : 604170 : return OutputFunctionCall(&flinfo, val);
1770 : : }
1771 : :
1772 : : Datum
1773 : 155948 : OidReceiveFunctionCall(Oid functionId, StringInfo buf,
1774 : : Oid typioparam, int32 typmod)
1775 : : {
1776 : : FmgrInfo flinfo;
1777 : :
1778 : 155948 : fmgr_info(functionId, &flinfo);
1779 : 155948 : return ReceiveFunctionCall(&flinfo, buf, typioparam, typmod);
1780 : : }
1781 : :
1782 : : bytea *
1783 : 116386 : OidSendFunctionCall(Oid functionId, Datum val)
1784 : : {
1785 : : FmgrInfo flinfo;
1786 : :
1787 : 116386 : fmgr_info(functionId, &flinfo);
1788 : 116386 : return SendFunctionCall(&flinfo, val);
1789 : : }
1790 : :
1791 : :
1792 : : /*-------------------------------------------------------------------------
1793 : : * Support routines for toastable datatypes
1794 : : *-------------------------------------------------------------------------
1795 : : */
1796 : :
1797 : : varlena *
108 michael@paquier.xyz 1798 :GNC 92331347 : pg_detoast_datum(varlena *datum)
1799 : : {
9459 tgl@sss.pgh.pa.us 1800 [ + + ]:CBC 92331347 : if (VARATT_IS_EXTENDED(datum))
2430 rhaas@postgresql.org 1801 : 32041631 : return detoast_attr(datum);
1802 : : else
9459 tgl@sss.pgh.pa.us 1803 : 60289716 : return datum;
1804 : : }
1805 : :
1806 : : varlena *
108 michael@paquier.xyz 1807 :GNC 3877886 : pg_detoast_datum_copy(varlena *datum)
1808 : : {
9459 tgl@sss.pgh.pa.us 1809 [ + + ]:CBC 3877886 : if (VARATT_IS_EXTENDED(datum))
2430 rhaas@postgresql.org 1810 : 2208870 : return detoast_attr(datum);
1811 : : else
1812 : : {
1813 : : /* Make a modifiable copy of the varlena object */
9200 bruce@momjian.us 1814 : 1669016 : Size len = VARSIZE(datum);
108 michael@paquier.xyz 1815 :GNC 1669016 : varlena *result = (varlena *) palloc(len);
1816 : :
9459 tgl@sss.pgh.pa.us 1817 :CBC 1669016 : memcpy(result, datum, len);
1818 : 1669016 : return result;
1819 : : }
1820 : : }
1821 : :
1822 : : varlena *
108 michael@paquier.xyz 1823 :GNC 2429 : pg_detoast_datum_slice(varlena *datum, int32 first, int32 count)
1824 : : {
1825 : : /* Only get the specified portion from the toast rel */
2430 rhaas@postgresql.org 1826 :CBC 2429 : return detoast_attr_slice(datum, first, count);
1827 : : }
1828 : :
1829 : : varlena *
108 michael@paquier.xyz 1830 :GNC 151632585 : pg_detoast_datum_packed(varlena *datum)
1831 : : {
6994 tgl@sss.pgh.pa.us 1832 [ + + + + ]:CBC 151632585 : if (VARATT_IS_COMPRESSED(datum) || VARATT_IS_EXTERNAL(datum))
2430 rhaas@postgresql.org 1833 : 36066 : return detoast_attr(datum);
1834 : : else
6994 tgl@sss.pgh.pa.us 1835 : 151596519 : return datum;
1836 : : }
1837 : :
1838 : : /*-------------------------------------------------------------------------
1839 : : * Support routines for extracting info from fn_expr parse tree
1840 : : *
1841 : : * These are needed by polymorphic functions, which accept multiple possible
1842 : : * input types and need help from the parser to know what they've got.
1843 : : * Also, some functions might be interested in whether a parameter is constant.
1844 : : * Functions taking VARIADIC ANY also need to know about the VARIADIC keyword.
1845 : : *-------------------------------------------------------------------------
1846 : : */
1847 : :
1848 : : /*
1849 : : * Get the actual type OID of the function return type
1850 : : *
1851 : : * Returns InvalidOid if information is not available
1852 : : */
1853 : : Oid
8369 1854 : 97585 : get_fn_expr_rettype(FmgrInfo *flinfo)
1855 : : {
1856 : : Node *expr;
1857 : :
1858 : : /*
1859 : : * can't return anything useful if we have no FmgrInfo or if its fn_expr
1860 : : * node has not been initialized
1861 : : */
1862 [ + - - + ]: 97585 : if (!flinfo || !flinfo->fn_expr)
8453 tgl@sss.pgh.pa.us 1863 :UBC 0 : return InvalidOid;
1864 : :
8369 tgl@sss.pgh.pa.us 1865 :CBC 97585 : expr = flinfo->fn_expr;
1866 : :
8453 1867 : 97585 : return exprType(expr);
1868 : : }
1869 : :
1870 : : /*
1871 : : * Get the actual type OID of a specific function argument (counting from 0)
1872 : : *
1873 : : * Returns InvalidOid if information is not available
1874 : : */
1875 : : Oid
8369 1876 : 1616651 : get_fn_expr_argtype(FmgrInfo *flinfo, int argnum)
1877 : : {
1878 : : /*
1879 : : * can't return anything useful if we have no FmgrInfo or if its fn_expr
1880 : : * node has not been initialized
1881 : : */
1882 [ + - - + ]: 1616651 : if (!flinfo || !flinfo->fn_expr)
8453 tgl@sss.pgh.pa.us 1883 :UBC 0 : return InvalidOid;
1884 : :
7730 tgl@sss.pgh.pa.us 1885 :CBC 1616651 : return get_call_expr_argtype(flinfo->fn_expr, argnum);
1886 : : }
1887 : :
1888 : : /*
1889 : : * Get the actual type OID of a specific function argument (counting from 0),
1890 : : * but working from the calling expression tree instead of FmgrInfo
1891 : : *
1892 : : * Returns InvalidOid if information is not available
1893 : : */
1894 : : Oid
1895 : 1622637 : get_call_expr_argtype(Node *expr, int argnum)
1896 : : {
1897 : : List *args;
1898 : : Oid argtype;
1899 : :
1900 [ - + ]: 1622637 : if (expr == NULL)
7730 tgl@sss.pgh.pa.us 1901 :UBC 0 : return InvalidOid;
1902 : :
8453 tgl@sss.pgh.pa.us 1903 [ + + ]:CBC 1622637 : if (IsA(expr, FuncExpr))
1904 : 1622618 : args = ((FuncExpr *) expr)->args;
1905 [ + - ]: 19 : else if (IsA(expr, OpExpr))
1906 : 19 : args = ((OpExpr *) expr)->args;
8371 tgl@sss.pgh.pa.us 1907 [ # # ]:UBC 0 : else if (IsA(expr, DistinctExpr))
1908 : 0 : args = ((DistinctExpr *) expr)->args;
1909 [ # # ]: 0 : else if (IsA(expr, ScalarArrayOpExpr))
1910 : 0 : args = ((ScalarArrayOpExpr *) expr)->args;
1911 [ # # ]: 0 : else if (IsA(expr, NullIfExpr))
1912 : 0 : args = ((NullIfExpr *) expr)->args;
6362 1913 [ # # ]: 0 : else if (IsA(expr, WindowFunc))
1914 : 0 : args = ((WindowFunc *) expr)->args;
1915 : : else
8453 1916 : 0 : return InvalidOid;
1917 : :
8035 neilc@samurai.com 1918 [ + - - + ]:CBC 1622637 : if (argnum < 0 || argnum >= list_length(args))
8453 tgl@sss.pgh.pa.us 1919 :UBC 0 : return InvalidOid;
1920 : :
8035 neilc@samurai.com 1921 :CBC 1622637 : argtype = exprType((Node *) list_nth(args, argnum));
1922 : :
1923 : : /*
1924 : : * special hack for ScalarArrayOpExpr: what the underlying function will
1925 : : * actually get passed is the element type of the array.
1926 : : */
8371 tgl@sss.pgh.pa.us 1927 [ - + - - ]: 1622637 : if (IsA(expr, ScalarArrayOpExpr) &&
1928 : : argnum == 1)
5700 tgl@sss.pgh.pa.us 1929 :UBC 0 : argtype = get_base_element_type(argtype);
1930 : :
8371 tgl@sss.pgh.pa.us 1931 :CBC 1622637 : return argtype;
1932 : : }
1933 : :
1934 : : /*
1935 : : * Find out whether a specific function argument is constant for the
1936 : : * duration of a query
1937 : : *
1938 : : * Returns false if information is not available
1939 : : */
1940 : : bool
6362 1941 : 28967 : get_fn_expr_arg_stable(FmgrInfo *flinfo, int argnum)
1942 : : {
1943 : : /*
1944 : : * can't return anything useful if we have no FmgrInfo or if its fn_expr
1945 : : * node has not been initialized
1946 : : */
1947 [ + - - + ]: 28967 : if (!flinfo || !flinfo->fn_expr)
6362 tgl@sss.pgh.pa.us 1948 :UBC 0 : return false;
1949 : :
6362 tgl@sss.pgh.pa.us 1950 :CBC 28967 : return get_call_expr_arg_stable(flinfo->fn_expr, argnum);
1951 : : }
1952 : :
1953 : : /*
1954 : : * Find out whether a specific function argument is constant for the
1955 : : * duration of a query, but working from the calling expression tree
1956 : : *
1957 : : * Returns false if information is not available
1958 : : */
1959 : : bool
1960 : 28967 : get_call_expr_arg_stable(Node *expr, int argnum)
1961 : : {
1962 : : List *args;
1963 : : Node *arg;
1964 : :
1965 [ - + ]: 28967 : if (expr == NULL)
6362 tgl@sss.pgh.pa.us 1966 :UBC 0 : return false;
1967 : :
6362 tgl@sss.pgh.pa.us 1968 [ + + ]:CBC 28967 : if (IsA(expr, FuncExpr))
1969 : 27703 : args = ((FuncExpr *) expr)->args;
1970 [ - + ]: 1264 : else if (IsA(expr, OpExpr))
6362 tgl@sss.pgh.pa.us 1971 :UBC 0 : args = ((OpExpr *) expr)->args;
6362 tgl@sss.pgh.pa.us 1972 [ - + ]:CBC 1264 : else if (IsA(expr, DistinctExpr))
6362 tgl@sss.pgh.pa.us 1973 :UBC 0 : args = ((DistinctExpr *) expr)->args;
6362 tgl@sss.pgh.pa.us 1974 [ - + ]:CBC 1264 : else if (IsA(expr, ScalarArrayOpExpr))
6362 tgl@sss.pgh.pa.us 1975 :UBC 0 : args = ((ScalarArrayOpExpr *) expr)->args;
6362 tgl@sss.pgh.pa.us 1976 [ - + ]:CBC 1264 : else if (IsA(expr, NullIfExpr))
6362 tgl@sss.pgh.pa.us 1977 :UBC 0 : args = ((NullIfExpr *) expr)->args;
6362 tgl@sss.pgh.pa.us 1978 [ + - ]:CBC 1264 : else if (IsA(expr, WindowFunc))
1979 : 1264 : args = ((WindowFunc *) expr)->args;
1980 : : else
6362 tgl@sss.pgh.pa.us 1981 :UBC 0 : return false;
1982 : :
6362 tgl@sss.pgh.pa.us 1983 [ + - - + ]:CBC 28967 : if (argnum < 0 || argnum >= list_length(args))
6362 tgl@sss.pgh.pa.us 1984 :UBC 0 : return false;
1985 : :
6362 tgl@sss.pgh.pa.us 1986 :CBC 28967 : arg = (Node *) list_nth(args, argnum);
1987 : :
1988 : : /*
1989 : : * Either a true Const or an external Param will have a value that doesn't
1990 : : * change during the execution of the query. In future we might want to
1991 : : * consider other cases too, e.g. now().
1992 : : */
1993 [ + + ]: 28967 : if (IsA(arg, Const))
1994 : 28763 : return true;
1995 [ + + ]: 204 : if (IsA(arg, Param) &&
1996 [ - + ]: 2 : ((Param *) arg)->paramkind == PARAM_EXTERN)
6362 tgl@sss.pgh.pa.us 1997 :UBC 0 : return true;
1998 : :
6362 tgl@sss.pgh.pa.us 1999 :CBC 204 : return false;
2000 : : }
2001 : :
2002 : : /*
2003 : : * Get the VARIADIC flag from the function invocation
2004 : : *
2005 : : * Returns false (the default assumption) if information is not available
2006 : : *
2007 : : * Note this is generally only of interest to VARIADIC ANY functions
2008 : : */
2009 : : bool
4877 2010 : 26488 : get_fn_expr_variadic(FmgrInfo *flinfo)
2011 : : {
2012 : : Node *expr;
2013 : :
2014 : : /*
2015 : : * can't return anything useful if we have no FmgrInfo or if its fn_expr
2016 : : * node has not been initialized
2017 : : */
2018 [ + - - + ]: 26488 : if (!flinfo || !flinfo->fn_expr)
4877 tgl@sss.pgh.pa.us 2019 :UBC 0 : return false;
2020 : :
4877 tgl@sss.pgh.pa.us 2021 :CBC 26488 : expr = flinfo->fn_expr;
2022 : :
2023 [ + - ]: 26488 : if (IsA(expr, FuncExpr))
2024 : 26488 : return ((FuncExpr *) expr)->funcvariadic;
2025 : : else
4877 tgl@sss.pgh.pa.us 2026 :UBC 0 : return false;
2027 : : }
2028 : :
2029 : : /*
2030 : : * Set options to FmgrInfo of opclass support function.
2031 : : *
2032 : : * Opclass support functions are called outside of expressions. Thanks to that
2033 : : * we can use fn_expr to store opclass options as bytea constant.
2034 : : */
2035 : : void
2252 akorotkov@postgresql 2036 :CBC 681061 : set_fn_opclass_options(FmgrInfo *flinfo, bytea *options)
2037 : : {
2038 : 681061 : flinfo->fn_expr = (Node *) makeConst(BYTEAOID, -1, InvalidOid, -1,
2039 : : PointerGetDatum(options),
2040 : : options == NULL, false);
2041 : 681061 : }
2042 : :
2043 : : /*
2044 : : * Check if options are defined for opclass support function.
2045 : : */
2046 : : bool
2047 : 2763639 : has_fn_opclass_options(FmgrInfo *flinfo)
2048 : : {
2049 [ + - + - : 2763639 : if (flinfo && flinfo->fn_expr && IsA(flinfo->fn_expr, Const))
+ - ]
2050 : : {
2051 : 2763639 : Const *expr = (Const *) flinfo->fn_expr;
2052 : :
2053 [ + - ]: 2763639 : if (expr->consttype == BYTEAOID)
2054 : 2763639 : return !expr->constisnull;
2055 : : }
2252 akorotkov@postgresql 2056 :UBC 0 : return false;
2057 : : }
2058 : :
2059 : : /*
2060 : : * Get options for opclass support function.
2061 : : */
2062 : : bytea *
2252 akorotkov@postgresql 2063 :CBC 2895498 : get_fn_opclass_options(FmgrInfo *flinfo)
2064 : : {
2065 [ + - + - : 2895498 : if (flinfo && flinfo->fn_expr && IsA(flinfo->fn_expr, Const))
+ - ]
2066 : : {
2067 : 2895498 : Const *expr = (Const *) flinfo->fn_expr;
2068 : :
2069 [ + - ]: 2895498 : if (expr->consttype == BYTEAOID)
2070 [ + - ]: 2895498 : return expr->constisnull ? NULL : DatumGetByteaP(expr->constvalue);
2071 : : }
2072 : :
2252 akorotkov@postgresql 2073 [ # # ]:UBC 0 : ereport(ERROR,
2074 : : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
2075 : : errmsg("operator class options info is absent in function call context")));
2076 : :
2077 : : return NULL;
2078 : : }
2079 : :
2080 : : /*-------------------------------------------------------------------------
2081 : : * Support routines for procedural language implementations
2082 : : *-------------------------------------------------------------------------
2083 : : */
2084 : :
2085 : : /*
2086 : : * Verify that a validator is actually associated with the language of a
2087 : : * particular function and that the user has access to both the language and
2088 : : * the function. All validators should call this before doing anything
2089 : : * substantial. Doing so ensures a user cannot achieve anything with explicit
2090 : : * calls to validators that he could not achieve with CREATE FUNCTION or by
2091 : : * simply calling an existing function.
2092 : : *
2093 : : * When this function returns false, callers should skip all validation work
2094 : : * and call PG_RETURN_VOID(). This never happens at present; it is reserved
2095 : : * for future expansion.
2096 : : *
2097 : : * In particular, checking that the validator corresponds to the function's
2098 : : * language allows untrusted language validators to assume they process only
2099 : : * superuser-chosen source code. (Untrusted language call handlers, by
2100 : : * definition, do assume that.) A user lacking the USAGE language privilege
2101 : : * would be unable to reach the validator through CREATE FUNCTION, so we check
2102 : : * that to block explicit calls as well. Checking the EXECUTE privilege on
2103 : : * the function is often superfluous, because most users can clone the
2104 : : * function to get an executable copy. It is meaningful against users with no
2105 : : * database TEMP right and no permanent schema CREATE right, thereby unable to
2106 : : * create any function. Also, if the function tracks persistent state by
2107 : : * function OID or name, validating the original function might permit more
2108 : : * mischief than creating and validating a clone thereof.
2109 : : */
2110 : : bool
4485 noah@leadboat.com 2111 :CBC 13414 : CheckFunctionValidatorAccess(Oid validatorOid, Oid functionOid)
2112 : : {
2113 : : HeapTuple procTup;
2114 : : HeapTuple langTup;
2115 : : Form_pg_proc procStruct;
2116 : : Form_pg_language langStruct;
2117 : : AclResult aclresult;
2118 : :
2119 : : /*
2120 : : * Get the function's pg_proc entry. Throw a user-facing error for bad
2121 : : * OID, because validators can be called with user-specified OIDs.
2122 : : */
2123 : 13414 : procTup = SearchSysCache1(PROCOID, ObjectIdGetDatum(functionOid));
2124 [ - + ]: 13414 : if (!HeapTupleIsValid(procTup))
3550 tgl@sss.pgh.pa.us 2125 [ # # ]:UBC 0 : ereport(ERROR,
2126 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
2127 : : errmsg("function with OID %u does not exist", functionOid)));
4485 noah@leadboat.com 2128 :CBC 13414 : procStruct = (Form_pg_proc) GETSTRUCT(procTup);
2129 : :
2130 : : /*
2131 : : * Fetch pg_language entry to know if this is the correct validation
2132 : : * function for that pg_proc entry.
2133 : : */
2134 : 13414 : langTup = SearchSysCache1(LANGOID, ObjectIdGetDatum(procStruct->prolang));
2135 [ - + ]: 13414 : if (!HeapTupleIsValid(langTup))
4485 noah@leadboat.com 2136 [ # # ]:UBC 0 : elog(ERROR, "cache lookup failed for language %u", procStruct->prolang);
4485 noah@leadboat.com 2137 :CBC 13414 : langStruct = (Form_pg_language) GETSTRUCT(langTup);
2138 : :
2139 [ - + ]: 13414 : if (langStruct->lanvalidator != validatorOid)
4485 noah@leadboat.com 2140 [ # # ]:UBC 0 : ereport(ERROR,
2141 : : (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
2142 : : errmsg("language validation function %u called for language %u instead of %u",
2143 : : validatorOid, procStruct->prolang,
2144 : : langStruct->lanvalidator)));
2145 : :
2146 : : /* first validate that we have permissions to use the language */
1294 peter@eisentraut.org 2147 :CBC 13414 : aclresult = object_aclcheck(LanguageRelationId, procStruct->prolang, GetUserId(),
2148 : : ACL_USAGE);
4485 noah@leadboat.com 2149 [ - + ]: 13414 : if (aclresult != ACLCHECK_OK)
3101 peter_e@gmx.net 2150 :UBC 0 : aclcheck_error(aclresult, OBJECT_LANGUAGE,
4485 noah@leadboat.com 2151 : 0 : NameStr(langStruct->lanname));
2152 : :
2153 : : /*
2154 : : * Check whether we are allowed to execute the function itself. If we can
2155 : : * execute it, there should be no possible side-effect of
2156 : : * compiling/validation that execution can't have.
2157 : : */
1294 peter@eisentraut.org 2158 :CBC 13414 : aclresult = object_aclcheck(ProcedureRelationId, functionOid, GetUserId(), ACL_EXECUTE);
4485 noah@leadboat.com 2159 [ - + ]: 13414 : if (aclresult != ACLCHECK_OK)
3101 peter_e@gmx.net 2160 :UBC 0 : aclcheck_error(aclresult, OBJECT_FUNCTION, NameStr(procStruct->proname));
2161 : :
4485 noah@leadboat.com 2162 :CBC 13414 : ReleaseSysCache(procTup);
2163 : 13414 : ReleaseSysCache(langTup);
2164 : :
2165 : 13414 : return true;
2166 : : }
|