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