1996-07-09 06:22:35 +00:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
1999-02-13 23:22:53 +00:00
|
|
|
* define.c
|
1996-10-31 09:08:10 +00:00
|
|
|
*
|
1997-09-07 05:04:48 +00:00
|
|
|
* These routines execute some of the CREATE statements. In an earlier
|
|
|
|
* version of Postgres, these were "define" statements.
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
2001-01-24 19:43:33 +00:00
|
|
|
* Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
|
2000-01-26 05:58:53 +00:00
|
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* IDENTIFICATION
|
2002-03-20 19:45:13 +00:00
|
|
|
* $Header: /cvsroot/pgsql/src/backend/commands/define.c,v 1.71 2002/03/20 19:43:44 tgl Exp $
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
* DESCRIPTION
|
1997-09-07 05:04:48 +00:00
|
|
|
* The "DefineFoo" routines take the parse tree and pick out the
|
|
|
|
* appropriate arguments/flags, passing the results to the
|
|
|
|
* corresponding "FooDefine" routines (in src/catalog) that do
|
|
|
|
* the actual catalog-munging. These routines also verify permission
|
|
|
|
* of the user to execute the command.
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
* NOTES
|
1997-09-07 05:04:48 +00:00
|
|
|
* These things must be defined and committed in the following order:
|
|
|
|
* "create function":
|
|
|
|
* input/output, recv/send procedures
|
|
|
|
* "create type":
|
|
|
|
* type
|
|
|
|
* "create operator":
|
|
|
|
* operators
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
1997-09-07 05:04:48 +00:00
|
|
|
* Most of the parse-tree manipulation routines are defined in
|
|
|
|
* commands/manip.c.
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
2001-06-13 21:44:41 +00:00
|
|
|
#include "postgres.h"
|
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
#include <ctype.h>
|
1996-11-06 08:21:43 +00:00
|
|
|
#include <math.h>
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1999-07-15 23:04:24 +00:00
|
|
|
#include "access/heapam.h"
|
|
|
|
#include "catalog/catname.h"
|
2002-03-19 02:18:25 +00:00
|
|
|
#include "catalog/heap.h"
|
1999-07-15 23:04:24 +00:00
|
|
|
#include "catalog/pg_aggregate.h"
|
1999-07-16 05:00:38 +00:00
|
|
|
#include "catalog/pg_language.h"
|
1999-07-15 23:04:24 +00:00
|
|
|
#include "catalog/pg_operator.h"
|
|
|
|
#include "catalog/pg_proc.h"
|
|
|
|
#include "catalog/pg_type.h"
|
|
|
|
#include "commands/defrem.h"
|
1999-07-16 05:00:38 +00:00
|
|
|
#include "fmgr.h"
|
2001-06-13 21:44:41 +00:00
|
|
|
#include "miscadmin.h"
|
1999-07-15 23:04:24 +00:00
|
|
|
#include "optimizer/cost.h"
|
2000-10-07 00:58:23 +00:00
|
|
|
#include "parser/parse_expr.h"
|
2002-02-18 23:11:58 +00:00
|
|
|
#include "utils/acl.h"
|
1999-07-16 05:00:38 +00:00
|
|
|
#include "utils/builtins.h"
|
|
|
|
#include "utils/syscache.h"
|
1996-11-03 23:57:43 +00:00
|
|
|
|
1997-09-08 21:56:23 +00:00
|
|
|
static char *defGetString(DefElem *def);
|
1999-10-02 21:33:33 +00:00
|
|
|
static double defGetNumeric(DefElem *def);
|
1997-09-08 21:56:23 +00:00
|
|
|
static int defGetTypeLength(DefElem *def);
|
1996-07-09 06:22:35 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
#define DEFAULT_TYPDELIM ','
|
1996-10-31 09:08:10 +00:00
|
|
|
|
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
/*
|
|
|
|
* Translate the input language name to lower case.
|
|
|
|
*/
|
1996-11-10 03:06:38 +00:00
|
|
|
static void
|
1997-09-07 05:04:48 +00:00
|
|
|
case_translate_language_name(const char *input, char *output)
|
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
int i;
|
1996-10-31 09:08:10 +00:00
|
|
|
|
2001-03-22 04:01:46 +00:00
|
|
|
for (i = 0; i < NAMEDATALEN - 1 && input[i]; ++i)
|
2000-12-03 20:45:40 +00:00
|
|
|
output[i] = tolower((unsigned char) input[i]);
|
1996-10-31 09:08:10 +00:00
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
output[i] = '\0';
|
|
|
|
}
|
1996-10-31 09:08:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
1996-11-10 03:06:38 +00:00
|
|
|
static void
|
2000-10-07 00:58:23 +00:00
|
|
|
compute_return_type(TypeName *returnType,
|
1997-09-08 21:56:23 +00:00
|
|
|
char **prorettype_p, bool *returnsSet_p)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
2001-06-21 18:25:54 +00:00
|
|
|
/*
|
2001-10-25 05:50:21 +00:00
|
|
|
* Examine the "returns" clause returnType of the CREATE FUNCTION statement
|
|
|
|
* and return information about it as *prorettype_p and *returnsSet.
|
2001-06-21 18:25:54 +00:00
|
|
|
*/
|
2000-10-07 00:58:23 +00:00
|
|
|
*prorettype_p = TypeNameToInternalName(returnType);
|
|
|
|
*returnsSet_p = returnType->setof;
|
1996-10-31 09:08:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
static void
|
2000-05-28 17:56:29 +00:00
|
|
|
compute_full_attributes(List *parameters,
|
|
|
|
int32 *byte_pct_p, int32 *perbyte_cpu_p,
|
|
|
|
int32 *percall_cpu_p, int32 *outin_ratio_p,
|
|
|
|
bool *canCache_p, bool *isStrict_p)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
2001-06-21 18:25:54 +00:00
|
|
|
/*-------------
|
2001-10-25 05:50:21 +00:00
|
|
|
* Interpret the parameters *parameters and return their contents as
|
|
|
|
* *byte_pct_p, etc.
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* These parameters supply optional information about a function.
|
|
|
|
* All have defaults if not specified.
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* Note: currently, only two of these parameters actually do anything:
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* * canCache means the optimizer's constant-folder is allowed to
|
|
|
|
* pre-evaluate the function when all its inputs are constants.
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* * isStrict means the function should not be called when any NULL
|
|
|
|
* inputs are present; instead a NULL result value should be assumed.
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* The other four parameters are not used anywhere. They used to be
|
|
|
|
* used in the "expensive functions" optimizer, but that's been dead code
|
|
|
|
* for a long time.
|
2001-06-21 18:25:54 +00:00
|
|
|
*
|
2001-10-25 05:50:21 +00:00
|
|
|
* Since canCache and isStrict are useful for any function, we now allow
|
|
|
|
* attributes to be supplied for all functions regardless of language.
|
2001-06-21 18:25:54 +00:00
|
|
|
*------------
|
|
|
|
*/
|
1997-09-08 02:41:22 +00:00
|
|
|
List *pl;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/* the defaults */
|
|
|
|
*byte_pct_p = BYTE_PCT;
|
|
|
|
*perbyte_cpu_p = PERBYTE_CPU;
|
|
|
|
*percall_cpu_p = PERCALL_CPU;
|
|
|
|
*outin_ratio_p = OUTIN_RATIO;
|
1999-10-02 21:33:33 +00:00
|
|
|
*canCache_p = false;
|
2000-05-28 17:56:29 +00:00
|
|
|
*isStrict_p = false;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-10-02 21:33:33 +00:00
|
|
|
foreach(pl, parameters)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
2000-04-12 17:17:23 +00:00
|
|
|
DefElem *param = (DefElem *) lfirst(pl);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-10-02 21:33:33 +00:00
|
|
|
if (strcasecmp(param->defname, "iscachable") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
*canCache_p = true;
|
2000-05-28 17:56:29 +00:00
|
|
|
else if (strcasecmp(param->defname, "isstrict") == 0)
|
|
|
|
*isStrict_p = true;
|
1999-10-02 21:33:33 +00:00
|
|
|
else if (strcasecmp(param->defname, "trusted") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* we don't have untrusted functions any more. The 4.2
|
|
|
|
* implementation is lousy anyway so I took it out. -ay 10/94
|
|
|
|
*/
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "untrusted function has been decommissioned.");
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1999-10-02 21:33:33 +00:00
|
|
|
else if (strcasecmp(param->defname, "byte_pct") == 0)
|
|
|
|
*byte_pct_p = (int) defGetNumeric(param);
|
|
|
|
else if (strcasecmp(param->defname, "perbyte_cpu") == 0)
|
|
|
|
*perbyte_cpu_p = (int) defGetNumeric(param);
|
|
|
|
else if (strcasecmp(param->defname, "percall_cpu") == 0)
|
|
|
|
*percall_cpu_p = (int) defGetNumeric(param);
|
|
|
|
else if (strcasecmp(param->defname, "outin_ratio") == 0)
|
|
|
|
*outin_ratio_p = (int) defGetNumeric(param);
|
|
|
|
else
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(WARNING, "Unrecognized function attribute '%s' ignored",
|
1999-10-02 21:33:33 +00:00
|
|
|
param->defname);
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1996-10-31 09:08:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
/*
|
|
|
|
* For a dynamically linked C language object, the form of the clause is
|
|
|
|
*
|
|
|
|
* AS <object file name> [, <link symbol name> ]
|
|
|
|
*
|
|
|
|
* In all other cases
|
|
|
|
*
|
|
|
|
* AS <object reference, or sql code>
|
|
|
|
*
|
|
|
|
*/
|
1996-10-31 09:08:10 +00:00
|
|
|
|
1996-11-10 03:06:38 +00:00
|
|
|
static void
|
2002-02-18 23:11:58 +00:00
|
|
|
interpret_AS_clause(Oid languageOid, const char *languageName, const List *as,
|
1997-09-07 05:04:48 +00:00
|
|
|
char **prosrc_str_p, char **probin_str_p)
|
|
|
|
{
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
Assert(as != NIL);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
if (languageOid == ClanguageId)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
/*
|
|
|
|
* For "C" language, store the file name in probin and, when
|
2000-05-28 17:56:29 +00:00
|
|
|
* given, the link symbol name in prosrc.
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
*/
|
|
|
|
*probin_str_p = strVal(lfirst(as));
|
|
|
|
if (lnext(as) == NULL)
|
|
|
|
*prosrc_str_p = "-";
|
|
|
|
else
|
|
|
|
*prosrc_str_p = strVal(lsecond(as));
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
/* Everything else wants the given string in prosrc. */
|
|
|
|
*prosrc_str_p = strVal(lfirst(as));
|
1997-09-07 05:04:48 +00:00
|
|
|
*probin_str_p = "-";
|
I have been working with user defined types and user defined c
functions. One problem that I have encountered with the function
manager is that it does not allow the user to define type conversion
functions that convert between user types. For instance if mytype1,
mytype2, and mytype3 are three Postgresql user types, and if I wish to
define Postgresql conversion functions like
I run into problems, because the Postgresql dynamic loader would look
for a single link symbol, mytype3, for both pieces of object code. If
I just change the name of one of the Postgresql functions (to make the
symbols distinct), the automatic type conversion that Postgresql uses,
for example, when matching operators to arguments no longer finds the
type conversion function.
The solution that I propose, and have implemented in the attatched
patch extends the CREATE FUNCTION syntax as follows. In the first case
above I use the link symbol mytype2_to_mytype3 for the link object
that implements the first conversion function, and define the
Postgresql operator with the following syntax
The patch includes changes to the parser to include the altered
syntax, changes to the ProcedureStmt node in nodes/parsenodes.h,
changes to commands/define.c to handle the extra information in the AS
clause, and changes to utils/fmgr/dfmgr.c that alter the way that the
dynamic loader figures out what link symbol to use. I store the
string for the link symbol in the prosrc text attribute of the pg_proc
table which is currently unused in rows that reference dynamically
loaded
functions.
Bernie Frankpitt
1999-09-28 04:34:56 +00:00
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
if (lnext(as) != NIL)
|
1999-10-02 21:33:33 +00:00
|
|
|
elog(ERROR, "CREATE FUNCTION: only one AS item needed for %s language",
|
|
|
|
languageName);
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
1996-10-31 09:08:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
1999-05-25 16:15:34 +00:00
|
|
|
* CreateFunction
|
1997-09-07 05:04:48 +00:00
|
|
|
* Execute a CREATE FUNCTION utility statement.
|
1996-10-31 09:08:10 +00:00
|
|
|
*/
|
|
|
|
void
|
2001-09-08 01:10:21 +00:00
|
|
|
CreateFunction(ProcedureStmt *stmt)
|
1996-10-31 09:08:10 +00:00
|
|
|
{
|
1997-09-07 05:04:48 +00:00
|
|
|
/* pathname of executable file that executes this function, if any */
|
2002-02-18 23:11:58 +00:00
|
|
|
char *probin_str;
|
1997-09-07 05:04:48 +00:00
|
|
|
/* SQL that executes this function, if any */
|
2002-02-18 23:11:58 +00:00
|
|
|
char *prosrc_str;
|
1997-09-07 05:04:48 +00:00
|
|
|
/* Type of return value (or member of set of values) from function */
|
2002-02-18 23:11:58 +00:00
|
|
|
char *prorettype;
|
|
|
|
/* name of language of function, with case adjusted */
|
2000-10-07 00:58:23 +00:00
|
|
|
char languageName[NAMEDATALEN];
|
1999-10-02 21:33:33 +00:00
|
|
|
/* The function returns a set of values, as opposed to a singleton. */
|
2002-02-18 23:11:58 +00:00
|
|
|
bool returnsSet;
|
1997-09-07 05:04:48 +00:00
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* The following are optional user-supplied attributes of the
|
|
|
|
* function.
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
1997-09-08 02:41:22 +00:00
|
|
|
int32 byte_pct,
|
|
|
|
perbyte_cpu,
|
|
|
|
percall_cpu,
|
|
|
|
outin_ratio;
|
2000-05-28 17:56:29 +00:00
|
|
|
bool canCache,
|
|
|
|
isStrict;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
HeapTuple languageTuple;
|
|
|
|
Form_pg_language languageStruct;
|
|
|
|
Oid languageOid;
|
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
/* Convert language name to canonical case */
|
1997-09-07 05:04:48 +00:00
|
|
|
case_translate_language_name(stmt->language, languageName);
|
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
languageTuple = SearchSysCache(LANGNAME,
|
|
|
|
PointerGetDatum(languageName),
|
|
|
|
0, 0, 0);
|
|
|
|
if (!HeapTupleIsValid(languageTuple))
|
|
|
|
elog(ERROR, "language \"%s\" does not exist", languageName);
|
1997-10-28 15:11:45 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
languageOid = languageTuple->t_data->t_oid;
|
|
|
|
languageStruct = (Form_pg_language) GETSTRUCT(languageTuple);
|
1997-10-28 15:11:45 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
if (!((languageStruct->lanpltrusted
|
|
|
|
&& pg_language_aclcheck(languageOid, GetUserId()) == ACLCHECK_OK)
|
|
|
|
|| superuser()))
|
|
|
|
elog(ERROR, "permission denied");
|
2000-11-16 22:30:52 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
ReleaseSysCache(languageTuple);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
/*
|
|
|
|
* Convert remaining parameters of CREATE to form wanted by
|
|
|
|
* ProcedureCreate.
|
|
|
|
*/
|
|
|
|
Assert(IsA(stmt->returnType, TypeName));
|
|
|
|
compute_return_type((TypeName *) stmt->returnType,
|
|
|
|
&prorettype, &returnsSet);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
1999-10-02 21:33:33 +00:00
|
|
|
compute_full_attributes(stmt->withClause,
|
|
|
|
&byte_pct, &perbyte_cpu, &percall_cpu,
|
2000-05-28 17:56:29 +00:00
|
|
|
&outin_ratio, &canCache, &isStrict);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2002-02-18 23:11:58 +00:00
|
|
|
interpret_AS_clause(languageOid, languageName, stmt->as, &prosrc_str, &probin_str);
|
1999-10-02 21:33:33 +00:00
|
|
|
|
|
|
|
/*
|
2000-04-12 17:17:23 +00:00
|
|
|
* And now that we have all the parameters, and know we're permitted
|
|
|
|
* to do so, go ahead and create the function.
|
1999-10-02 21:33:33 +00:00
|
|
|
*/
|
|
|
|
ProcedureCreate(stmt->funcname,
|
2001-10-02 21:39:36 +00:00
|
|
|
stmt->replace,
|
1999-10-02 21:33:33 +00:00
|
|
|
returnsSet,
|
|
|
|
prorettype,
|
2002-02-18 23:11:58 +00:00
|
|
|
languageOid,
|
2000-04-12 17:17:23 +00:00
|
|
|
prosrc_str, /* converted to text later */
|
|
|
|
probin_str, /* converted to text later */
|
|
|
|
true, /* (obsolete "trusted") */
|
2000-05-28 17:56:29 +00:00
|
|
|
canCache,
|
|
|
|
isStrict,
|
1999-10-02 21:33:33 +00:00
|
|
|
byte_pct,
|
|
|
|
perbyte_cpu,
|
|
|
|
percall_cpu,
|
|
|
|
outin_ratio,
|
2001-09-08 01:10:21 +00:00
|
|
|
stmt->argTypes);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1996-10-31 09:08:10 +00:00
|
|
|
|
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
/* --------------------------------
|
1999-02-13 23:22:53 +00:00
|
|
|
* DefineOperator
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
1997-09-07 05:04:48 +00:00
|
|
|
* this function extracts all the information from the
|
|
|
|
* parameter list generated by the parser and then has
|
|
|
|
* OperatorCreate() do all the actual work.
|
1996-07-09 06:22:35 +00:00
|
|
|
*
|
|
|
|
* 'parameters' is a list of DefElem
|
|
|
|
* --------------------------------
|
|
|
|
*/
|
|
|
|
void
|
1997-09-07 05:04:48 +00:00
|
|
|
DefineOperator(char *oprName,
|
1997-09-08 21:56:23 +00:00
|
|
|
List *parameters)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
uint16 precedence = 0; /* operator precedence */
|
2001-10-28 06:26:15 +00:00
|
|
|
bool canHash = false; /* operator hashes */
|
1997-09-08 02:41:22 +00:00
|
|
|
bool isLeftAssociative = true; /* operator is left
|
1997-09-07 05:04:48 +00:00
|
|
|
* associative */
|
1997-09-08 02:41:22 +00:00
|
|
|
char *functionName = NULL; /* function for operator */
|
|
|
|
char *typeName1 = NULL; /* first type name */
|
|
|
|
char *typeName2 = NULL; /* second type name */
|
|
|
|
char *commutatorName = NULL; /* optional commutator operator
|
1997-09-07 05:04:48 +00:00
|
|
|
* name */
|
1997-09-08 02:41:22 +00:00
|
|
|
char *negatorName = NULL; /* optional negator operator name */
|
|
|
|
char *restrictionName = NULL; /* optional restrict. sel.
|
|
|
|
* procedure */
|
2001-10-28 06:26:15 +00:00
|
|
|
char *joinName = NULL; /* optional join sel. procedure name */
|
1997-09-08 02:41:22 +00:00
|
|
|
char *sortName1 = NULL; /* optional first sort operator */
|
|
|
|
char *sortName2 = NULL; /* optional second sort operator */
|
|
|
|
List *pl;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* loop over the definition list and extract the information we need.
|
|
|
|
*/
|
|
|
|
foreach(pl, parameters)
|
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
DefElem *defel = (DefElem *) lfirst(pl);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
if (strcasecmp(defel->defname, "leftarg") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
2000-04-07 13:40:45 +00:00
|
|
|
typeName1 = defGetString(defel);
|
2001-05-18 21:24:20 +00:00
|
|
|
if (IsA(defel->arg, TypeName) &&
|
|
|
|
((TypeName *) defel->arg)->setof)
|
2000-10-07 00:58:23 +00:00
|
|
|
elog(ERROR, "setof type not implemented for leftarg");
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "rightarg") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
2000-04-07 13:40:45 +00:00
|
|
|
typeName2 = defGetString(defel);
|
2001-05-18 21:24:20 +00:00
|
|
|
if (IsA(defel->arg, TypeName) &&
|
|
|
|
((TypeName *) defel->arg)->setof)
|
2000-10-07 00:58:23 +00:00
|
|
|
elog(ERROR, "setof type not implemented for rightarg");
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "procedure") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
functionName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "precedence") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
|
|
|
/* NOT IMPLEMENTED (never worked in v4.2) */
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(NOTICE, "CREATE OPERATOR: precedence not implemented");
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "associativity") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
|
|
|
/* NOT IMPLEMENTED (never worked in v4.2) */
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(NOTICE, "CREATE OPERATOR: associativity not implemented");
|
1997-09-07 05:04:48 +00:00
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "commutator") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
commutatorName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "negator") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
negatorName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "restrict") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
restrictionName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "join") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
joinName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "hashes") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
canHash = TRUE;
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "sort1") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
|
|
|
/* ----------------
|
|
|
|
* XXX ( ... [ , sort1 = oprname ] [ , sort2 = oprname ] ... )
|
|
|
|
* XXX is undocumented in the reference manual source as of
|
|
|
|
* 89/8/22.
|
|
|
|
* ----------------
|
|
|
|
*/
|
|
|
|
sortName1 = defGetString(defel);
|
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "sort2") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
sortName2 = defGetString(defel);
|
|
|
|
else
|
|
|
|
{
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(WARNING, "DefineOperator: attribute \"%s\" not recognized",
|
1997-09-07 05:04:48 +00:00
|
|
|
defel->defname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* make sure we have our required definitions
|
|
|
|
*/
|
|
|
|
if (functionName == NULL)
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "Define: \"procedure\" unspecified");
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2001-03-22 06:16:21 +00:00
|
|
|
/*
|
|
|
|
* now have OperatorCreate do all the work..
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
|
|
|
OperatorCreate(oprName, /* operator name */
|
|
|
|
typeName1, /* first type name */
|
|
|
|
typeName2, /* second type name */
|
2001-10-28 06:26:15 +00:00
|
|
|
functionName, /* function for operator */
|
1997-09-07 05:04:48 +00:00
|
|
|
precedence, /* operator precedence */
|
|
|
|
isLeftAssociative, /* operator is left associative */
|
|
|
|
commutatorName, /* optional commutator operator
|
|
|
|
* name */
|
|
|
|
negatorName, /* optional negator operator name */
|
|
|
|
restrictionName, /* optional restrict. sel.
|
|
|
|
* procedure */
|
|
|
|
joinName, /* optional join sel. procedure name */
|
|
|
|
canHash, /* operator hashes */
|
|
|
|
sortName1, /* optional first sort operator */
|
|
|
|
sortName2); /* optional second sort operator */
|
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* -------------------
|
1997-09-07 05:04:48 +00:00
|
|
|
* DefineAggregate
|
1996-07-09 06:22:35 +00:00
|
|
|
* ------------------
|
|
|
|
*/
|
|
|
|
void
|
1997-09-08 21:56:23 +00:00
|
|
|
DefineAggregate(char *aggName, List *parameters)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
2000-07-17 03:05:41 +00:00
|
|
|
char *transfuncName = NULL;
|
1997-09-08 02:41:22 +00:00
|
|
|
char *finalfuncName = NULL;
|
|
|
|
char *baseType = NULL;
|
2000-07-17 03:05:41 +00:00
|
|
|
char *transType = NULL;
|
|
|
|
char *initval = NULL;
|
1997-09-08 02:41:22 +00:00
|
|
|
List *pl;
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
foreach(pl, parameters)
|
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
DefElem *defel = (DefElem *) lfirst(pl);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/*
|
2001-03-22 04:01:46 +00:00
|
|
|
* sfunc1, stype1, and initcond1 are accepted as obsolete
|
|
|
|
* spellings for sfunc, stype, initcond.
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
2000-07-17 03:05:41 +00:00
|
|
|
if (strcasecmp(defel->defname, "sfunc") == 0)
|
|
|
|
transfuncName = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "sfunc1") == 0)
|
|
|
|
transfuncName = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "finalfunc") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
finalfuncName = defGetString(defel);
|
2000-07-17 03:05:41 +00:00
|
|
|
else if (strcasecmp(defel->defname, "basetype") == 0)
|
|
|
|
baseType = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "stype") == 0)
|
|
|
|
transType = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "stype1") == 0)
|
|
|
|
transType = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "initcond") == 0)
|
|
|
|
initval = defGetString(defel);
|
|
|
|
else if (strcasecmp(defel->defname, "initcond1") == 0)
|
|
|
|
initval = defGetString(defel);
|
1997-09-07 05:04:48 +00:00
|
|
|
else
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(WARNING, "DefineAggregate: attribute \"%s\" not recognized",
|
1997-09-07 05:04:48 +00:00
|
|
|
defel->defname);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* make sure we have our required definitions
|
|
|
|
*/
|
|
|
|
if (baseType == NULL)
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "Define: \"basetype\" unspecified");
|
2000-07-17 03:05:41 +00:00
|
|
|
if (transType == NULL)
|
|
|
|
elog(ERROR, "Define: \"stype\" unspecified");
|
|
|
|
if (transfuncName == NULL)
|
|
|
|
elog(ERROR, "Define: \"sfunc\" unspecified");
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Most of the argument-checking is done inside of AggregateCreate
|
|
|
|
*/
|
2001-03-22 04:01:46 +00:00
|
|
|
AggregateCreate(aggName, /* aggregate name */
|
|
|
|
transfuncName, /* step function name */
|
|
|
|
finalfuncName, /* final function name */
|
|
|
|
baseType, /* type of data being aggregated */
|
|
|
|
transType, /* transition data type */
|
|
|
|
initval); /* initial condition */
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
2002-03-19 02:18:25 +00:00
|
|
|
/*
|
|
|
|
* DefineDomain
|
|
|
|
* Registers a new domain.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
DefineDomain(CreateDomainStmt *stmt)
|
|
|
|
{
|
2002-03-20 19:45:13 +00:00
|
|
|
int16 internalLength;
|
|
|
|
int16 externalLength;
|
|
|
|
char *inputName;
|
|
|
|
char *outputName;
|
|
|
|
char *sendName;
|
|
|
|
char *receiveName;
|
|
|
|
bool byValue;
|
|
|
|
char delimiter;
|
|
|
|
char alignment;
|
|
|
|
char storage;
|
2002-03-19 02:18:25 +00:00
|
|
|
char typtype;
|
|
|
|
Datum datum;
|
2002-03-20 19:45:13 +00:00
|
|
|
bool isnull;
|
|
|
|
char *defaultValue = NULL;
|
|
|
|
char *defaultValueBin = NULL;
|
2002-03-19 02:18:25 +00:00
|
|
|
bool typNotNull = false;
|
2002-03-20 19:45:13 +00:00
|
|
|
Oid basetypelem;
|
2002-03-19 02:18:25 +00:00
|
|
|
char *elemName = NULL;
|
|
|
|
int32 typNDims = 0; /* No array dimensions by default */
|
|
|
|
HeapTuple typeTup;
|
|
|
|
char *typeName = stmt->typename->name;
|
|
|
|
List *schema = stmt->constraints;
|
2002-03-20 19:45:13 +00:00
|
|
|
List *listptr;
|
2002-03-19 02:18:25 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Domainnames, unlike typenames don't need to account for the '_'
|
|
|
|
* prefix. So they can be one character longer.
|
|
|
|
*/
|
|
|
|
if (strlen(stmt->domainname) > (NAMEDATALEN - 1))
|
|
|
|
elog(ERROR, "CREATE DOMAIN: domain names must be %d characters or less",
|
|
|
|
NAMEDATALEN - 1);
|
|
|
|
|
|
|
|
/* Test for existing Domain (or type) of that name */
|
2002-03-20 19:45:13 +00:00
|
|
|
typeTup = SearchSysCache(TYPENAME,
|
|
|
|
PointerGetDatum(stmt->domainname),
|
|
|
|
0, 0, 0);
|
2002-03-19 02:18:25 +00:00
|
|
|
if (HeapTupleIsValid(typeTup))
|
2002-03-20 19:45:13 +00:00
|
|
|
elog(ERROR, "CREATE DOMAIN: domain or type %s already exists",
|
2002-03-19 02:18:25 +00:00
|
|
|
stmt->domainname);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* When the type is an array for some reason we don't actually receive
|
|
|
|
* the name here. We receive the base types name. Lets set Dims while
|
|
|
|
* were at it.
|
|
|
|
*/
|
|
|
|
if (stmt->typename->arrayBounds > 0) {
|
|
|
|
typeName = makeArrayTypeName(stmt->typename->name);
|
|
|
|
|
|
|
|
typNDims = length(stmt->typename->arrayBounds);
|
|
|
|
}
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
typeTup = SearchSysCache(TYPENAME,
|
|
|
|
PointerGetDatum(typeName),
|
|
|
|
0, 0, 0);
|
2002-03-19 02:18:25 +00:00
|
|
|
if (!HeapTupleIsValid(typeTup))
|
|
|
|
elog(ERROR, "CREATE DOMAIN: type %s does not exist",
|
|
|
|
stmt->typename->name);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* What we really don't want is domains of domains. This could cause all sorts
|
|
|
|
* of neat issues if we allow that.
|
|
|
|
*
|
|
|
|
* With testing, we may determine complex types should be allowed
|
|
|
|
*/
|
2002-03-20 19:45:13 +00:00
|
|
|
typtype = ((Form_pg_type) GETSTRUCT(typeTup))->typtype;
|
|
|
|
if (typtype != 'b')
|
|
|
|
elog(ERROR, "DefineDomain: %s is not a basetype",
|
|
|
|
stmt->typename->name);
|
2002-03-19 02:18:25 +00:00
|
|
|
|
|
|
|
/* passed by value */
|
|
|
|
byValue = ((Form_pg_type) GETSTRUCT(typeTup))->typbyval;
|
|
|
|
|
|
|
|
/* Required Alignment */
|
|
|
|
alignment = ((Form_pg_type) GETSTRUCT(typeTup))->typalign;
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
/* TOAST Strategy */
|
|
|
|
storage = ((Form_pg_type) GETSTRUCT(typeTup))->typstorage;
|
|
|
|
|
2002-03-19 02:18:25 +00:00
|
|
|
/* Storage Length */
|
|
|
|
internalLength = ((Form_pg_type) GETSTRUCT(typeTup))->typlen;
|
|
|
|
|
|
|
|
/* External Length (unused) */
|
|
|
|
externalLength = ((Form_pg_type) GETSTRUCT(typeTup))->typprtlen;
|
|
|
|
|
|
|
|
/* Array element Delimiter */
|
|
|
|
delimiter = ((Form_pg_type) GETSTRUCT(typeTup))->typdelim;
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
/*
|
|
|
|
* XXX this is pretty bogus: should be passing function OIDs to
|
|
|
|
* TypeCreate, not names which aren't unique.
|
|
|
|
*/
|
|
|
|
|
2002-03-19 02:18:25 +00:00
|
|
|
/* Input Function Name */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup, Anum_pg_type_typinput, &isnull);
|
2002-03-19 02:18:25 +00:00
|
|
|
Assert(!isnull);
|
|
|
|
|
|
|
|
inputName = DatumGetCString(DirectFunctionCall1(regprocout, datum));
|
|
|
|
|
|
|
|
/* Output Function Name */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup, Anum_pg_type_typoutput, &isnull);
|
2002-03-19 02:18:25 +00:00
|
|
|
Assert(!isnull);
|
|
|
|
|
|
|
|
outputName = DatumGetCString(DirectFunctionCall1(regprocout, datum));
|
|
|
|
|
|
|
|
/* ReceiveName */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup, Anum_pg_type_typreceive, &isnull);
|
2002-03-19 02:18:25 +00:00
|
|
|
Assert(!isnull);
|
|
|
|
|
|
|
|
receiveName = DatumGetCString(DirectFunctionCall1(regprocout, datum));
|
|
|
|
|
|
|
|
/* SendName */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup, Anum_pg_type_typsend, &isnull);
|
2002-03-19 02:18:25 +00:00
|
|
|
Assert(!isnull);
|
|
|
|
|
|
|
|
sendName = DatumGetCString(DirectFunctionCall1(regprocout, datum));
|
|
|
|
|
|
|
|
/* Inherited default value */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup,
|
|
|
|
Anum_pg_type_typdefault, &isnull);
|
|
|
|
if (!isnull)
|
|
|
|
defaultValue = DatumGetCString(DirectFunctionCall1(textout, datum));
|
2002-03-19 02:18:25 +00:00
|
|
|
|
|
|
|
/* Inherited default binary value */
|
2002-03-20 19:45:13 +00:00
|
|
|
datum = SysCacheGetAttr(TYPENAME, typeTup,
|
|
|
|
Anum_pg_type_typdefaultbin, &isnull);
|
|
|
|
if (!isnull)
|
|
|
|
defaultValueBin = DatumGetCString(DirectFunctionCall1(textout, datum));
|
2002-03-19 02:18:25 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Pull out the typelem name of the parent OID.
|
|
|
|
*
|
|
|
|
* This is what enables us to make a domain of an array
|
|
|
|
*/
|
2002-03-20 19:45:13 +00:00
|
|
|
basetypelem = ((Form_pg_type) GETSTRUCT(typeTup))->typelem;
|
|
|
|
if (basetypelem != InvalidOid)
|
|
|
|
{
|
2002-03-19 02:18:25 +00:00
|
|
|
HeapTuple tup;
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
tup = SearchSysCache(TYPEOID,
|
|
|
|
ObjectIdGetDatum(basetypelem),
|
|
|
|
0, 0, 0);
|
|
|
|
elemName = pstrdup(NameStr(((Form_pg_type) GETSTRUCT(tup))->typname));
|
2002-03-19 02:18:25 +00:00
|
|
|
ReleaseSysCache(tup);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2002-03-20 19:45:13 +00:00
|
|
|
* Run through constraints manually to avoid the additional
|
2002-03-19 02:18:25 +00:00
|
|
|
* processing conducted by DefineRelation() and friends.
|
|
|
|
*
|
|
|
|
* Besides, we don't want any constraints to be cooked. We'll
|
|
|
|
* do that when the table is created via MergeDomainAttributes().
|
|
|
|
*/
|
|
|
|
foreach(listptr, schema)
|
|
|
|
{
|
2002-03-20 19:45:13 +00:00
|
|
|
Constraint *colDef = lfirst(listptr);
|
2002-03-19 02:18:25 +00:00
|
|
|
bool nullDefined = false;
|
|
|
|
Node *expr;
|
|
|
|
ParseState *pstate;
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
switch (colDef->contype)
|
|
|
|
{
|
2002-03-19 02:18:25 +00:00
|
|
|
/*
|
|
|
|
* The inherited default value may be overridden by the user
|
|
|
|
* with the DEFAULT <expr> statement.
|
|
|
|
*
|
|
|
|
* We have to search the entire constraint tree returned as we
|
|
|
|
* don't want to cook or fiddle too much.
|
|
|
|
*/
|
|
|
|
case CONSTR_DEFAULT:
|
2002-03-20 19:45:13 +00:00
|
|
|
/* Create a dummy ParseState for transformExpr */
|
|
|
|
pstate = make_parsestate(NULL);
|
2002-03-19 02:18:25 +00:00
|
|
|
/*
|
2002-03-20 19:45:13 +00:00
|
|
|
* Cook the colDef->raw_expr into an expression.
|
2002-03-19 02:18:25 +00:00
|
|
|
* Note: Name is strictly for error message
|
|
|
|
*/
|
2002-03-20 19:45:13 +00:00
|
|
|
expr = cookDefault(pstate, colDef->raw_expr,
|
|
|
|
typeTup->t_data->t_oid,
|
|
|
|
stmt->typename->typmod,
|
|
|
|
stmt->typename->name);
|
|
|
|
/*
|
|
|
|
* Expression must be stored as a nodeToString result,
|
|
|
|
* but we also require a valid textual representation
|
|
|
|
* (mainly to make life easier for pg_dump).
|
|
|
|
*/
|
2002-03-19 02:18:25 +00:00
|
|
|
defaultValue = deparse_expression(expr,
|
|
|
|
deparse_context_for(stmt->domainname,
|
|
|
|
InvalidOid),
|
|
|
|
false);
|
|
|
|
defaultValueBin = nodeToString(expr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find the NULL constraint.
|
|
|
|
*/
|
|
|
|
case CONSTR_NOTNULL:
|
|
|
|
if (nullDefined) {
|
|
|
|
elog(ERROR, "CREATE DOMAIN has conflicting NULL / NOT NULL constraint");
|
|
|
|
} else {
|
|
|
|
typNotNull = true;
|
|
|
|
nullDefined = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSTR_NULL:
|
|
|
|
if (nullDefined) {
|
|
|
|
elog(ERROR, "CREATE DOMAIN has conflicting NULL / NOT NULL constraint");
|
|
|
|
} else {
|
|
|
|
typNotNull = false;
|
|
|
|
nullDefined = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSTR_UNIQUE:
|
2002-03-20 19:45:13 +00:00
|
|
|
elog(ERROR, "CREATE DOMAIN / UNIQUE indexes not supported");
|
2002-03-19 02:18:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSTR_PRIMARY:
|
2002-03-20 19:45:13 +00:00
|
|
|
elog(ERROR, "CREATE DOMAIN / PRIMARY KEY indexes not supported");
|
2002-03-19 02:18:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSTR_CHECK:
|
2002-03-20 19:45:13 +00:00
|
|
|
elog(ERROR, "DefineDomain: CHECK Constraints not supported");
|
2002-03-19 02:18:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CONSTR_ATTR_DEFERRABLE:
|
|
|
|
case CONSTR_ATTR_NOT_DEFERRABLE:
|
|
|
|
case CONSTR_ATTR_DEFERRED:
|
|
|
|
case CONSTR_ATTR_IMMEDIATE:
|
2002-03-20 19:45:13 +00:00
|
|
|
elog(ERROR, "DefineDomain: DEFERRABLE, NON DEFERRABLE, DEFERRED and IMMEDIATE not supported");
|
2002-03-19 02:18:25 +00:00
|
|
|
break;
|
|
|
|
|
2002-03-20 19:45:13 +00:00
|
|
|
default:
|
|
|
|
elog(ERROR, "DefineDomain: unrecognized constraint node type");
|
|
|
|
break;
|
|
|
|
}
|
2002-03-19 02:18:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Have TypeCreate do all the real work.
|
|
|
|
*/
|
|
|
|
TypeCreate(stmt->domainname, /* type name */
|
|
|
|
InvalidOid, /* preassigned type oid (not done here) */
|
|
|
|
InvalidOid, /* relation oid (n/a here) */
|
|
|
|
internalLength, /* internal size */
|
|
|
|
externalLength, /* external size */
|
|
|
|
'd', /* type-type (domain type) */
|
|
|
|
delimiter, /* array element delimiter */
|
|
|
|
inputName, /* input procedure */
|
|
|
|
outputName, /* output procedure */
|
|
|
|
receiveName, /* receive procedure */
|
|
|
|
sendName, /* send procedure */
|
|
|
|
elemName, /* element type name */
|
|
|
|
typeName, /* base type name */
|
2002-03-20 19:45:13 +00:00
|
|
|
defaultValue, /* default type value (text) */
|
|
|
|
defaultValueBin, /* default type value (binary) */
|
2002-03-19 02:18:25 +00:00
|
|
|
byValue, /* passed by value */
|
|
|
|
alignment, /* required alignment */
|
|
|
|
storage, /* TOAST strategy */
|
|
|
|
stmt->typename->typmod, /* typeMod value */
|
|
|
|
typNDims, /* Array dimensions for base type */
|
|
|
|
typNotNull); /* Type NOT NULL */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now we can clean up.
|
|
|
|
*/
|
|
|
|
ReleaseSysCache(typeTup);
|
|
|
|
}
|
|
|
|
|
1996-07-09 06:22:35 +00:00
|
|
|
/*
|
1999-05-25 16:15:34 +00:00
|
|
|
* DefineType
|
1997-09-07 05:04:48 +00:00
|
|
|
* Registers a new type.
|
1996-07-09 06:22:35 +00:00
|
|
|
*/
|
|
|
|
void
|
1997-09-08 21:56:23 +00:00
|
|
|
DefineType(char *typeName, List *parameters)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
2001-10-25 05:50:21 +00:00
|
|
|
int16 internalLength = -1; /* int2 */
|
|
|
|
int16 externalLength = -1; /* int2 */
|
1997-09-08 02:41:22 +00:00
|
|
|
char *elemName = NULL;
|
|
|
|
char *inputName = NULL;
|
|
|
|
char *outputName = NULL;
|
|
|
|
char *sendName = NULL;
|
|
|
|
char *receiveName = NULL;
|
2001-09-06 02:07:42 +00:00
|
|
|
char *defaultValue = NULL;
|
1997-09-08 02:41:22 +00:00
|
|
|
bool byValue = false;
|
|
|
|
char delimiter = DEFAULT_TYPDELIM;
|
|
|
|
char *shadow_type;
|
|
|
|
List *pl;
|
2001-10-28 06:26:15 +00:00
|
|
|
char alignment = 'i'; /* default alignment */
|
2001-09-06 02:07:42 +00:00
|
|
|
char storage = 'p'; /* default TOAST storage method */
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
/*
|
2001-03-22 04:01:46 +00:00
|
|
|
* Type names must be one character shorter than other names, allowing
|
|
|
|
* room to create the corresponding array type name with prepended
|
|
|
|
* "_".
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
2000-10-07 00:58:23 +00:00
|
|
|
if (strlen(typeName) > (NAMEDATALEN - 2))
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "DefineType: type names must be %d characters or less",
|
2000-10-07 00:58:23 +00:00
|
|
|
NAMEDATALEN - 2);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
foreach(pl, parameters)
|
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
DefElem *defel = (DefElem *) lfirst(pl);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
if (strcasecmp(defel->defname, "internallength") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
internalLength = defGetTypeLength(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "externallength") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
externalLength = defGetTypeLength(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "input") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
inputName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "output") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
outputName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "send") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
sendName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "delimiter") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
char *p = defGetString(defel);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
delimiter = p[0];
|
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "receive") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
receiveName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "element") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
elemName = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "default") == 0)
|
2002-03-20 19:45:13 +00:00
|
|
|
defaultValue = defGetString(defel);
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "passedbyvalue") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
byValue = true;
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "alignment") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
{
|
1997-09-08 02:41:22 +00:00
|
|
|
char *a = defGetString(defel);
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2001-08-03 20:47:40 +00:00
|
|
|
/*
|
2001-10-25 05:50:21 +00:00
|
|
|
* Note: if argument was an unquoted identifier, parser will
|
|
|
|
* have applied xlateSqlType() to it, so be prepared to
|
|
|
|
* recognize translated type names as well as the nominal
|
|
|
|
* form.
|
2001-08-03 20:47:40 +00:00
|
|
|
*/
|
2000-10-07 00:58:23 +00:00
|
|
|
if (strcasecmp(a, "double") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
alignment = 'd';
|
2001-08-03 20:47:40 +00:00
|
|
|
else if (strcasecmp(a, "float8") == 0)
|
|
|
|
alignment = 'd';
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(a, "int4") == 0)
|
1997-09-07 05:04:48 +00:00
|
|
|
alignment = 'i';
|
2001-08-03 20:47:40 +00:00
|
|
|
else if (strcasecmp(a, "int2") == 0)
|
|
|
|
alignment = 's';
|
|
|
|
else if (strcasecmp(a, "char") == 0)
|
|
|
|
alignment = 'c';
|
|
|
|
else if (strcasecmp(a, "bpchar") == 0)
|
|
|
|
alignment = 'c';
|
1997-09-07 05:04:48 +00:00
|
|
|
else
|
2000-10-07 00:58:23 +00:00
|
|
|
elog(ERROR, "DefineType: \"%s\" alignment not recognized",
|
1997-09-07 05:04:48 +00:00
|
|
|
a);
|
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(defel->defname, "storage") == 0)
|
2000-07-03 23:10:14 +00:00
|
|
|
{
|
|
|
|
char *a = defGetString(defel);
|
|
|
|
|
2000-10-07 00:58:23 +00:00
|
|
|
if (strcasecmp(a, "plain") == 0)
|
2000-07-03 23:10:14 +00:00
|
|
|
storage = 'p';
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(a, "external") == 0)
|
2000-07-03 23:10:14 +00:00
|
|
|
storage = 'e';
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(a, "extended") == 0)
|
2000-07-03 23:10:14 +00:00
|
|
|
storage = 'x';
|
2000-10-07 00:58:23 +00:00
|
|
|
else if (strcasecmp(a, "main") == 0)
|
2000-07-03 23:10:14 +00:00
|
|
|
storage = 'm';
|
|
|
|
else
|
|
|
|
elog(ERROR, "DefineType: \"%s\" storage not recognized",
|
|
|
|
a);
|
|
|
|
}
|
1997-09-07 05:04:48 +00:00
|
|
|
else
|
|
|
|
{
|
2002-03-06 06:10:59 +00:00
|
|
|
elog(WARNING, "DefineType: attribute \"%s\" not recognized",
|
1997-09-07 05:04:48 +00:00
|
|
|
defel->defname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* make sure we have our required definitions
|
|
|
|
*/
|
|
|
|
if (inputName == NULL)
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "Define: \"input\" unspecified");
|
1997-09-07 05:04:48 +00:00
|
|
|
if (outputName == NULL)
|
1998-01-05 16:40:20 +00:00
|
|
|
elog(ERROR, "Define: \"output\" unspecified");
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2001-03-22 06:16:21 +00:00
|
|
|
/*
|
|
|
|
* now have TypeCreate do all the real work.
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
|
|
|
TypeCreate(typeName, /* type name */
|
2001-02-12 20:07:21 +00:00
|
|
|
InvalidOid, /* preassigned type oid (not done here) */
|
1997-09-07 05:04:48 +00:00
|
|
|
InvalidOid, /* relation oid (n/a here) */
|
|
|
|
internalLength, /* internal size */
|
|
|
|
externalLength, /* external size */
|
|
|
|
'b', /* type-type (base type) */
|
|
|
|
delimiter, /* array element delimiter */
|
|
|
|
inputName, /* input procedure */
|
|
|
|
outputName, /* output procedure */
|
|
|
|
receiveName, /* receive procedure */
|
1997-11-26 04:50:47 +00:00
|
|
|
sendName, /* send procedure */
|
1997-09-07 05:04:48 +00:00
|
|
|
elemName, /* element type name */
|
2002-03-20 19:45:13 +00:00
|
|
|
NULL, /* base type name (only for domains) */
|
1997-09-07 05:04:48 +00:00
|
|
|
defaultValue, /* default type value */
|
2002-03-20 19:45:13 +00:00
|
|
|
NULL, /* no binary form available */
|
1997-09-07 05:04:48 +00:00
|
|
|
byValue, /* passed by value */
|
2001-02-12 20:07:21 +00:00
|
|
|
alignment, /* required alignment */
|
2002-03-19 02:18:25 +00:00
|
|
|
storage, /* TOAST strategy */
|
|
|
|
-1, /* typMod (Domains only) */
|
|
|
|
0, /* Array Dimensions of typbasetype */
|
|
|
|
'f'); /* Type NOT NULL */
|
1997-09-07 05:04:48 +00:00
|
|
|
|
2001-03-22 06:16:21 +00:00
|
|
|
/*
|
|
|
|
* When we create a base type (as opposed to a complex type) we need
|
|
|
|
* to have an array entry for it in pg_type as well.
|
1997-09-07 05:04:48 +00:00
|
|
|
*/
|
|
|
|
shadow_type = makeArrayTypeName(typeName);
|
|
|
|
|
2001-09-06 02:07:42 +00:00
|
|
|
/* alignment must be 'i' or 'd' for arrays */
|
|
|
|
alignment = (alignment == 'd') ? 'd' : 'i';
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
TypeCreate(shadow_type, /* type name */
|
2001-02-12 20:07:21 +00:00
|
|
|
InvalidOid, /* preassigned type oid (not done here) */
|
1997-09-07 05:04:48 +00:00
|
|
|
InvalidOid, /* relation oid (n/a here) */
|
|
|
|
-1, /* internal size */
|
|
|
|
-1, /* external size */
|
|
|
|
'b', /* type-type (base type) */
|
2001-10-28 06:26:15 +00:00
|
|
|
DEFAULT_TYPDELIM, /* array element delimiter */
|
1997-09-07 05:04:48 +00:00
|
|
|
"array_in", /* input procedure */
|
|
|
|
"array_out", /* output procedure */
|
|
|
|
"array_in", /* receive procedure */
|
1997-11-26 04:50:47 +00:00
|
|
|
"array_out", /* send procedure */
|
1997-09-07 05:04:48 +00:00
|
|
|
typeName, /* element type name */
|
2002-03-19 02:18:25 +00:00
|
|
|
NULL, /* base type name */
|
2000-07-22 03:34:43 +00:00
|
|
|
NULL, /* never a default type value */
|
2002-03-19 02:18:25 +00:00
|
|
|
NULL, /* binary default isn't sent either */
|
1997-09-07 05:04:48 +00:00
|
|
|
false, /* never passed by value */
|
2001-09-06 02:07:42 +00:00
|
|
|
alignment, /* see above */
|
2002-03-19 02:18:25 +00:00
|
|
|
'x', /* ARRAY is always toastable */
|
|
|
|
-1, /* typMod (Domains only) */
|
|
|
|
0, /* Array dimensions of typbasetype */
|
|
|
|
'f'); /* Type NOT NULL */
|
1997-09-07 05:04:48 +00:00
|
|
|
|
|
|
|
pfree(shadow_type);
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1997-09-08 02:41:22 +00:00
|
|
|
static char *
|
1997-09-08 21:56:23 +00:00
|
|
|
defGetString(DefElem *def)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
2000-10-07 00:58:23 +00:00
|
|
|
if (def->arg == NULL)
|
|
|
|
elog(ERROR, "Define: \"%s\" requires a parameter",
|
|
|
|
def->defname);
|
|
|
|
switch (nodeTag(def->arg))
|
|
|
|
{
|
|
|
|
case T_Integer:
|
2001-03-22 04:01:46 +00:00
|
|
|
{
|
|
|
|
char *str = palloc(32);
|
2000-04-07 13:40:45 +00:00
|
|
|
|
2001-03-22 04:01:46 +00:00
|
|
|
snprintf(str, 32, "%ld", (long) intVal(def->arg));
|
|
|
|
return str;
|
|
|
|
}
|
2000-10-07 00:58:23 +00:00
|
|
|
case T_Float:
|
2001-03-22 04:01:46 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* T_Float values are kept in string form, so this type cheat
|
2000-10-07 00:58:23 +00:00
|
|
|
* works (and doesn't risk losing precision)
|
|
|
|
*/
|
|
|
|
return strVal(def->arg);
|
|
|
|
case T_String:
|
|
|
|
return strVal(def->arg);
|
|
|
|
case T_TypeName:
|
|
|
|
return TypeNameToInternalName((TypeName *) def->arg);
|
|
|
|
default:
|
|
|
|
elog(ERROR, "Define: cannot interpret argument of \"%s\"",
|
|
|
|
def->defname);
|
|
|
|
}
|
|
|
|
return NULL; /* keep compiler quiet */
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|
|
|
|
|
1999-10-02 21:33:33 +00:00
|
|
|
static double
|
|
|
|
defGetNumeric(DefElem *def)
|
|
|
|
{
|
|
|
|
if (def->arg == NULL)
|
|
|
|
elog(ERROR, "Define: \"%s\" requires a numeric value",
|
|
|
|
def->defname);
|
|
|
|
switch (nodeTag(def->arg))
|
|
|
|
{
|
|
|
|
case T_Integer:
|
|
|
|
return (double) intVal(def->arg);
|
|
|
|
case T_Float:
|
|
|
|
return floatVal(def->arg);
|
|
|
|
default:
|
|
|
|
elog(ERROR, "Define: \"%s\" requires a numeric value",
|
|
|
|
def->defname);
|
|
|
|
}
|
|
|
|
return 0; /* keep compiler quiet */
|
|
|
|
}
|
|
|
|
|
1997-09-07 05:04:48 +00:00
|
|
|
static int
|
1997-09-08 21:56:23 +00:00
|
|
|
defGetTypeLength(DefElem *def)
|
1996-07-09 06:22:35 +00:00
|
|
|
{
|
2000-10-07 00:58:23 +00:00
|
|
|
if (def->arg == NULL)
|
|
|
|
elog(ERROR, "Define: \"%s\" requires a parameter",
|
|
|
|
def->defname);
|
|
|
|
switch (nodeTag(def->arg))
|
|
|
|
{
|
|
|
|
case T_Integer:
|
|
|
|
return intVal(def->arg);
|
|
|
|
case T_Float:
|
|
|
|
elog(ERROR, "Define: \"%s\" requires an integral value",
|
|
|
|
def->defname);
|
|
|
|
break;
|
|
|
|
case T_String:
|
|
|
|
if (strcasecmp(strVal(def->arg), "variable") == 0)
|
|
|
|
return -1; /* variable length */
|
|
|
|
break;
|
|
|
|
case T_TypeName:
|
|
|
|
/* cope if grammar chooses to believe "variable" is a typename */
|
|
|
|
if (strcasecmp(TypeNameToInternalName((TypeName *) def->arg),
|
|
|
|
"variable") == 0)
|
|
|
|
return -1; /* variable length */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
elog(ERROR, "Define: cannot interpret argument of \"%s\"",
|
|
|
|
def->defname);
|
|
|
|
}
|
|
|
|
elog(ERROR, "Define: invalid argument for \"%s\"",
|
|
|
|
def->defname);
|
|
|
|
return 0; /* keep compiler quiet */
|
1996-07-09 06:22:35 +00:00
|
|
|
}
|