summaryrefslogtreecommitdiff
path: root/src/backend/utils/fmgr/fmgr.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/backend/utils/fmgr/fmgr.c')
-rw-r--r--src/backend/utils/fmgr/fmgr.c1589
1 files changed, 0 insertions, 1589 deletions
diff --git a/src/backend/utils/fmgr/fmgr.c b/src/backend/utils/fmgr/fmgr.c
deleted file mode 100644
index 1b2e0d9ef26..00000000000
--- a/src/backend/utils/fmgr/fmgr.c
+++ /dev/null
@@ -1,1589 +0,0 @@
-/*-------------------------------------------------------------------------
- *
- * fmgr.c
- * The Postgres function manager.
- *
- * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
- * Portions Copyright (c) 1994, Regents of the University of California
- *
- *
- * IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/fmgr/fmgr.c,v 1.60 2002/06/20 20:29:39 momjian Exp $
- *
- *-------------------------------------------------------------------------
- */
-
-#include "postgres.h"
-
-#include "access/tuptoaster.h"
-#include "catalog/pg_language.h"
-#include "catalog/pg_proc.h"
-#include "executor/functions.h"
-#include "miscadmin.h"
-#include "utils/builtins.h"
-#include "utils/fmgrtab.h"
-#include "utils/lsyscache.h"
-#include "utils/syscache.h"
-
-/*
- * Declaration for old-style function pointer type. This is now used only
- * in fmgr_oldstyle() and is no longer exported.
- *
- * The m68k SVR4 ABI defines that pointers are returned in %a0 instead of
- * %d0. So if a function pointer is declared to return a pointer, the
- * compiler may look only into %a0, but if the called function was declared
- * to return an integer type, it puts its value only into %d0. So the
- * caller doesn't pink up the correct return value. The solution is to
- * declare the function pointer to return int, so the compiler picks up the
- * return value from %d0. (Functions returning pointers put their value
- * *additionally* into %d0 for compatibility.) The price is that there are
- * some warnings about int->pointer conversions...
- */
-#if defined(__mc68000__) && defined(__ELF__)
-typedef int32 ((*func_ptr) ());
-
-#else
-typedef char *((*func_ptr) ());
-#endif
-
-/*
- * For an oldstyle function, fn_extra points to a record like this:
- */
-typedef struct
-{
- func_ptr func; /* Address of the oldstyle function */
- bool arg_toastable[FUNC_MAX_ARGS]; /* is n'th arg of a
- * toastable datatype? */
-} Oldstyle_fnextra;
-
-
-static void fmgr_info_cxt_security(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt,
- bool ignore_security);
-static void fmgr_info_C_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple);
-static void fmgr_info_other_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple);
-static Datum fmgr_oldstyle(PG_FUNCTION_ARGS);
-static Datum fmgr_security_definer(PG_FUNCTION_ARGS);
-
-
-/*
- * Lookup routines for builtin-function table. We can search by either Oid
- * or name, but search by Oid is much faster.
- */
-
-static const FmgrBuiltin *
-fmgr_isbuiltin(Oid id)
-{
- int low = 0;
- int high = fmgr_nbuiltins - 1;
-
- /*
- * Loop invariant: low is the first index that could contain target
- * entry, and high is the last index that could contain it.
- */
- while (low <= high)
- {
- int i = (high + low) / 2;
- const FmgrBuiltin *ptr = &fmgr_builtins[i];
-
- if (id == ptr->foid)
- return ptr;
- else if (id > ptr->foid)
- low = i + 1;
- else
- high = i - 1;
- }
- return (const FmgrBuiltin *) NULL;
-}
-
-/*
- * Lookup a builtin by name. Note there can be more than one entry in
- * the array with the same name, but they should all point to the same
- * routine.
- */
-static const FmgrBuiltin *
-fmgr_lookupByName(const char *name)
-{
- int i;
-
- for (i = 0; i < fmgr_nbuiltins; i++)
- {
- if (strcmp(name, fmgr_builtins[i].funcName) == 0)
- return fmgr_builtins + i;
- }
- return (const FmgrBuiltin *) NULL;
-}
-
-/*
- * This routine fills a FmgrInfo struct, given the OID
- * of the function to be called.
- *
- * The caller's CurrentMemoryContext is used as the fn_mcxt of the info
- * struct; this means that any subsidiary data attached to the info struct
- * (either by fmgr_info itself, or later on by a function call handler)
- * will be allocated in that context. The caller must ensure that this
- * context is at least as long-lived as the info struct itself. This is
- * not a problem in typical cases where the info struct is on the stack or
- * in freshly-palloc'd space. However, if one intends to store an info
- * struct in a long-lived table, it's better to use fmgr_info_cxt.
- */
-void
-fmgr_info(Oid functionId, FmgrInfo *finfo)
-{
- fmgr_info_cxt(functionId, finfo, CurrentMemoryContext);
-}
-
-/*
- * Fill a FmgrInfo struct, specifying a memory context in which its
- * subsidiary data should go.
- */
-void
-fmgr_info_cxt(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt)
-{
- fmgr_info_cxt_security(functionId, finfo, mcxt, false);
-}
-
-/*
- * This one does the actual work. ignore_security is ordinarily false
- * but is set to true by fmgr_security_definer to avoid infinite
- * recursive lookups.
- */
-static void
-fmgr_info_cxt_security(Oid functionId, FmgrInfo *finfo, MemoryContext mcxt,
- bool ignore_security)
-{
- const FmgrBuiltin *fbp;
- HeapTuple procedureTuple;
- Form_pg_proc procedureStruct;
- char *prosrc;
-
- /*
- * fn_oid *must* be filled in last. Some code assumes that if fn_oid
- * is valid, the whole struct is valid. Some FmgrInfo struct's do
- * survive elogs.
- */
- finfo->fn_oid = InvalidOid;
- finfo->fn_extra = NULL;
- finfo->fn_mcxt = mcxt;
-
- if ((fbp = fmgr_isbuiltin(functionId)) != NULL)
- {
- /*
- * Fast path for builtin functions: don't bother consulting
- * pg_proc
- */
- finfo->fn_nargs = fbp->nargs;
- finfo->fn_strict = fbp->strict;
- finfo->fn_retset = fbp->retset;
- finfo->fn_addr = fbp->func;
- finfo->fn_oid = functionId;
- return;
- }
-
- /* Otherwise we need the pg_proc entry */
- procedureTuple = SearchSysCache(PROCOID,
- ObjectIdGetDatum(functionId),
- 0, 0, 0);
- if (!HeapTupleIsValid(procedureTuple))
- elog(ERROR, "fmgr_info: function %u: cache lookup failed",
- functionId);
- procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
-
- finfo->fn_nargs = procedureStruct->pronargs;
- finfo->fn_strict = procedureStruct->proisstrict;
- finfo->fn_retset = procedureStruct->proretset;
-
- if (procedureStruct->prosecdef && !ignore_security)
- {
- finfo->fn_addr = fmgr_security_definer;
- finfo->fn_oid = functionId;
- ReleaseSysCache(procedureTuple);
- return;
- }
-
- switch (procedureStruct->prolang)
- {
- case INTERNALlanguageId:
-
- /*
- * For an ordinary builtin function, we should never get here
- * because the isbuiltin() search above will have succeeded.
- * However, if the user has done a CREATE FUNCTION to create
- * an alias for a builtin function, we can end up here. In
- * that case we have to look up the function by name. The
- * name of the internal function is stored in prosrc (it
- * doesn't have to be the same as the name of the alias!)
- */
- prosrc = DatumGetCString(DirectFunctionCall1(textout,
- PointerGetDatum(&procedureStruct->prosrc)));
- fbp = fmgr_lookupByName(prosrc);
- if (fbp == NULL)
- elog(ERROR, "fmgr_info: function %s not in internal table",
- prosrc);
- pfree(prosrc);
- /* Should we check that nargs, strict, retset match the table? */
- finfo->fn_addr = fbp->func;
- break;
-
- case ClanguageId:
- fmgr_info_C_lang(functionId, finfo, procedureTuple);
- break;
-
- case SQLlanguageId:
- finfo->fn_addr = fmgr_sql;
- break;
-
- default:
- fmgr_info_other_lang(functionId, finfo, procedureTuple);
- break;
- }
-
- finfo->fn_oid = functionId;
- ReleaseSysCache(procedureTuple);
-}
-
-/*
- * Special fmgr_info processing for C-language functions. Note that
- * finfo->fn_oid is not valid yet.
- */
-static void
-fmgr_info_C_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple)
-{
- Form_pg_proc procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
- Datum prosrcattr,
- probinattr;
- char *prosrcstring,
- *probinstring;
- void *libraryhandle;
- PGFunction user_fn;
- Pg_finfo_record *inforec;
- Oldstyle_fnextra *fnextra;
- bool isnull;
- int i;
-
- /* Get prosrc and probin strings (link symbol and library filename) */
- prosrcattr = SysCacheGetAttr(PROCOID, procedureTuple,
- Anum_pg_proc_prosrc, &isnull);
- if (isnull)
- elog(ERROR, "fmgr: Could not extract prosrc for %u from pg_proc",
- functionId);
- prosrcstring = DatumGetCString(DirectFunctionCall1(textout, prosrcattr));
-
- probinattr = SysCacheGetAttr(PROCOID, procedureTuple,
- Anum_pg_proc_probin, &isnull);
- if (isnull)
- elog(ERROR, "fmgr: Could not extract probin for %u from pg_proc",
- functionId);
- probinstring = DatumGetCString(DirectFunctionCall1(textout, probinattr));
-
- /* Look up the function itself */
- user_fn = load_external_function(probinstring, prosrcstring, true,
- &libraryhandle);
-
- /* Get the function information record (real or default) */
- inforec = fetch_finfo_record(libraryhandle, prosrcstring);
-
- switch (inforec->api_version)
- {
- case 0:
- /* Old style: need to use a handler */
- finfo->fn_addr = fmgr_oldstyle;
- fnextra = (Oldstyle_fnextra *)
- MemoryContextAlloc(finfo->fn_mcxt, sizeof(Oldstyle_fnextra));
- finfo->fn_extra = (void *) fnextra;
- MemSet(fnextra, 0, sizeof(Oldstyle_fnextra));
- fnextra->func = (func_ptr) user_fn;
- for (i = 0; i < procedureStruct->pronargs; i++)
- {
- fnextra->arg_toastable[i] =
- TypeIsToastable(procedureStruct->proargtypes[i]);
- }
- break;
- case 1:
- /* New style: call directly */
- finfo->fn_addr = user_fn;
- break;
- default:
- /* Shouldn't get here if fetch_finfo_record did its job */
- elog(ERROR, "Unknown function API version %d",
- inforec->api_version);
- break;
- }
-
- pfree(prosrcstring);
- pfree(probinstring);
-}
-
-/*
- * Special fmgr_info processing for other-language functions. Note
- * that finfo->fn_oid is not valid yet.
- */
-static void
-fmgr_info_other_lang(Oid functionId, FmgrInfo *finfo, HeapTuple procedureTuple)
-{
- Form_pg_proc procedureStruct = (Form_pg_proc) GETSTRUCT(procedureTuple);
- Oid language = procedureStruct->prolang;
- HeapTuple languageTuple;
- Form_pg_language languageStruct;
-
- languageTuple = SearchSysCache(LANGOID,
- ObjectIdGetDatum(language),
- 0, 0, 0);
- if (!HeapTupleIsValid(languageTuple))
- elog(ERROR, "fmgr_info: cache lookup for language %u failed",
- language);
- languageStruct = (Form_pg_language) GETSTRUCT(languageTuple);
- if (languageStruct->lanispl)
- {
- FmgrInfo plfinfo;
-
- fmgr_info(languageStruct->lanplcallfoid, &plfinfo);
- finfo->fn_addr = plfinfo.fn_addr;
-
- /*
- * If lookup of the PL handler function produced nonnull fn_extra,
- * complain --- it must be an oldstyle function! We no longer
- * support oldstyle PL handlers.
- */
- if (plfinfo.fn_extra != NULL)
- elog(ERROR, "fmgr_info: language %u has old-style handler",
- language);
- }
- else
- {
- elog(ERROR, "fmgr_info: function %u: unsupported language %u",
- functionId, language);
- }
- ReleaseSysCache(languageTuple);
-}
-
-/*
- * Fetch and validate the information record for the given external function.
- * The function is specified by a handle for the containing library
- * (obtained from load_external_function) as well as the function name.
- *
- * If no info function exists for the given name, it is not an error.
- * Instead we return a default info record for a version-0 function.
- * We want to raise an error here only if the info function returns
- * something bogus.
- *
- * This function is broken out of fmgr_info_C_lang() so that ProcedureCreate()
- * can validate the information record for a function not yet entered into
- * pg_proc.
- */
-Pg_finfo_record *
-fetch_finfo_record(void *filehandle, char *funcname)
-{
- char *infofuncname;
- PGFInfoFunction infofunc;
- Pg_finfo_record *inforec;
- static Pg_finfo_record default_inforec = {0};
-
- /* Compute name of info func */
- infofuncname = (char *) palloc(strlen(funcname) + 10);
- strcpy(infofuncname, "pg_finfo_");
- strcat(infofuncname, funcname);
-
- /* Try to look up the info function */
- infofunc = (PGFInfoFunction) lookup_external_function(filehandle,
- infofuncname);
- if (infofunc == (PGFInfoFunction) NULL)
- {
- /* Not found --- assume version 0 */
- pfree(infofuncname);
- return &default_inforec;
- }
-
- /* Found, so call it */
- inforec = (*infofunc) ();
-
- /* Validate result as best we can */
- if (inforec == NULL)
- elog(ERROR, "Null result from %s", infofuncname);
- switch (inforec->api_version)
- {
- case 0:
- case 1:
- /* OK, no additional fields to validate */
- break;
- default:
- elog(ERROR, "Unknown version %d reported by %s",
- inforec->api_version, infofuncname);
- break;
- }
-
- pfree(infofuncname);
- return inforec;
-}
-
-
-/*
- * Copy an FmgrInfo struct
- *
- * This is inherently somewhat bogus since we can't reliably duplicate
- * language-dependent subsidiary info. We cheat by zeroing fn_extra,
- * instead, meaning that subsidiary info will have to be recomputed.
- */
-void
-fmgr_info_copy(FmgrInfo *dstinfo, FmgrInfo *srcinfo,
- MemoryContext destcxt)
-{
- memcpy(dstinfo, srcinfo, sizeof(FmgrInfo));
- dstinfo->fn_mcxt = destcxt;
- if (dstinfo->fn_addr == fmgr_oldstyle)
- {
- /* For oldstyle functions we must copy fn_extra */
- Oldstyle_fnextra *fnextra;
-
- fnextra = (Oldstyle_fnextra *)
- MemoryContextAlloc(destcxt, sizeof(Oldstyle_fnextra));
- memcpy(fnextra, srcinfo->fn_extra, sizeof(Oldstyle_fnextra));
- dstinfo->fn_extra = (void *) fnextra;
- }
- else
- dstinfo->fn_extra = NULL;
-}
-
-
-/*
- * Specialized lookup routine for ProcedureCreate(): given the alleged name
- * of an internal function, return the OID of the function.
- * If the name is not recognized, return InvalidOid.
- */
-Oid
-fmgr_internal_function(const char *proname)
-{
- const FmgrBuiltin *fbp = fmgr_lookupByName(proname);
-
- if (fbp == NULL)
- return InvalidOid;
- return fbp->foid;
-}
-
-
-/*
- * Handler for old-style "C" language functions
- */
-static Datum
-fmgr_oldstyle(PG_FUNCTION_ARGS)
-{
- Oldstyle_fnextra *fnextra;
- int n_arguments = fcinfo->nargs;
- int i;
- bool isnull;
- func_ptr user_fn;
- char *returnValue;
-
- if (fcinfo->flinfo == NULL || fcinfo->flinfo->fn_extra == NULL)
- elog(ERROR, "Internal error: fmgr_oldstyle received NULL pointer");
- fnextra = (Oldstyle_fnextra *) fcinfo->flinfo->fn_extra;
-
- /*
- * Result is NULL if any argument is NULL, but we still call the
- * function (peculiar, but that's the way it worked before, and after
- * all this is a backwards-compatibility wrapper). Note, however,
- * that we'll never get here with NULL arguments if the function is
- * marked strict.
- *
- * We also need to detoast any TOAST-ed inputs, since it's unlikely that
- * an old-style function knows about TOASTing.
- */
- isnull = false;
- for (i = 0; i < n_arguments; i++)
- {
- if (PG_ARGISNULL(i))
- isnull = true;
- else if (fnextra->arg_toastable[i])
- fcinfo->arg[i] = PointerGetDatum(PG_DETOAST_DATUM(fcinfo->arg[i]));
- }
- fcinfo->isnull = isnull;
-
- user_fn = fnextra->func;
-
- switch (n_arguments)
- {
- case 0:
- returnValue = (*user_fn) ();
- break;
- case 1:
-
- /*
- * nullvalue() used to use isNull to check if arg is NULL;
- * perhaps there are other functions still out there that also
- * rely on this undocumented hack?
- */
- returnValue = (*user_fn) (fcinfo->arg[0], &fcinfo->isnull);
- break;
- case 2:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1]);
- break;
- case 3:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2]);
- break;
- case 4:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3]);
- break;
- case 5:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4]);
- break;
- case 6:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5]);
- break;
- case 7:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6]);
- break;
- case 8:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7]);
- break;
- case 9:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8]);
- break;
- case 10:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9]);
- break;
- case 11:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10]);
- break;
- case 12:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10], fcinfo->arg[11]);
- break;
- case 13:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10], fcinfo->arg[11],
- fcinfo->arg[12]);
- break;
- case 14:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10], fcinfo->arg[11],
- fcinfo->arg[12], fcinfo->arg[13]);
- break;
- case 15:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10], fcinfo->arg[11],
- fcinfo->arg[12], fcinfo->arg[13],
- fcinfo->arg[14]);
- break;
- case 16:
- returnValue = (*user_fn) (fcinfo->arg[0], fcinfo->arg[1],
- fcinfo->arg[2], fcinfo->arg[3],
- fcinfo->arg[4], fcinfo->arg[5],
- fcinfo->arg[6], fcinfo->arg[7],
- fcinfo->arg[8], fcinfo->arg[9],
- fcinfo->arg[10], fcinfo->arg[11],
- fcinfo->arg[12], fcinfo->arg[13],
- fcinfo->arg[14], fcinfo->arg[15]);
- break;
- default:
-
- /*
- * Increasing FUNC_MAX_ARGS doesn't automatically add cases to
- * the above code, so mention the actual value in this error
- * not FUNC_MAX_ARGS. You could add cases to the above if you
- * needed to support old-style functions with many arguments,
- * but making 'em be new-style is probably a better idea.
- */
- elog(ERROR, "fmgr_oldstyle: function %u: too many arguments (%d > %d)",
- fcinfo->flinfo->fn_oid, n_arguments, 16);
- returnValue = NULL; /* keep compiler quiet */
- break;
- }
-
- return (Datum) returnValue;
-}
-
-
-/*
- * Support for security definer functions
- */
-
-struct fmgr_security_definer_cache
-{
- FmgrInfo flinfo;
- Oid userid;
-};
-
-/*
- * Function handler for security definer functions. We extract the
- * OID of the actual function and do a fmgr lookup again. Then we
- * look up the owner of the function and cache both the fmgr info and
- * the owner ID. During the call we temporarily replace the flinfo
- * with the cached/looked-up one, while keeping the outer fcinfo
- * (which contains all the actual arguments, etc.) intact.
- */
-static Datum
-fmgr_security_definer(PG_FUNCTION_ARGS)
-{
- Datum result;
- FmgrInfo *save_flinfo;
- struct fmgr_security_definer_cache *fcache;
- Oid save_userid;
- HeapTuple tuple;
-
- if (!fcinfo->flinfo->fn_extra)
- {
- fcache = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt, sizeof(*fcache));
- memset(fcache, 0, sizeof(*fcache));
-
- fmgr_info_cxt_security(fcinfo->flinfo->fn_oid, &fcache->flinfo,
- fcinfo->flinfo->fn_mcxt, true);
-
- tuple = SearchSysCache(PROCOID, ObjectIdGetDatum(fcinfo->flinfo->fn_oid), 0, 0, 0);
- if (!HeapTupleIsValid(tuple))
- elog(ERROR, "fmgr_security_definer: function %u: cache lookup failed",
- fcinfo->flinfo->fn_oid);
- fcache->userid = ((Form_pg_proc) GETSTRUCT(tuple))->proowner;
- ReleaseSysCache(tuple);
-
- fcinfo->flinfo->fn_extra = fcache;
- }
- else
- fcache = fcinfo->flinfo->fn_extra;
-
- save_flinfo = fcinfo->flinfo;
- fcinfo->flinfo = &fcache->flinfo;
-
- save_userid = GetUserId();
- SetUserId(fcache->userid);
- result = FunctionCallInvoke(fcinfo);
- SetUserId(save_userid);
-
- fcinfo->flinfo = save_flinfo;
-
- return result;
-}
-
-
-/*-------------------------------------------------------------------------
- * Support routines for callers of fmgr-compatible functions
- *-------------------------------------------------------------------------
- */
-
-/* These are for invocation of a specifically named function with a
- * directly-computed parameter list. Note that neither arguments nor result
- * are allowed to be NULL. Also, the function cannot be one that needs to
- * look at FmgrInfo, since there won't be any.
- */
-Datum
-DirectFunctionCall1(PGFunction func, Datum arg1)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 1;
- fcinfo.arg[0] = arg1;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall1: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall2(PGFunction func, Datum arg1, Datum arg2)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 2;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall2: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall3(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 3;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall3: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall4(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 4;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall4: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall5(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 5;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall5: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall6(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 6;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall6: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall7(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 7;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall7: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall8(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 8;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall8: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-Datum
-DirectFunctionCall9(PGFunction func, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8,
- Datum arg9)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.nargs = 9;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
- fcinfo.arg[8] = arg9;
-
- result = (*func) (&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "DirectFunctionCall9: function %p returned NULL",
- (void *) func);
-
- return result;
-}
-
-
-/* These are for invocation of a previously-looked-up function with a
- * directly-computed parameter list. Note that neither arguments nor result
- * are allowed to be NULL.
- */
-Datum
-FunctionCall1(FmgrInfo *flinfo, Datum arg1)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 1;
- fcinfo.arg[0] = arg1;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall1: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall2(FmgrInfo *flinfo, Datum arg1, Datum arg2)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 2;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall2: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall3(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 3;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall3: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall4(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 4;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall4: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall5(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 5;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall5: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall6(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 6;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall6: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall7(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 7;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall7: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall8(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 8;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall8: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-Datum
-FunctionCall9(FmgrInfo *flinfo, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8,
- Datum arg9)
-{
- FunctionCallInfoData fcinfo;
- Datum result;
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = flinfo;
- fcinfo.nargs = 9;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
- fcinfo.arg[8] = arg9;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "FunctionCall9: function %u returned NULL",
- fcinfo.flinfo->fn_oid);
-
- return result;
-}
-
-
-/* These are for invocation of a function identified by OID with a
- * directly-computed parameter list. Note that neither arguments nor result
- * are allowed to be NULL. These are essentially fmgr_info() followed
- * by FunctionCallN(). If the same function is to be invoked repeatedly,
- * do the fmgr_info() once and then use FunctionCallN().
- */
-Datum
-OidFunctionCall1(Oid functionId, Datum arg1)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 1;
- fcinfo.arg[0] = arg1;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall1: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall2(Oid functionId, Datum arg1, Datum arg2)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 2;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall2: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall3(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 3;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall3: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall4(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 4;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall4: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall5(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 5;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall5: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall6(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 6;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall6: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall7(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 7;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall7: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall8(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 8;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall8: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-Datum
-OidFunctionCall9(Oid functionId, Datum arg1, Datum arg2,
- Datum arg3, Datum arg4, Datum arg5,
- Datum arg6, Datum arg7, Datum arg8,
- Datum arg9)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- Datum result;
-
- fmgr_info(functionId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = 9;
- fcinfo.arg[0] = arg1;
- fcinfo.arg[1] = arg2;
- fcinfo.arg[2] = arg3;
- fcinfo.arg[3] = arg4;
- fcinfo.arg[4] = arg5;
- fcinfo.arg[5] = arg6;
- fcinfo.arg[6] = arg7;
- fcinfo.arg[7] = arg8;
- fcinfo.arg[8] = arg9;
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "OidFunctionCall9: function %u returned NULL",
- flinfo.fn_oid);
-
- return result;
-}
-
-
-/*
- * !!! OLD INTERFACE !!!
- *
- * fmgr() is the only remaining vestige of the old-style caller support
- * functions. It's no longer used anywhere in the Postgres distribution,
- * but we should leave it around for a release or two to ease the transition
- * for user-supplied C functions. OidFunctionCallN() replaces it for new
- * code.
- *
- * DEPRECATED, DO NOT USE IN NEW CODE
- */
-char *
-fmgr(Oid procedureId,...)
-{
- FmgrInfo flinfo;
- FunctionCallInfoData fcinfo;
- int n_arguments;
- Datum result;
-
- fmgr_info(procedureId, &flinfo);
-
- MemSet(&fcinfo, 0, sizeof(fcinfo));
- fcinfo.flinfo = &flinfo;
- fcinfo.nargs = flinfo.fn_nargs;
- n_arguments = fcinfo.nargs;
-
- if (n_arguments > 0)
- {
- va_list pvar;
- int i;
-
- if (n_arguments > FUNC_MAX_ARGS)
- elog(ERROR, "fmgr: function %u: too many arguments (%d > %d)",
- flinfo.fn_oid, n_arguments, FUNC_MAX_ARGS);
- va_start(pvar, procedureId);
- for (i = 0; i < n_arguments; i++)
- fcinfo.arg[i] = (Datum) va_arg(pvar, char *);
- va_end(pvar);
- }
-
- result = FunctionCallInvoke(&fcinfo);
-
- /* Check for null result, since caller is clearly not expecting one */
- if (fcinfo.isnull)
- elog(ERROR, "fmgr: function %u returned NULL",
- flinfo.fn_oid);
-
- return (char *) result;
-}
-
-
-/*-------------------------------------------------------------------------
- * Support routines for standard pass-by-reference datatypes
- *
- * Note: at some point, at least on some platforms, these might become
- * pass-by-value types. Obviously Datum must be >= 8 bytes to allow
- * int64 or float8 to be pass-by-value. I think that Float4GetDatum
- * and Float8GetDatum will need to be out-of-line routines anyway,
- * since just casting from float to Datum will not do the right thing;
- * some kind of trick with pointer-casting or a union will be needed.
- *-------------------------------------------------------------------------
- */
-
-Datum
-Int64GetDatum(int64 X)
-{
-#ifndef INT64_IS_BUSTED
- int64 *retval = (int64 *) palloc(sizeof(int64));
-
- *retval = X;
- return PointerGetDatum(retval);
-#else /* INT64_IS_BUSTED */
-
- /*
- * On a machine with no 64-bit-int C datatype, sizeof(int64) will not
- * be 8, but we want Int64GetDatum to return an 8-byte object anyway,
- * with zeroes in the unused bits. This is needed so that, for
- * example, hash join of int8 will behave properly.
- */
- int64 *retval = (int64 *) palloc(Max(sizeof(int64), 8));
-
- MemSet(retval, 0, Max(sizeof(int64), 8));
- *retval = X;
- return PointerGetDatum(retval);
-#endif /* INT64_IS_BUSTED */
-}
-
-Datum
-Float4GetDatum(float4 X)
-{
- float4 *retval = (float4 *) palloc(sizeof(float4));
-
- *retval = X;
- return PointerGetDatum(retval);
-}
-
-Datum
-Float8GetDatum(float8 X)
-{
- float8 *retval = (float8 *) palloc(sizeof(float8));
-
- *retval = X;
- return PointerGetDatum(retval);
-}
-
-/*-------------------------------------------------------------------------
- * Support routines for toastable datatypes
- *-------------------------------------------------------------------------
- */
-
-struct varlena *
-pg_detoast_datum(struct varlena * datum)
-{
- if (VARATT_IS_EXTENDED(datum))
- return (struct varlena *) heap_tuple_untoast_attr((varattrib *) datum);
- else
- return datum;
-}
-
-struct varlena *
-pg_detoast_datum_copy(struct varlena * datum)
-{
- if (VARATT_IS_EXTENDED(datum))
- return (struct varlena *) heap_tuple_untoast_attr((varattrib *) datum);
- else
- {
- /* Make a modifiable copy of the varlena object */
- Size len = VARSIZE(datum);
- struct varlena *result = (struct varlena *) palloc(len);
-
- memcpy(result, datum, len);
- return result;
- }
-}
-
-struct varlena *
-pg_detoast_datum_slice(struct varlena * datum, int32 first, int32 count)
-{
- /* Only get the specified portion from the toast rel */
- return (struct varlena *) heap_tuple_untoast_attr_slice((varattrib *) datum, first, count);
-}