8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-28 02:43:03 +01:00
firebird-mirror/src/dsql/parse.y

5179 lines
129 KiB
Plaintext
Raw Normal View History

2001-05-23 15:26:42 +02:00
%{
/*
* PROGRAM: Dynamic SQL runtime support
* MODULE: parse.y
* DESCRIPTION: Dynamic SQL parser
*
* The contents of this file are subject to the Interbase Public
* License Version 1.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy
* of the License at http://www.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
* 2002-02-24 Sean Leyne - Code Cleanup of old Win 3.1 port (WINDOWS_ONLY)
2002-06-29 08:56:51 +02:00
* 2001.05.20 Neil McCalden: Allow a udf to be used in a 'group by' clause.
* 2001.05.30 Claudio Valderrama: DROP TABLE and DROP VIEW lead now to two
* different node types so DDL can tell which is which.
* 2001.06.13 Claudio Valderrama: SUBSTRING is being surfaced.
* 2001.06.30 Claudio valderrama: Feed (line,column) for each node. See node.h.
* 2001.07.10 Claudio Valderrama: Better (line,column) report and "--" for comments.
* 2001.07.28 John Bellardo: Changes to support parsing LIMIT and FIRST
* 2001.08.03 John Bellardo: Finalized syntax for LIMIT, change LIMIT to SKIP
* 2001.08.05 Claudio Valderrama: closed Bug #448062 and other spaces that appear
* in rdb$*_source fields when altering domains plus one unexpected null pointer.
* 2001.08.12 Claudio Valderrama: adjust SUBSTRING's starting pos argument here
* and not in gen.c; this closes Bug #450301.
* 2001.10.01 Claudio Valderrama: enable explicit GRANT...to ROLE role_name.
* 2001.10.06 Claudio Valderrama: Honor explicit USER keyword in GRANTs and REVOKEs.
* 2002.07.05 Mark O'Donohue: change keyword DEBUG to KW_DEBUG to avoid
2003-09-04 23:26:15 +02:00
* clashes with normal DEBUG macro.
* 2002.07.30 Arno Brinkman:
* 2002.07.30 Let IN predicate handle value_expressions
* 2002.07.30 tokens CASE, NULLIF, COALESCE added
* 2002.07.30 See block < CASE expression > what is added to value as case_expression
* 2002.07.30 function is split up into aggregate_function, numeric_value_function, string_value_function, generate_value_function
* 2002.07.30 new group_by_function and added to grp_column_elem
* 2002.07.30 cast removed from function and added as cast_specification to value
* 2002.08.04 Claudio Valderrama: allow declaring and defining variables at the same time
* 2002.08.04 Dmitry Yemanov: ALTER VIEW
* 2002.08.06 Arno Brinkman: ordinal added to grp_column_elem for using positions in group by
* 2002.08.07 Dmitry Yemanov: INT64/LARGEINT are replaced with BIGINT and available in dialect 3 only
* 2002.08.31 Dmitry Yemanov: allowed user-defined index names for PK/FK/UK constraints
* 2002.09.01 Dmitry Yemanov: RECREATE VIEW
* 2002.09.28 Dmitry Yemanov: Reworked internal_info stuff, enhanced
2003-09-04 23:26:15 +02:00
* exception handling in SPs/triggers,
* implemented ROWS_AFFECTED system variable
* 2002.10.21 Nickolay Samofatov: Added support for explicit pessimistic locks
* 2002.10.29 Nickolay Samofatov: Added support for savepoints
* 2002.12.03 Dmitry Yemanov: Implemented ORDER BY clause in subqueries.
* 2002.12.18 Dmitry Yemanov: Added support for SQL-compliant labels and LEAVE statement
* 2002.12.28 Dmitry Yemanov: Added support for parametrized events.
* 2003.01.14 Dmitry Yemanov: Fixed bug with cursors in triggers.
* 2003.01.15 Dmitry Yemanov: Added support for runtime trigger action checks.
* 2003.02.10 Mike Nordell : Undefined Microsoft introduced macros to get a clean compile.
2003-05-24 17:18:35 +02:00
* 2003.05.24 Nickolay Samofatov: Make SKIP and FIRST non-reserved keywords
* 2003.06.13 Nickolay Samofatov: Make INSERTING/UPDATING/DELETING non-reserved keywords
* 2003.07.01 Blas Rodriguez Somoza: Change DEBUG and IN to avoid conflicts in win32 build/bison
* 2003.08.11 Arno Brinkman: Changed GROUP BY to support all expressions and added "AS" support
2003-09-04 23:26:15 +02:00
* with table alias. Also removed group_by_function and ordinal.
* 2003.08.14 Arno Brinkman: Added support for derived tables.
2001-05-23 15:26:42 +02:00
*/
2003-01-18 20:39:12 +01:00
#if defined(DEV_BUILD) && defined(WIN_NT) && defined(SUPERSERVER)
2002-06-29 08:56:51 +02:00
#include <windows.h>
#include <stdio.h>
2002-06-29 08:56:51 +02:00
/*#include <wincon.h>*/
#endif
#include "firebird.h"
2001-05-23 15:26:42 +02:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "../jrd/common.h"
#include <stdarg.h>
#include "gen/iberror.h"
2001-05-23 15:26:42 +02:00
#include "../dsql/dsql.h"
#include "../jrd/gds.h"
#include "../jrd/flags.h"
#include "../dsql/alld_proto.h"
#include "../dsql/errd_proto.h"
#include "../dsql/hsh_proto.h"
#include "../dsql/make_proto.h"
#include "../dsql/parse_proto.h"
#include "../dsql/keywords.h"
#include "../dsql/misc_func.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/gds_proto.h"
#include "../jrd/thd_proto.h"
#include "../wal/wal.h"
2003-09-12 03:41:03 +02:00
#include "../jrd/err_proto.h"
2001-07-12 07:46:06 +02:00
2001-05-23 15:26:42 +02:00
static void yyerror (TEXT *);
/* since UNIX isn't standard, we have to define
stuff which is in <limits.h> (which isn't available
on all UNIXes... */
2003-10-15 00:22:32 +02:00
const long SHRT_POS_MAX = 32767;
const long SHRT_UNSIGNED_MAX = 65535;
const long SHRT_NEG_MAX = 32768;
const long LONG_POS_MAX = 2147483647;
const int POSITIVE = 0;
const int NEGATIVE = 1;
const int UNSIGNED = 2;
2001-05-23 15:26:42 +02:00
2003-10-15 00:22:32 +02:00
const int MIN_CACHE_BUFFERS = 250;
const int DEF_CACHE_BUFFERS = 1000;
2001-05-23 15:26:42 +02:00
2002-06-29 08:56:51 +02:00
/* Fix 69th procedure problem - solution from Oleg Loa */
2003-10-15 00:22:32 +02:00
#define YYSTACKSIZE 2048
#define YYMAXDEPTH 2048
2002-06-29 08:56:51 +02:00
2002-11-11 20:08:37 +01:00
#define YYSTYPE DSQL_NOD
2002-06-29 08:56:51 +02:00
#if defined(DEBUG) || defined(DEV_BUILD)
2001-05-23 15:26:42 +02:00
#define YYDEBUG 1
#endif
static const char INTERNAL_FIELD_NAME [] = "DSQL internal"; /* NTX: placeholder */
static const char NULL_STRING [] = "";
2001-05-23 15:26:42 +02:00
2003-10-15 00:22:32 +02:00
inline SLONG trigger_type_suffix(int slot1, int slot2, int slot3){
return ((slot1 << 1) | (slot2 << 3) | (slot3 << 5));
}
2002-11-11 20:08:37 +01:00
DSQL_NOD DSQL_parse;
2001-05-23 15:26:42 +02:00
2002-06-29 08:56:51 +02:00
static void yyerror (TEXT *);
2001-05-23 15:26:42 +02:00
#define YYPARSE_PARAM_TYPE
#define YYPARSE_PARAM USHORT client_dialect, USHORT db_dialect, USHORT parser_version, BOOLEAN *stmt_ambiguous
#include "../dsql/chars.h"
2003-10-15 00:22:32 +02:00
const int MAX_TOKEN_LEN = 256;
static TEXT *lex_position (void);
2003-04-03 03:36:52 +02:00
#ifdef NOT_USED_OR_REPLACED
2003-09-04 23:26:15 +02:00
static bool long_int(DSQL_NOD, SLONG *);
2003-04-03 03:36:52 +02:00
#endif
static DSQL_FLD make_field (DSQL_NOD);
static FIL make_file (void);
static DSQL_NOD make_list (DSQL_NOD);
static DSQL_NOD make_node (NOD_TYPE, int, ...);
static DSQL_NOD make_parameter (void);
static DSQL_NOD make_flag_node (NOD_TYPE, SSHORT, int, ...);
static void prepare_console_debug (int, int *);
2003-04-03 03:36:52 +02:00
#ifdef NOT_USED_OR_REPLACED
2003-09-04 23:26:15 +02:00
static bool short_int(DSQL_NOD, SLONG *, SSHORT);
2003-04-03 03:36:52 +02:00
#endif
static void stack_nodes (DSQL_NOD, DLLS *);
2003-05-23 18:55:40 +02:00
inline static int yylex (USHORT, USHORT, USHORT, BOOLEAN *);
2003-04-10 08:32:58 +02:00
static void yyabandon (SSHORT, ISC_STATUS);
static void check_log_file_attrs (void);
2003-10-15 00:22:32 +02:00
inline void check_bound(char *to, char *string)
{
if ((to - string) >= MAX_TOKEN_LEN)
yyabandon (-104, isc_token_too_long);
}
inline void check_copy_incr(char*& to, char ch, char* string)
{
check_bound(to, string);
*to++=ch;
}
2003-05-23 18:55:40 +02:00
struct LexerState {
/* This is, in fact, parser state. Not used in lexer itself */
DSQL_FLD g_field;
2003-09-04 23:26:15 +02:00
FIL g_file;
2003-05-23 18:55:40 +02:00
DSQL_NOD g_field_name;
SSHORT log_defined, cache_defined;
int dsql_debug;
/* Actual lexer state begins from here */
TEXT *beginning;
TEXT *ptr, *end, *last_token, *line_start;
TEXT *last_token_bk, *line_start_bk;
SSHORT lines, att_charset;
SSHORT lines_bk;
int prev_keyword, prev_prev_keyword;
2003-05-23 18:55:40 +02:00
USHORT param_number;
/* Fields to handle FIRST/SKIP as non-reserved keywords */
bool limit_clause; /* We are inside of limit clause. Need to detect SKIP after FIRST */
bool first_detection; /* Detect FIRST unconditionally */
/* Fields to handle INSERTING/UPDATING/DELETING as non-reserved keywords */
bool brace_analysis; /* When this is true lexer is informed not to swallow braces around INSERTING/UPDATING/DELETING */
2003-05-23 18:55:40 +02:00
int yylex (
USHORT client_dialect,
USHORT db_dialect,
USHORT parser_version,
BOOLEAN *stmt_ambiguous);
};
/* Get ready for thread-safety. Move this to BISON object pointer when we
switch to generating "pure" reenterant parser. */
static LexerState lex;
2001-05-23 15:26:42 +02:00
%}
/* token declarations */
/* Tokens are organized chronologically by date added.
See dsql/keywords.cpp for a list organized alphabetically */
2001-05-23 15:26:42 +02:00
/* Tokens in v4.0 -- not separated into v3 and v4 tokens */
%token ACTIVE
%token ADD
%token AFTER
%token ALL
%token ALTER
%token AND
%token ANY
%token AS
%token ASC
%token AT
%token AVG
%token AUTO
%token BASENAME
%token BEFORE
%token BEGIN
%token BETWEEN
%token BLOB
%token BY
%token CACHE
%token CAST
%token CHARACTER
%token CHECK
%token CHECK_POINT_LEN
%token COLLATE
%token COMMA
%token COMMIT
%token COMMITTED
%token COMPUTED
%token CONCATENATE
%token CONDITIONAL
%token CONSTRAINT
%token CONTAINING
%token COUNT
%token CREATE
%token CSTRING
%token CURRENT
%token CURSOR
%token DATABASE
%token DATE
%token DB_KEY
%token KW_DEBUG
2001-05-23 15:26:42 +02:00
%token DECIMAL
%token DECLARE
%token DEFAULT
%token KW_DELETE
2001-05-23 15:26:42 +02:00
%token DESC
%token DISTINCT
%token DO
%token DOMAIN
%token DROP
%token ELSE
%token END
%token ENTRY_POINT
%token EQL
%token ESCAPE
%token EXCEPTION
%token EXECUTE
%token EXISTS
%token EXIT
%token EXTERNAL
%token FILTER
%token FOR
%token FOREIGN
%token FROM
%token FULL
%token FUNCTION
%token GDSCODE
%token GEQ
%token GENERATOR
%token GEN_ID
%token GRANT
%token GROUP
%token GROUP_COMMIT_WAIT
%token GTR
%token HAVING
%token IF
%token KW_IN
2001-05-23 15:26:42 +02:00
%token INACTIVE
%token INNER
%token INPUT_TYPE
%token INDEX
%token INSERT
%token INTEGER
%token INTO
%token IS
%token ISOLATION
%token JOIN
%token KEY
%token KW_CHAR
%token KW_DEC
%token KW_DOUBLE
%token KW_FILE
%token KW_FLOAT
%token KW_INT
%token KW_LONG
%token KW_NULL
%token KW_NUMERIC
%token KW_UPPER
%token KW_VALUE
%token LENGTH
%token LOGFILE
%token LPAREN
%token LEFT
%token LEQ
%token LEVEL
%token LIKE
%token LOG_BUF_SIZE
%token LSS
%token MANUAL
%token MAXIMUM
%token MAX_SEGMENT
%token MERGE
2003-04-06 13:34:31 +02:00
%token MESSAGE
2001-05-23 15:26:42 +02:00
%token MINIMUM
%token MODULE_NAME
%token NAMES
%token NATIONAL
%token NATURAL
%token NCHAR
%token NEQ
%token NO
%token NOT
%token NOT_GTR
%token NOT_LSS
%token NUM_LOG_BUFS
%token OF
%token ON
%token ONLY
%token OPTION
%token OR
%token ORDER
%token OUTER
%token OUTPUT_TYPE
%token OVERFLOW
%token PAGE
%token PAGES
%token PAGE_SIZE
%token PARAMETER
%token PASSWORD
%token PLAN
%token POSITION
%token POST_EVENT
%token PRECISION
%token PRIMARY
%token PRIVILEGES
%token PROCEDURE
%token PROTECTED
%token RAW_PARTITIONS
%token READ
%token REAL
%token REFERENCES
%token RESERVING
%token RETAIN
%token RETURNING_VALUES
%token RETURNS
%token REVOKE
%token RIGHT
%token RPAREN
%token ROLLBACK
%token SEGMENT
%token SELECT
%token SET
%token SHADOW
%token KW_SHARED
2001-05-23 15:26:42 +02:00
%token SINGULAR
%token KW_SIZE
2001-05-23 15:26:42 +02:00
%token SMALLINT
%token SNAPSHOT
%token SOME
%token SORT
%token SQLCODE
%token STABILITY
%token STARTING
%token STATISTICS
%token SUB_TYPE
%token SUSPEND
%token SUM
%token TABLE
%token THEN
%token TO
%token TRANSACTION
%token TRIGGER
%token UNCOMMITTED
%token UNION
%token UNIQUE
%token UPDATE
%token USER
%token VALUES
%token VARCHAR
%token VARIABLE
%token VARYING
%token VERSION
%token VIEW
%token WAIT
%token WHEN
%token WHERE
%token WHILE
%token WITH
%token WORK
%token WRITE
%token FLOAT_NUMBER NUMBER NUMERIC SYMBOL STRING INTRODUCER
2001-05-23 15:26:42 +02:00
/* New tokens added v5.0 */
%token ACTION
%token ADMIN
%token CASCADE
%token FREE_IT /* ISC SQL extension */
%token RESTRICT
%token ROLE
/* New tokens added v6.0 */
%token COLUMN
%token TYPE
%token EXTRACT
%token YEAR
%token MONTH
%token DAY
%token HOUR
%token MINUTE
%token SECOND
%token WEEKDAY /* ISC SQL extension */
%token YEARDAY /* ISC SQL extension */
%token TIME
%token TIMESTAMP
%token CURRENT_DATE
%token CURRENT_TIME
%token CURRENT_TIMESTAMP
/* special aggregate token types returned by lex in v6.0 */
%token NUMBER64BIT SCALEDINT
/* CVC: Special Firebird additions. */
%token CURRENT_USER
%token CURRENT_ROLE
%token KW_BREAK
2001-05-23 15:26:42 +02:00
%token SUBSTRING
2002-07-06 07:32:02 +02:00
%token RECREATE
%token KW_DESCRIPTOR
%token FIRST
%token SKIP
2001-05-23 15:26:42 +02:00
/* tokens added for Firebird 1.5 */
%token CURRENT_CONNECTION
%token CURRENT_TRANSACTION
%token BIGINT
%token CASE
%token NULLIF
%token COALESCE
%token USING
2002-09-10 20:28:23 +02:00
%token NULLS
%token LAST
%token ROW_COUNT
%token LOCK
%token SAVEPOINT
%token RELEASE
%token STATEMENT
2002-12-19 09:57:53 +01:00
%token LEAVE
%token INSERTING
%token UPDATING
%token DELETING
/* Special pseudo-tokens introduced to handle case
when our grammar is not LARL(1) */
%token KW_INSERTING
%token KW_UPDATING
%token KW_DELETING
2001-05-23 15:26:42 +02:00
/* tokens added for Firebird 2.0 */
%token BACKUP
%token KW_DIFFERENCE
2001-05-23 15:26:42 +02:00
/* precedence declarations for expression evaluation */
%left OR
%left AND
%left NOT
%left '=' '<' '>' LIKE EQL NEQ GTR LSS GEQ LEQ NOT_GTR NOT_LSS
%left '+' '-'
%left '*' '/'
%left CONCATENATE
%left COLLATE
/* Fix the dangling IF-THEN-ELSE problem */
%nonassoc THEN
%nonassoc ELSE
/* The same issue exists with ALTER COLUMN now that keywords can be used
2002-06-29 08:56:51 +02:00
in order to change their names. The syntax which shows the issue is:
2003-09-04 23:26:15 +02:00
ALTER COLUMN where column is part of the alter statement
or
ALTER COLUMN where column is the name of the column in the relation
2001-05-23 15:26:42 +02:00
*/
%nonassoc ALTER
%nonassoc COLUMN
%%
/* list of possible statements */
top : statement
{ DSQL_parse = $1; }
| statement ';'
{ DSQL_parse = $1; }
;
statement : alter
| blob
| commit
| create
| declare
| delete
| drop
| grant
| insert
| invoke_procedure
2003-09-04 23:26:15 +02:00
| recreate
| replace
2001-05-23 15:26:42 +02:00
| revoke
| rollback
| savepoint
2001-05-23 15:26:42 +02:00
| select
| set
| update
| KW_DEBUG signed_short_integer
2002-06-29 08:56:51 +02:00
{ prepare_console_debug ((int) $2, &yydebug);
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_null, (int) 0, NULL); }
;
/* GRANT statement */
grant : GRANT privileges ON prot_table_name
TO user_grantee_list grant_option
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $4, make_list($6), $7); }
| GRANT proc_privileges ON PROCEDURE simple_proc_name
TO user_grantee_list grant_option
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $5, make_list($7), $8); }
| GRANT privileges ON prot_table_name
TO grantee_list
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $4, make_list($6), NULL); }
| GRANT proc_privileges ON PROCEDURE simple_proc_name
TO grantee_list
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $5, make_list($7), NULL); }
| GRANT role_name_list TO role_grantee_list role_admin_option
{ $$ = make_node (nod_grant, (int) e_grant_count,
make_list($2), make_list($4), NULL, $5); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
prot_table_name : simple_table_name
| TABLE simple_table_name
{ $$ = $2; }
;
privileges : ALL
{ $$ = make_node (nod_all, (int) 0, NULL); }
| ALL PRIVILEGES
{ $$ = make_node (nod_all, (int) 0, NULL); }
| privilege_list
{ $$ = make_list ($1); }
;
privilege_list : privilege
| privilege_list ',' privilege
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
proc_privileges : EXECUTE
{ $$ = make_list (make_node (nod_execute, (int) 0, NULL)); }
;
privilege : SELECT
{ $$ = make_node (nod_select, (int) 0, NULL); }
| INSERT
{ $$ = make_node (nod_insert, (int) 0, NULL); }
| KW_DELETE
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_delete, (int) 0, NULL); }
| UPDATE column_parens_opt
{ $$ = make_node (nod_update, (int) 1, $2); }
| REFERENCES column_parens_opt
{ $$ = make_node (nod_references, (int) 1, $2); }
;
grant_option : WITH GRANT OPTION
{ $$ = make_node (nod_grant, (int) 0, NULL); }
|
{ $$ = 0; }
;
role_admin_option : WITH ADMIN OPTION
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_grant_admin, (int) 0, NULL); }
|
{ $$ = 0; }
;
2001-05-23 15:26:42 +02:00
simple_proc_name: symbol_procedure_name
{ $$ = make_node (nod_procedure_name, (int) 1, $1); }
;
/* REVOKE statement */
revoke : REVOKE rev_grant_option privileges ON prot_table_name
FROM user_grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $3, $5,
make_list($7), $2); }
| REVOKE rev_grant_option proc_privileges ON
PROCEDURE simple_proc_name FROM user_grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $3, $6,
make_list($8), $2); }
| REVOKE privileges ON prot_table_name
FROM user_grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $2, $4,
make_list($6), NULL); }
| REVOKE proc_privileges ON
PROCEDURE simple_proc_name FROM user_grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $2, $5,
make_list($7), NULL); }
| REVOKE privileges ON prot_table_name
FROM grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $2, $4,
make_list($6), NULL); }
| REVOKE proc_privileges ON
PROCEDURE simple_proc_name FROM grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, $2, $5,
make_list($7), NULL); }
| REVOKE role_name_list FROM role_grantee_list
{ $$ = make_node (nod_revoke,
(int) e_grant_count, make_list($2), make_list($4),
NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
rev_grant_option : GRANT OPTION FOR
{ $$ = make_node (nod_grant, (int) 0, NULL); }
;
grantee_list : grantee
| grantee_list ',' grantee
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
| grantee_list ',' user_grantee
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
| user_grantee_list ',' grantee
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
grantee : PROCEDURE symbol_procedure_name
{ $$ = make_node (nod_proc_obj, (int) 1, $2); }
| TRIGGER symbol_trigger_name
{ $$ = make_node (nod_trig_obj, (int) 1, $2); }
| VIEW symbol_view_name
{ $$ = make_node (nod_view_obj, (int) 1, $2); }
2002-06-29 08:56:51 +02:00
| ROLE symbol_role_name
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_role_name, (int) 1, $2); }
2001-05-23 15:26:42 +02:00
;
user_grantee_list : user_grantee
| user_grantee_list ',' user_grantee
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
2002-06-29 08:56:51 +02:00
/* CVC: In the future we can deprecate the first implicit form since we'll support
explicit grant/revoke for both USER and ROLE keywords & object types. */
2001-05-23 15:26:42 +02:00
user_grantee : symbol_user_name
{ $$ = make_node (nod_user_name, (int) 1, $1); }
| USER symbol_user_name
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_user_name, (int) 2, $2, NULL); }
2001-05-23 15:26:42 +02:00
| GROUP symbol_user_name
{ $$ = make_node (nod_user_group, (int) 1, $2); }
;
role_name_list : role_name
2003-09-04 23:26:15 +02:00
| role_name_list ',' role_name
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
2001-05-23 15:26:42 +02:00
role_name : symbol_role_name
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_role_name, (int) 1, $1); }
;
2001-05-23 15:26:42 +02:00
role_grantee_list : role_grantee
2003-09-04 23:26:15 +02:00
| role_grantee_list ',' role_grantee
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
2001-05-23 15:26:42 +02:00
role_grantee : symbol_user_name
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_user_name, (int) 1, $1); }
| USER symbol_user_name
{ $$ = make_node (nod_user_name, (int) 1, $2); }
;
2001-05-23 15:26:42 +02:00
/* DECLARE operations */
declare : DECLARE declare_clause
{ $$ = $2;}
;
declare_clause : FILTER filter_decl_clause
{ $$ = $2; }
| EXTERNAL FUNCTION udf_decl_clause
{ $$ = $3; }
;
udf_decl_clause : symbol_UDF_name arg_desc_list1 RETURNS return_value1
ENTRY_POINT sql_string MODULE_NAME sql_string
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_udf, (int) e_udf_count,
2001-05-23 15:26:42 +02:00
$1, $6, $8, make_list ($2), $4); }
;
udf_data_type : simple_type
| BLOB
2003-05-23 18:55:40 +02:00
{ lex.g_field->fld_dtype = dtype_blob; }
2001-05-23 15:26:42 +02:00
| CSTRING '(' pos_short_integer ')' charset_clause
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_cstring;
lex.g_field->fld_character_length = (USHORT)(ULONG) $3; }
2001-05-23 15:26:42 +02:00
;
arg_desc_list1 :
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| arg_desc_list
| '(' arg_desc_list ')'
{ $$ = $2; }
;
arg_desc_list : arg_desc
| arg_desc_list ',' arg_desc
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
2002-06-29 08:56:51 +02:00
/*arg_desc : init_data_type udf_data_type
{ $$ = $1; } */
2001-05-23 15:26:42 +02:00
arg_desc : init_data_type udf_data_type
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_udf_param, (int) e_udf_param_count,
$1, NULL); }
2002-06-29 08:56:51 +02:00
| init_data_type udf_data_type BY KW_DESCRIPTOR
{ $$ = make_node (nod_udf_param, (int) e_udf_param_count,
$1, MAKE_constant ((STR) FUN_descriptor, CONSTANT_SLONG)); }
2001-05-23 15:26:42 +02:00
;
2002-06-29 08:56:51 +02:00
2001-05-23 15:26:42 +02:00
return_value1 : return_value
| '(' return_value ')'
{ $$ = $2; }
;
return_value : init_data_type udf_data_type
{ $$ = make_node (nod_udf_return_value, (int) 2, $1,
MAKE_constant ((STR) FUN_reference, CONSTANT_SLONG));}
| init_data_type udf_data_type FREE_IT
{ $$ = make_node (nod_udf_return_value, (int) 2, $1,
MAKE_constant ((STR) (-1 * FUN_reference), CONSTANT_SLONG));}
2003-09-04 23:26:15 +02:00
/* FUN_refrence with FREE_IT is -ve */
2001-05-23 15:26:42 +02:00
| init_data_type udf_data_type BY KW_VALUE
{ $$ = make_node (nod_udf_return_value, (int) 2, $1,
MAKE_constant ((STR) FUN_value, CONSTANT_SLONG));}
2002-06-29 08:56:51 +02:00
/* CVC: Enable return by descriptor for the future.*/
| init_data_type udf_data_type BY KW_DESCRIPTOR
{ $$ = make_node (nod_udf_return_value, (int) 2, $1,
MAKE_constant ((STR) FUN_descriptor, CONSTANT_SLONG));}
2001-05-23 15:26:42 +02:00
| PARAMETER pos_short_integer
{ $$ = make_node (nod_udf_return_value, (int) 2,
2003-09-04 23:26:15 +02:00
NULL, MAKE_constant ((STR) $2, CONSTANT_SLONG));}
2001-05-23 15:26:42 +02:00
;
filter_decl_clause : symbol_filter_name INPUT_TYPE blob_subtype OUTPUT_TYPE blob_subtype
ENTRY_POINT sql_string MODULE_NAME sql_string
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_filter, (int) e_filter_count,
2001-05-23 15:26:42 +02:00
$1, $3, $5, $7, $9); }
;
/* CREATE metadata operations */
create : CREATE create_clause
{ $$ = $2; }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
create_clause : EXCEPTION symbol_exception_name sql_string
{ $$ = make_node (nod_def_exception, (int) e_xcp_count,
$2, $3); }
| unique_opt order_direction INDEX symbol_index_name ON simple_table_name index_definition
{ $$ = make_node (nod_def_index, (int) e_idx_count,
$1, $2, $4, $6, $7); }
| PROCEDURE procedure_clause
{ $$ = $2; }
| TABLE table_clause
{ $$ = $2; }
| TRIGGER def_trigger_clause
{ $$ = $2; }
| VIEW view_clause
{ $$ = $2; }
| GENERATOR generator_clause
{ $$ = $2; }
| DATABASE db_clause
{ $$ = $2; }
| DOMAIN domain_clause
2003-09-04 23:26:15 +02:00
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| SHADOW shadow_clause
2003-09-04 23:26:15 +02:00
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| ROLE role_clause
{ $$ = $2; }
;
2002-06-29 08:56:51 +02:00
recreate : RECREATE recreate_clause
{ $$ = $2; }
;
recreate_clause : PROCEDURE rprocedure_clause
{ $$ = $2; }
| TABLE rtable_clause
{ $$ = $2; }
| VIEW rview_clause
{ $$ = $2; }
/*
| TRIGGER def_trigger_clause
{ $$ = $2; }
2002-06-29 08:56:51 +02:00
| DOMAIN rdomain_clause
2003-09-04 23:26:15 +02:00
{ $$ = $2; }
2002-06-29 08:56:51 +02:00
*/
;
replace : CREATE OR ALTER replace_clause
{ $$ = $4; }
;
replace_clause : PROCEDURE replace_procedure_clause
{ $$ = $2; }
| TRIGGER replace_trigger_clause
{ $$ = $2; }
/*
| VIEW replace_view_clause
{ $$ = $2; }
*/
;
2002-06-29 08:56:51 +02:00
2001-05-23 15:26:42 +02:00
/* CREATE INDEX */
unique_opt : UNIQUE
{ $$ = make_node (nod_unique, 0, NULL); }
2001-05-23 15:26:42 +02:00
|
{ $$ = NULL; }
;
index_definition : column_list
{ $$ = make_list ($1); }
| column_parens
| computed_by '(' begin_trigger value end_trigger ')'
{ $$ = make_node (nod_def_computed, 2, $4, $5); }
;
/* CREATE SHADOW */
shadow_clause : pos_short_integer manual_auto conditional sql_string
first_file_length sec_shadow_files
{ $$ = make_node (nod_def_shadow, (int) e_shadow_count,
2003-09-04 23:26:15 +02:00
$1, $2, $3, $4, $5, make_list ($6)); }
2001-05-23 15:26:42 +02:00
;
manual_auto : MANUAL
{ $$ = MAKE_constant ((STR) 1, CONSTANT_SLONG); }
| AUTO
{ $$ = MAKE_constant ((STR) 0, CONSTANT_SLONG); }
|
{ $$ = MAKE_constant ((STR) 0, CONSTANT_SLONG); }
;
conditional :
{ $$ = MAKE_constant ((STR) 0, CONSTANT_SLONG); }
| CONDITIONAL
{ $$ = MAKE_constant ((STR) 1, CONSTANT_SLONG); }
;
first_file_length :
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) 0;}
2001-05-23 15:26:42 +02:00
| LENGTH equals long_integer page_noise
{ $$ = $3; }
;
sec_shadow_files :
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| db_file_list
;
2003-09-04 23:26:15 +02:00
db_file_list : db_file
2001-05-23 15:26:42 +02:00
| db_file_list db_file
{ $$ = make_node (nod_list, (int) 2, $1, $2); }
;
/* CREATE DOMAIN */
domain_clause : column_def_name
2003-09-04 23:26:15 +02:00
as_opt
data_type
begin_trigger
domain_default_opt
end_trigger
domain_constraint_clause
collate_clause
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_def_domain, (int) e_dom_count,
2003-09-04 23:26:15 +02:00
$1, $5, $6, make_list ($7), $8); }
;
2001-05-23 15:26:42 +02:00
2002-06-29 08:56:51 +02:00
/*
rdomain_clause : DOMAIN alter_column_name alter_domain_ops
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_mod_domain, (int) e_alt_count,
$2, make_list ($3)); }
2002-06-29 08:56:51 +02:00
*/
2001-05-23 15:26:42 +02:00
as_opt : AS
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
|
{ $$ = NULL; }
;
2001-05-23 15:26:42 +02:00
domain_default_opt : DEFAULT begin_trigger default_value
{ $$ = $3; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
domain_constraint_clause :
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| domain_constraint_list
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
domain_constraint_list : domain_constraint_def
| domain_constraint_list domain_constraint_def
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $2); }
;
2001-05-23 15:26:42 +02:00
domain_constraint_def : domain_constraint
{ $$ = make_node (nod_rel_constraint, (int) 2, NULL, $1);}
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
domain_constraint : null_constraint
| domain_check_constraint
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
null_constraint : NOT KW_NULL
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_null, (int) 0, NULL); }
;
2001-05-23 15:26:42 +02:00
domain_check_constraint : begin_trigger CHECK '(' search_condition ')' end_trigger
{ $$ = make_node (nod_def_constraint,
(int) e_cnstr_count, MAKE_string(NULL_STRING, 0), NULL,
2001-05-23 15:26:42 +02:00
NULL, NULL, $4, NULL, $6, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
/* CREATE GENERATOR */
generator_clause : symbol_generator_name
{ $$ = make_node (nod_def_generator,
(int) e_gen_count, $1); }
;
/* CREATE ROLE */
role_clause : symbol_role_name
{ $$ = make_node (nod_def_role,
(int) 1, $1); }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
/* CREATE DATABASE */
db_clause : db_name db_initial_desc1 db_rem_desc1
{ $$ = make_node (nod_def_database, (int) e_cdb_count,
$1, make_list($2), make_list ($3));}
;
equals :
| '='
;
db_name : sql_string
2003-05-23 18:55:40 +02:00
{ lex.log_defined = FALSE;
lex.cache_defined = FALSE;
2002-11-11 20:08:37 +01:00
$$ = (DSQL_NOD) $1; }
2001-05-23 15:26:42 +02:00
;
db_initial_desc1 :
{$$ = NULL;}
2001-05-23 15:26:42 +02:00
| db_initial_desc
;
db_initial_desc : db_initial_option
| db_initial_desc db_initial_option
{ $$ = make_node (nod_list, 2, $1, $2); }
;
db_initial_option: PAGE_SIZE equals pos_short_integer
{ $$ = make_node (nod_page_size, 1, $3);}
| LENGTH equals long_integer page_noise
{ $$ = make_node (nod_file_length, 1, $3);}
| USER sql_string
{ $$ = make_node (nod_user_name, 1, $2);}
| PASSWORD sql_string
{ $$ = make_node (nod_password, 1, $2);}
| SET NAMES sql_string
{ $$ = make_node (nod_lc_ctype, 1, $3);}
;
db_rem_desc1 :
{$$ = NULL;}
2001-05-23 15:26:42 +02:00
| db_rem_desc
;
db_rem_desc : db_rem_option
| db_rem_desc db_rem_option
{ $$ = make_node (nod_list, 2, $1, $2); }
;
db_rem_option : db_file
/* | db_cache */
| db_log
| db_log_option
| DEFAULT CHARACTER SET symbol_character_set_name
{ $$ = make_node (nod_dfl_charset, 1, $4);}
| KW_DIFFERENCE KW_FILE sql_string
{ $$ = make_node (nod_difference_file, 1, $3); }
2001-05-23 15:26:42 +02:00
;
db_log_option : GROUP_COMMIT_WAIT equals long_integer
{ $$ = make_node (nod_group_commit_wait, 1, $3);}
| CHECK_POINT_LEN equals long_integer
{ $$ = make_node (nod_check_point_len, 1, $3);}
| NUM_LOG_BUFS equals pos_short_integer
{ $$ = make_node (nod_num_log_buffers, 1, $3);}
| LOG_BUF_SIZE equals unsigned_short_integer
{ $$ = make_node (nod_log_buffer_size, 1, $3);}
;
db_log : db_default_log_spec
2003-05-23 18:55:40 +02:00
{ if (lex.log_defined)
2003-09-04 23:26:15 +02:00
yyabandon (-260, isc_log_redef); /* Log redefined */
2003-05-23 18:55:40 +02:00
lex.log_defined = TRUE;
2001-05-23 15:26:42 +02:00
$$ = $1; }
| db_rem_log_spec
2003-05-23 18:55:40 +02:00
{ if (lex.log_defined)
2003-09-04 23:26:15 +02:00
yyabandon (-260, isc_log_redef);
2003-05-23 18:55:40 +02:00
lex.log_defined = TRUE;
2001-05-23 15:26:42 +02:00
$$ = $1; }
;
db_rem_log_spec : LOGFILE '(' logfiles ')' OVERFLOW logfile_desc
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_flags |= LOG_serial | LOG_overflow;
if (lex.g_file->fil_partitions)
2003-09-04 23:26:15 +02:00
yyabandon (-261, isc_partition_not_supp);
2001-05-23 15:26:42 +02:00
/* Partitions not supported in series of log file specification */
$$ = make_node (nod_list, 2, $3, $6); }
| LOGFILE BASENAME logfile_desc
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_flags |= LOG_serial;
if (lex.g_file->fil_partitions)
2003-09-04 23:26:15 +02:00
yyabandon (-261, isc_partition_not_supp);
2001-05-23 15:26:42 +02:00
$$ = $3; }
;
db_default_log_spec : LOGFILE
2003-05-23 18:55:40 +02:00
{ lex.g_file = make_file();
lex.g_file->fil_flags = LOG_serial | LOG_default;
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_log_file_desc, (int) 1,
2003-05-23 18:55:40 +02:00
(DSQL_NOD) lex.g_file);}
2001-05-23 15:26:42 +02:00
;
db_file : file1 sql_string file_desc1
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_name = (STR) $2;
2002-11-11 20:08:37 +01:00
$$ = (DSQL_NOD) make_node (nod_file_desc, (int) 1,
2003-05-23 18:55:40 +02:00
(DSQL_NOD) lex.g_file); }
2001-05-23 15:26:42 +02:00
;
/*
db_cache : CACHE sql_string cache_length
{
2003-05-23 18:55:40 +02:00
if (lex.cache_defined)
2003-09-04 23:26:15 +02:00
yyabandon (-260, isc_cache_redef);
2001-05-23 15:26:42 +02:00
*/ /* Cache redefined */ /*
2003-05-23 18:55:40 +02:00
lex.g_file = make_file();
lex.g_file->fil_length = (SLONG) $3;
lex.g_file->fil_name = (STR) $2;
lex.cache_defined = TRUE;
2002-11-11 20:08:37 +01:00
$$ = (DSQL_NOD) make_node (nod_cache_file_desc, (int) 1,
2003-05-23 18:55:40 +02:00
(DSQL_NOD) lex.g_file); }
2001-05-23 15:26:42 +02:00
;
*/
/*
cache_length :
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) (SLONG) DEF_CACHE_BUFFERS; }
2001-05-23 15:26:42 +02:00
| LENGTH equals long_integer page_noise
{ if ((SLONG) $3 < MIN_CACHE_BUFFERS)
2003-09-04 23:26:15 +02:00
yyabandon (-239, isc_cache_too_small);
2001-05-23 15:26:42 +02:00
*/ /* Cache length too small */ /*
else
2003-09-04 23:26:15 +02:00
$$ = (DSQL_NOD) $3; }
2001-05-23 15:26:42 +02:00
;
*/
logfiles : logfile_desc
| logfiles ',' logfile_desc
{ $$ = make_node (nod_list, 2, $1, $3); }
;
logfile_desc : logfile_name logfile_attrs
{
2003-09-04 23:26:15 +02:00
check_log_file_attrs();
2002-11-11 20:08:37 +01:00
$$ = (DSQL_NOD) make_node (nod_log_file_desc, (int) 1,
2003-09-04 23:26:15 +02:00
(DSQL_NOD) lex.g_file); }
2001-05-23 15:26:42 +02:00
;
logfile_name : sql_string
2003-05-23 18:55:40 +02:00
{ lex.g_file = make_file();
lex.g_file->fil_name = (STR) $1; }
2001-05-23 15:26:42 +02:00
;
logfile_attrs :
| logfile_attrs logfile_attr
;
logfile_attr : KW_SIZE equals long_integer
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_length = (SLONG) $3; }
2001-05-23 15:26:42 +02:00
/*
| RAW_PARTITIONS equals pos_short_integer
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_partitions = (SSHORT) $3;
lex.g_file->fil_flags |= LOG_raw; } */
2001-05-23 15:26:42 +02:00
;
file1 : KW_FILE
2003-05-23 18:55:40 +02:00
{ lex.g_file = make_file ();}
2001-05-23 15:26:42 +02:00
;
file_desc1 :
| file_desc
;
file_desc : file_clause
| file_desc file_clause
;
file_clause : STARTING file_clause_noise long_integer
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_start = (SLONG) $3;}
2001-05-23 15:26:42 +02:00
| LENGTH equals long_integer page_noise
2003-05-23 18:55:40 +02:00
{ lex.g_file->fil_length = (SLONG) $3;}
2001-05-23 15:26:42 +02:00
;
file_clause_noise :
| AT
| AT PAGE
;
page_noise :
| PAGE
| PAGES
;
/* CREATE TABLE */
table_clause : simple_table_name external_file '(' table_elements ')'
{ $$ = make_node (nod_def_relation,
(int) e_drl_count, $1, make_list ($4), $2); }
;
2002-06-29 08:56:51 +02:00
rtable_clause : simple_table_name external_file '(' table_elements ')'
{ $$ = make_node (nod_redef_relation,
(int) e_drl_count, $1, make_list ($4), $2); }
;
2001-05-23 15:26:42 +02:00
external_file : EXTERNAL KW_FILE sql_string
{ $$ = $3; }
| EXTERNAL sql_string
{ $$ = $2; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
table_elements : table_element
| table_elements ',' table_element
{ $$ = make_node (nod_list, 2, $1, $3); }
;
table_element : column_def
| table_constraint_definition
;
/* column definition */
column_def : column_def_name data_type_or_domain default_opt
end_trigger column_constraint_clause collate_clause
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
$1, $3, $4, make_list ($5), $6, $2, NULL); }
| column_def_name non_array_type def_computed
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
2003-09-04 23:26:15 +02:00
$1, NULL, NULL, NULL, NULL, NULL, $3); }
2001-05-23 15:26:42 +02:00
| column_def_name def_computed
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
2003-09-04 23:26:15 +02:00
$1, NULL, NULL, NULL, NULL, NULL, $2); }
2001-05-23 15:26:42 +02:00
;
2003-09-04 23:26:15 +02:00
2001-05-23 15:26:42 +02:00
/* value does allow parens around it, but there is a problem getting the
* source text
*/
def_computed : computed_by '(' begin_trigger value end_trigger ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_flags |= FLD_computed;
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_def_computed, 2, $4, $5); }
;
computed_by : COMPUTED BY
| COMPUTED
;
data_type_or_domain : data_type begin_trigger
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| simple_column_name begin_string
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_domain, (int) e_dom_count,
$1, NULL, NULL, NULL, NULL); }
;
2001-05-23 15:26:42 +02:00
collate_clause : COLLATE symbol_collation_name
{ $$ = $2; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
2002-06-29 08:56:51 +02:00
column_def_name : simple_column_name
2003-05-23 18:55:40 +02:00
{ lex.g_field_name = $1;
lex.g_field = make_field ($1);
$$ = (DSQL_NOD) lex.g_field; }
2001-05-23 15:26:42 +02:00
;
simple_column_def_name : simple_column_name
2003-05-23 18:55:40 +02:00
{ lex.g_field = make_field ($1);
$$ = (DSQL_NOD) lex.g_field; }
2001-05-23 15:26:42 +02:00
;
data_type_descriptor : init_data_type data_type
{ $$ = $1; }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
init_data_type :
2003-05-23 18:55:40 +02:00
{ lex.g_field = make_field (NULL);
$$ = (DSQL_NOD) lex.g_field; }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
default_opt : DEFAULT default_value
{ $$ = $2; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
default_value : constant
2002-06-29 08:56:51 +02:00
/* | USER
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_user_name, (int) 0, NULL); }
2002-06-29 08:56:51 +02:00
| CURRENT_USER
{ $$ = make_node (nod_user_name, (int) 0, NULL); }*/
| current_user
| current_role
| internal_info
{ $$ = $1; }
2001-05-23 15:26:42 +02:00
| null_value
{ $$ = $1; }
| datetime_value_expression
{ $$ = $1; }
;
2003-09-04 23:26:15 +02:00
2001-05-23 15:26:42 +02:00
column_constraint_clause :
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| column_constraint_list
;
column_constraint_list : column_constraint_def
2003-09-04 23:26:15 +02:00
| column_constraint_list column_constraint_def
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $2); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
column_constraint_def : constraint_name_opt column_constraint
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_rel_constraint, (int) 2, $1, $2);}
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
column_constraint : NOT KW_NULL
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_null, (int) 1, NULL); }
| REFERENCES simple_table_name column_parens_opt
referential_trigger_action constraint_index_opt
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_foreign, e_for_count,
make_node (nod_list, (int) 1, lex.g_field_name), $2, $3, $4, $5); }
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
| check_constraint
| UNIQUE constraint_index_opt
{ $$ = make_node (nod_unique, 2, NULL, $2); }
| PRIMARY KEY constraint_index_opt
{ $$ = make_node (nod_primary, e_pri_count, NULL, $3); }
2001-05-23 15:26:42 +02:00
;
2003-09-04 23:26:15 +02:00
2001-05-23 15:26:42 +02:00
/* table constraints */
table_constraint_definition : constraint_name_opt table_constraint
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_rel_constraint, (int) 2, $1, $2);}
;
2001-05-23 15:26:42 +02:00
constraint_name_opt : CONSTRAINT symbol_constraint_name
2003-09-04 23:26:15 +02:00
{ $$ = $2; }
| { $$ = NULL ;}
;
2001-05-23 15:26:42 +02:00
table_constraint : unique_constraint
| primary_constraint
| referential_constraint
| check_constraint
;
2001-05-23 15:26:42 +02:00
unique_constraint : UNIQUE column_parens constraint_index_opt
{ $$ = make_node (nod_unique, 2, $2, $3); }
;
2001-05-23 15:26:42 +02:00
primary_constraint : PRIMARY KEY column_parens constraint_index_opt
{ $$ = make_node (nod_primary, e_pri_count, $3, $4); }
2001-05-23 15:26:42 +02:00
;
referential_constraint : FOREIGN KEY column_parens REFERENCES
2001-05-23 15:26:42 +02:00
simple_table_name column_parens_opt
referential_trigger_action constraint_index_opt
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_foreign, e_for_count, $3, $5,
2003-09-04 23:26:15 +02:00
$6, $7, $8); }
;
constraint_index_opt : USING order_direction INDEX symbol_index_name
{ $$ = make_node (nod_def_index, (int) e_idx_count,
NULL, $2, $4, NULL, NULL); }
/*
| NO INDEX
{ $$ = NULL; }
*/
|
{ $$ = make_node (nod_def_index, (int) e_idx_count,
NULL, NULL, NULL, NULL, NULL); }
2001-05-23 15:26:42 +02:00
;
check_constraint : begin_trigger CHECK '(' search_condition ')' end_trigger
{ $$ = make_node (nod_def_constraint,
(int) e_cnstr_count, MAKE_string(NULL_STRING, 0), NULL,
2001-05-23 15:26:42 +02:00
NULL, NULL, $4, NULL, $6, NULL, NULL); }
;
referential_trigger_action:
update_rule
{ $$ = make_node (nod_ref_upd_del, e_ref_upd_del_count, $1, NULL);}
| delete_rule
{ $$ = make_node (nod_ref_upd_del, e_ref_upd_del_count, NULL, $1);}
| delete_rule update_rule
{ $$ = make_node (nod_ref_upd_del, e_ref_upd_del_count, $2, $1); }
| update_rule delete_rule
{ $$ = make_node (nod_ref_upd_del, e_ref_upd_del_count, $1, $2);}
| /* empty */
{ $$ = NULL;}
;
update_rule : ON UPDATE referential_action
{ $$ = $3;}
;
delete_rule : ON KW_DELETE referential_action
2001-05-23 15:26:42 +02:00
{ $$ = $3;}
;
referential_action: CASCADE
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_CASCADE, e_ref_trig_action_count, NULL);}
2003-09-04 23:26:15 +02:00
| SET DEFAULT
2001-05-23 15:26:42 +02:00
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_SET_DEFAULT, e_ref_trig_action_count, NULL);}
| SET KW_NULL
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_SET_NULL, e_ref_trig_action_count, NULL);}
| NO ACTION
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_NONE, e_ref_trig_action_count, NULL);}
;
/* PROCEDURE */
procedure_clause : symbol_procedure_name input_parameters
2003-09-04 23:26:15 +02:00
output_parameters
AS begin_string
2001-05-23 15:26:42 +02:00
var_declaration_list
full_proc_block
end_trigger
{ $$ = make_node (nod_def_procedure,
(int) e_prc_count,
2003-09-04 23:26:15 +02:00
$1, $2, $3, $6, $7, $8, NULL); }
;
2001-05-23 15:26:42 +02:00
2002-06-29 08:56:51 +02:00
rprocedure_clause : symbol_procedure_name input_parameters
2003-09-04 23:26:15 +02:00
output_parameters
AS begin_string
2002-06-29 08:56:51 +02:00
var_declaration_list
full_proc_block
end_trigger
{ $$ = make_node (nod_redef_procedure,
(int) e_prc_count,
2003-09-04 23:26:15 +02:00
$1, $2, $3, $6, $7, $8, NULL); }
;
2002-06-29 08:56:51 +02:00
replace_procedure_clause : symbol_procedure_name input_parameters
2003-09-04 23:26:15 +02:00
output_parameters
AS begin_string
var_declaration_list
full_proc_block
end_trigger
{ $$ = make_node (nod_replace_procedure,
(int) e_prc_count,
2003-09-04 23:26:15 +02:00
$1, $2, $3, $6, $7, $8, NULL); }
;
2001-05-23 15:26:42 +02:00
alter_procedure_clause : symbol_procedure_name input_parameters
2003-09-04 23:26:15 +02:00
output_parameters
AS begin_string
2001-05-23 15:26:42 +02:00
var_declaration_list
full_proc_block
end_trigger
{ $$ = make_node (nod_mod_procedure,
(int) e_prc_count,
2003-09-04 23:26:15 +02:00
$1, $2, $3, $6, $7, $8, NULL); }
;
2001-05-23 15:26:42 +02:00
input_parameters : '(' proc_parameters ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
output_parameters : RETURNS input_parameters
{ $$ = $2; }
|
{ $$ = NULL; }
;
proc_parameters : proc_parameter
| proc_parameters ',' proc_parameter
{ $$ = make_node (nod_list, 2, $1, $3); }
;
proc_parameter : simple_column_def_name non_array_type
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
$1, NULL, NULL, NULL, NULL, NULL, NULL); }
;
var_declaration_list : var_declarations
{ $$ = make_list ($1); }
|
{ $$ = NULL; }
;
var_declarations : var_declaration
| var_declarations var_declaration
{ $$ = make_node (nod_list, 2, $1, $2); }
;
var_declaration : DECLARE var_decl_opt column_def_name non_array_type var_init_opt ';'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
$3, $5, NULL, NULL, NULL, NULL, NULL); }
;
var_decl_opt : VARIABLE
{ $$ = NULL; }
|
{ $$ = NULL; }
;
var_init_opt : '=' default_value
{ $$ = $2; }
| default_opt
{ $$ = $1; }
2001-05-23 15:26:42 +02:00
;
proc_block : proc_statement
| full_proc_block
;
full_proc_block : BEGIN full_proc_block_body END
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
;
full_proc_block_body : proc_statements
{ $$ = make_node (nod_block, e_blk_count, make_list ($1), NULL); }
| proc_statements excp_hndl_statements
{ $$ = make_node (nod_block, e_blk_count, make_list ($1), make_list ($2)); }
|
{ $$ = make_node (nod_block, e_blk_count, NULL, NULL);}
;
2001-05-23 15:26:42 +02:00
proc_statements : proc_block
| proc_statements proc_block
{ $$ = make_node (nod_list, 2, $1, $2); }
;
proc_statement : assignment ';'
| delete ';'
| excp_statement
| raise_statement
2001-05-23 15:26:42 +02:00
| exec_procedure
2002-04-04 15:53:20 +02:00
| exec_sql
2001-05-23 15:26:42 +02:00
| for_select
| if_then_else
| insert ';'
| post_event
2001-05-23 15:26:42 +02:00
| singleton_select
| update ';'
| while
| for_exec_into
| exec_into
2001-05-23 15:26:42 +02:00
| SUSPEND ';'
{ $$ = make_node (nod_return, e_rtn_count, NULL); }
| EXIT ';'
{ $$ = make_node (nod_exit, 0, NULL); }
/* dimitr: commented out until pass1.cpp is ready to assign label numbers properly
| label
*/
| breakleave
2001-05-23 15:26:42 +02:00
;
excp_statement : EXCEPTION symbol_exception_name ';'
{ $$ = make_node (nod_exception_stmt, e_xcp_count, $2, NULL); }
| EXCEPTION symbol_exception_name value ';'
{ $$ = make_node (nod_exception_stmt, e_xcp_count, $2, $3); }
;
raise_statement : EXCEPTION ';'
{ $$ = make_node (nod_exception_stmt, e_xcp_count, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
exec_procedure : EXECUTE PROCEDURE symbol_procedure_name proc_inputs proc_outputs ';'
{ $$ = make_node (nod_exec_procedure, e_exe_count, $3,
$4, $5); }
;
exec_sql : EXECUTE varstate value ';'
{ $$ = make_node (nod_exec_sql, e_exec_sql_count, $3); }
2002-04-04 15:53:20 +02:00
;
varstate : VARCHAR | STATEMENT ;
for_select : label_opt FOR select INTO variable_list cursor_def DO proc_block
{ $$ = make_node (nod_for_select, e_flp_count, $3,
make_list ($5), $6, $8, $1); }
2001-05-23 15:26:42 +02:00
;
for_exec_into : label_opt FOR EXECUTE varstate value INTO variable_list DO proc_block
{ $$ = make_node (nod_exec_into, e_exec_into_count, $5, $9, make_list ($7), $1); }
;
exec_into : EXECUTE varstate value INTO variable_list ';'
{ $$ = make_node (nod_exec_into, e_exec_into_count, $3, 0, make_list ($5)); }
;
2001-05-23 15:26:42 +02:00
if_then_else : IF '(' search_condition ')' THEN proc_block ELSE proc_block
{ $$ = make_node (nod_if, e_if_count, $3, $6, $8); }
| IF '(' search_condition ')' THEN proc_block
{ $$ = make_node (nod_if, e_if_count, $3, $6, NULL); }
;
post_event : POST_EVENT value event_argument_opt ';'
{ $$ = make_node (nod_post, e_pst_count, $2, $3); }
;
2003-07-02 11:40:55 +02:00
event_argument_opt : /*',' value
{ $$ = $2; }
2003-07-02 11:40:55 +02:00
|*/
{ $$ = NULL; }
;
2001-05-23 15:26:42 +02:00
singleton_select : select INTO variable_list ';'
{ $$ = make_node (nod_for_select, e_flp_count, $1,
make_list ($3), NULL, NULL); }
;
variable : ':' symbol_variable_name
{ $$ = make_node (nod_var_name, (int) e_vrn_count,
$2); }
;
proc_inputs : null_or_value_list
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($1); }
| '(' null_or_value_list ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
proc_outputs : RETURNING_VALUES variable_list
{ $$ = make_list ($2); }
| RETURNING_VALUES '(' variable_list ')'
{ $$ = make_list ($3); }
|
{ $$ = NULL; }
;
variable_list : variable
| column_name
| variable_list ',' column_name
{ $$ = make_node (nod_list, 2, $1, $3); }
| variable_list ',' variable
{ $$ = make_node (nod_list, 2, $1, $3); }
;
while : label_opt WHILE '(' search_condition ')' DO proc_block
{ $$ = make_node (nod_while, e_while_count, $4, $7, $1); }
2001-05-23 15:26:42 +02:00
;
label_opt : symbol_label_name ':'
{ $$ = make_node (nod_label, e_label_count, $1, NULL); }
|
{ $$ = NULL; }
;
breakleave : KW_BREAK ';'
{ $$ = make_node (nod_breakleave, e_breakleave_count, NULL); }
| LEAVE ';'
{ $$ = make_node (nod_breakleave, e_breakleave_count, NULL); }
| LEAVE symbol_label_name ';'
{ $$ = make_node (nod_breakleave, e_breakleave_count,
make_node (nod_label, e_label_count, $2, NULL)); }
;
2001-05-23 15:26:42 +02:00
cursor_def : AS CURSOR symbol_cursor_name
{ $$ = make_node (nod_cursor, e_cur_count, $3, NULL, NULL); }
|
{ $$ = NULL; }
;
excp_hndl_statements : excp_hndl_statement
| excp_hndl_statements excp_hndl_statement
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_list, 2, $1, $2); }
;
excp_hndl_statement : WHEN errors DO proc_block
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_on_error, e_err_count,
make_list ($2), $4); }
;
errors : err
| errors ',' err
{ $$ = make_node (nod_list, 2, $1, $3); }
;
err : SQLCODE signed_short_integer
{ $$ = make_node (nod_sqlcode, 1, $2); }
| GDSCODE symbol_gdscode_name
{ $$ = make_node (nod_gdscode, 1, $2); }
| EXCEPTION symbol_exception_name
{ $$ = make_node (nod_exception, 1, $2); }
| ANY
{ $$ = make_node (nod_default, 1, NULL); }
;
/* Direct EXECUTE PROCEDURE */
invoke_procedure : EXECUTE PROCEDURE symbol_procedure_name proc_inputs
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_exec_procedure, e_exe_count, $3,
$4, make_node (nod_all, (int) 0, NULL)); }
;
/* CREATE VIEW */
view_clause : symbol_view_name column_parens_opt AS begin_string union_view
2003-09-04 23:26:15 +02:00
check_opt end_string
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_def_view, (int) e_view_count,
$1, $2, $5, $6, $7); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
2002-06-29 08:56:51 +02:00
rview_clause : symbol_view_name column_parens_opt AS begin_string union_view
2003-09-04 23:26:15 +02:00
check_opt end_string
2002-06-29 08:56:51 +02:00
{ $$ = make_node (nod_redef_view, (int) e_view_count,
$1, $2, $5, $6, $7); }
2003-09-04 23:26:15 +02:00
;
/*
replace_view_clause : symbol_view_name column_parens_opt AS begin_string union_view
2003-09-04 23:26:15 +02:00
check_opt end_string
{ $$ = make_node (nod_replace_view, (int) e_view_count,
$1, $2, $5, $6, $7); }
2003-09-04 23:26:15 +02:00
;
alter_view_clause : symbol_view_name column_parens_opt AS begin_string union_view
2003-09-04 23:26:15 +02:00
check_opt end_string
{ $$ = make_node (nod_mod_view, (int) e_view_count,
$1, $2, $5, $6, $7); }
2003-09-04 23:26:15 +02:00
;
*/
2003-09-04 23:26:15 +02:00
union_view : union_view_expr
{ $$ = make_node (nod_select, e_select_count, $1, NULL, NULL, NULL); }
2001-05-23 15:26:42 +02:00
;
union_view_expr : select_view_expr
{ $$ = make_node (nod_list, (int) 1, $1); }
| union_view_expr UNION select_view_expr
{ $$ = make_node (nod_list, 2, $1, $3); }
2003-09-04 23:26:15 +02:00
| union_view_expr UNION ALL select_view_expr
{ $$ = make_flag_node (nod_list, NOD_UNION_ALL, 2, $1, $4); }
2001-05-23 15:26:42 +02:00
;
select_view_expr: SELECT
2003-09-04 23:26:15 +02:00
distinct_clause
2001-05-23 15:26:42 +02:00
select_list
from_view_clause
where_clause
group_clause
having_clause
{ $$ = make_node (nod_select_expr, e_sel_count,
NULL, $2, $3, $4, $5, $6, $7, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
from_view_clause : FROM from_view_list
{ $$ = make_list ($2); }
;
from_view_list : view_table
| from_view_list ',' view_table
{ $$ = make_node (nod_list, 2, $1, $3); }
;
2003-08-15 02:02:18 +02:00
view_table : joined_view_table
2003-09-04 23:26:15 +02:00
| table_name
/* AB: Temporary disable derived tables in VIEWS
a derived table could hold a selectable SP for example, which by default isn't
allowed in VIEWs.
| derived_table
*/
2003-08-15 02:02:18 +02:00
;
2001-05-23 15:26:42 +02:00
joined_view_table : view_table join_type JOIN view_table ON search_condition
{ $$ = make_node (nod_join, (int) e_join_count,
$1, $2, $4, $6); }
| '(' joined_view_table ')'
{ $$ = $2; }
;
/* these rules will capture the input string for storage in metadata */
begin_string :
2003-05-23 18:55:40 +02:00
{ lex.beginning = lex_position(); }
2001-05-23 15:26:42 +02:00
;
end_string :
2003-05-23 18:55:40 +02:00
{ $$ = (DSQL_NOD) MAKE_string(lex.beginning,
2003-09-04 23:26:15 +02:00
(lex_position() == lex.end) ?
lex_position()-lex.beginning : lex.last_token-lex.beginning);}
2001-05-23 15:26:42 +02:00
;
begin_trigger :
2003-05-23 18:55:40 +02:00
{ lex.beginning = lex.last_token; }
2001-05-23 15:26:42 +02:00
;
end_trigger :
2003-05-23 18:55:40 +02:00
{ $$ = (DSQL_NOD) MAKE_string(lex.beginning,
lex_position()-lex.beginning); }
2001-05-23 15:26:42 +02:00
;
check_opt : WITH CHECK OPTION
{ $$ = make_node (nod_def_constraint, (int) e_cnstr_count,
MAKE_string(NULL_STRING, 0), NULL, NULL, NULL,
2001-05-23 15:26:42 +02:00
NULL, NULL, NULL, NULL, NULL); }
|
{ $$ = 0; }
;
/* CREATE TRIGGER */
def_trigger_clause : symbol_trigger_name FOR simple_table_name
trigger_active
trigger_type
trigger_position
begin_trigger
trigger_action
end_trigger
{ $$ = make_node (nod_def_trigger, (int) e_trg_count,
$1, $3, $4, $5, $6, $8, $9, NULL, NULL); }
2001-05-23 15:26:42 +02:00
;
replace_trigger_clause : symbol_trigger_name FOR simple_table_name
trigger_active
trigger_type
trigger_position
begin_trigger
trigger_action
end_trigger
{ $$ = make_node (nod_replace_trigger, (int) e_trg_count,
$1, $3, $4, $5, $6, $8, $9, NULL, NULL); }
;
2001-05-23 15:26:42 +02:00
trigger_active : ACTIVE
{ $$ = MAKE_constant ((STR) 0, CONSTANT_SLONG); }
| INACTIVE
{ $$ = MAKE_constant ((STR) 1, CONSTANT_SLONG); }
|
{ $$ = NULL; }
;
trigger_type : trigger_type_prefix trigger_type_suffix
{ $$ = MAKE_trigger_type ($1, $2); }
;
trigger_type_prefix : BEFORE
{ $$ = MAKE_constant ((STR) 0, CONSTANT_SLONG); }
| AFTER
2001-05-23 15:26:42 +02:00
{ $$ = MAKE_constant ((STR) 1, CONSTANT_SLONG); }
;
trigger_type_suffix : INSERT
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (1, 0, 0), CONSTANT_SLONG); }
| UPDATE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (2, 0, 0), CONSTANT_SLONG); }
| KW_DELETE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (3, 0, 0), CONSTANT_SLONG); }
| INSERT OR UPDATE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (1, 2, 0), CONSTANT_SLONG); }
| INSERT OR KW_DELETE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (1, 3, 0), CONSTANT_SLONG); }
| UPDATE OR INSERT
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (2, 1, 0), CONSTANT_SLONG); }
| UPDATE OR KW_DELETE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (2, 3, 0), CONSTANT_SLONG); }
| KW_DELETE OR INSERT
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (3, 1, 0), CONSTANT_SLONG); }
| KW_DELETE OR UPDATE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (3, 2, 0), CONSTANT_SLONG); }
| INSERT OR UPDATE OR KW_DELETE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (1, 2, 3), CONSTANT_SLONG); }
| INSERT OR KW_DELETE OR UPDATE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (1, 3, 2), CONSTANT_SLONG); }
| UPDATE OR INSERT OR KW_DELETE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (2, 1, 3), CONSTANT_SLONG); }
| UPDATE OR KW_DELETE OR INSERT
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (2, 3, 1), CONSTANT_SLONG); }
| KW_DELETE OR INSERT OR UPDATE
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (3, 1, 2), CONSTANT_SLONG); }
| KW_DELETE OR UPDATE OR INSERT
2003-10-15 00:22:32 +02:00
{ $$ = MAKE_constant ((STR) trigger_type_suffix (3, 2, 1), CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
trigger_position : POSITION nonneg_short_integer
{ $$ = MAKE_constant ((STR) $2, CONSTANT_SLONG); }
|
{ $$ = NULL; }
;
trigger_action : AS begin_trigger var_declaration_list full_proc_block
{ $$ = make_node (nod_list, 2, $3, $4); }
;
/* ALTER statement */
alter : ALTER alter_clause
{ $$ = $2; }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
alter_clause : EXCEPTION symbol_exception_name sql_string
{ $$ = make_node (nod_mod_exception, (int) e_xcp_count,
$2, $3); }
| TABLE simple_table_name alter_ops
{ $$ = make_node (nod_mod_relation, (int) e_alt_count,
$2, make_list ($3)); }
/*
| VIEW alter_view_clause
{ $$ = $2; }
*/
2001-05-23 15:26:42 +02:00
| TRIGGER alter_trigger_clause
{ $$ = $2; }
| PROCEDURE alter_procedure_clause
{ $$ = $2; }
| DATABASE init_alter_db alter_db
{ $$ = make_node (nod_mod_database, (int) e_adb_count,
make_list ($3)); }
2003-09-04 23:26:15 +02:00
| DOMAIN alter_column_name alter_domain_ops
{ $$ = make_node (nod_mod_domain, (int) e_alt_count,
$2, make_list ($3)); }
2001-05-23 15:26:42 +02:00
| INDEX alter_index_clause
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_mod_index,
(int) e_mod_idx_count, $2); }
2001-05-23 15:26:42 +02:00
;
2002-06-29 08:56:51 +02:00
domain_default_opt2 : DEFAULT begin_trigger default_value
{ $$ = $3; }
2003-06-05 14:37:49 +02:00
;
2002-06-29 08:56:51 +02:00
domain_check_constraint2 : CHECK begin_trigger '(' search_condition ')' end_trigger
{ $$ = make_node (nod_def_constraint,
(int) e_cnstr_count, MAKE_string(NULL_STRING, 0), NULL,
2002-06-29 08:56:51 +02:00
NULL, NULL, $4, NULL, $6, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
2002-06-29 08:56:51 +02:00
2001-05-23 15:26:42 +02:00
alter_domain_ops : alter_domain_op
| alter_domain_ops alter_domain_op
{ $$ = make_node (nod_list, 2, $1, $2); }
;
2002-06-29 08:56:51 +02:00
alter_domain_op : SET begin_trigger domain_default_opt2 end_trigger
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_default, (int) e_dft_count,
$3, $4); }
2002-06-29 08:56:51 +02:00
/* SET begin_string default_opt end_trigger
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_default, (int) e_dft_count,
$3, $4); }
| begin_trigger default_opt end_trigger
{ $$ = make_node (nod_def_default, (int) e_dft_count,
$2, $3); } */
| ADD CONSTRAINT domain_check_constraint2
{ $$ = $3; }
/* | ADD CONSTRAINT domain_check_constraint
{ $$ = $3; } */
| ADD domain_check_constraint
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| DROP DEFAULT
2003-09-04 23:26:15 +02:00
{$$ = make_node (nod_del_default, (int) 0, NULL); }
2001-05-23 15:26:42 +02:00
| DROP CONSTRAINT
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_delete_rel_constraint, (int) 1, NULL); }
2001-05-23 15:26:42 +02:00
| TO simple_column_name
{ $$ = $2; }
| TYPE init_data_type non_array_type
{ $$ = make_node (nod_mod_domain_type, 2, $2); }
;
alter_ops : alter_op
| alter_ops ',' alter_op
{ $$ = make_node (nod_list, 2, $1, $3); }
;
alter_op : DROP simple_column_name drop_behaviour
{ $$ = make_node (nod_del_field, 2, $2, $3); }
2003-09-04 23:26:15 +02:00
| DROP CONSTRAINT symbol_constraint_name
{ $$ = make_node (nod_delete_rel_constraint, (int) 1, $3);}
2001-05-23 15:26:42 +02:00
| ADD column_def
{ $$ = $2; }
2003-09-04 23:26:15 +02:00
| ADD table_constraint_definition
{ $$ = $2; }
2002-06-29 08:56:51 +02:00
/* CVC: From SQL, field positions start at 1, not zero. Think in ORDER BY, for example.
| col_opt simple_column_name POSITION nonneg_short_integer
{ $$ = make_node (nod_mod_field_pos, 2, $2,
2003-09-04 23:26:15 +02:00
MAKE_constant ((STR) $4, CONSTANT_SLONG)); } */
2002-06-29 08:56:51 +02:00
| col_opt simple_column_name POSITION pos_short_integer
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_mod_field_pos, 2, $2,
MAKE_constant ((STR) $4, CONSTANT_SLONG)); }
| col_opt alter_column_name TO simple_column_name
{ $$ = make_node (nod_mod_field_name, 2, $2, $4); }
| col_opt alter_col_name TYPE alter_data_type_or_domain end_trigger
{ $$ = make_node (nod_mod_field_type, 3, $2, $5, $4); }
;
alter_column_name : keyword_or_column
{ $$ = make_node (nod_field_name, (int) e_fln_count,
NULL, $1); }
;
2003-04-09 12:18:54 +02:00
/* below are reserved words that could be used as column identifiers
in the previous versions */
keyword_or_column : valid_symbol_name
| COLUMN /* added in IB 6.0 */
| TYPE
| EXTRACT
| YEAR
| MONTH
| DAY
| HOUR
| MINUTE
| SECOND
| WEEKDAY
| YEARDAY
| TIME
| TIMESTAMP
| CURRENT_DATE
| CURRENT_TIME
2003-04-09 12:51:42 +02:00
| CURRENT_TIMESTAMP
| CURRENT_USER /* added in FB 1.0 */
| CURRENT_ROLE
| CURRENT_CONNECTION /* added in FB 1.5 */
| CURRENT_TRANSACTION
| ROW_COUNT
| SAVEPOINT
;
2001-05-23 15:26:42 +02:00
col_opt : ALTER
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| ALTER COLUMN
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
alter_data_type_or_domain : non_array_type begin_trigger
2003-09-04 23:26:15 +02:00
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
| simple_column_name begin_string
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_def_domain, (int) e_dom_count,
$1, NULL, NULL, NULL, NULL); }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
alter_col_name : simple_column_name
2003-05-23 18:55:40 +02:00
{ lex.g_field_name = $1;
lex.g_field = make_field ($1);
$$ = (DSQL_NOD) lex.g_field; }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
drop_behaviour : RESTRICT
{ $$ = make_node (nod_restrict, 0, NULL); }
| CASCADE
{ $$ = make_node (nod_cascade, 0, NULL); }
|
{ $$ = make_node (nod_restrict, 0, NULL); }
;
alter_index_clause : symbol_index_name ACTIVE
{ $$ = make_node (nod_idx_active, 1, $1); }
| symbol_index_name INACTIVE
{ $$ = make_node (nod_idx_inactive, 1, $1); }
;
/* ALTER DATABASE */
init_alter_db :
2003-05-23 18:55:40 +02:00
{ lex.log_defined = FALSE;
lex.cache_defined = FALSE;
$$ = NULL; }
2001-05-23 15:26:42 +02:00
;
alter_db : db_alter_clause
| alter_db db_alter_clause
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $2); }
2001-05-23 15:26:42 +02:00
;
db_alter_clause : ADD db_file_list
{ $$ = $2; }
/*
| ADD db_cache
{ $$ = $2; }
2003-09-04 23:26:15 +02:00
| DROP CACHE
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_drop_cache, (int) 0, NULL); }
*/
| DROP LOGFILE
{ $$ = make_node (nod_drop_log, (int) 0, NULL); }
| SET db_log_option_list
{ $$ = $2; }
| ADD db_log
{ $$ = $2; }
| ADD KW_DIFFERENCE KW_FILE sql_string
{ $$ = make_node (nod_difference_file, (int) 1, $4); }
| DROP KW_DIFFERENCE KW_FILE
{ $$ = make_node (nod_drop_difference, (int) 0, NULL); }
| BEGIN BACKUP
{ $$ = make_node (nod_begin_backup, (int) 0, NULL); }
| END BACKUP
{ $$ = make_node (nod_end_backup, (int) 0, NULL); }
2001-05-23 15:26:42 +02:00
;
db_log_option_list : db_log_option
| db_log_option_list ',' db_log_option
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
2001-05-23 15:26:42 +02:00
;
/* ALTER TRIGGER */
alter_trigger_clause : symbol_trigger_name trigger_active
new_trigger_type
trigger_position
begin_trigger
new_trigger_action
end_trigger
{ $$ = make_node (nod_mod_trigger, (int) e_trg_count,
$1, NULL, $2, $3, $4, $6, $7, NULL, NULL); }
2001-05-23 15:26:42 +02:00
;
new_trigger_type : trigger_type
|
{ $$ = NULL; }
;
new_trigger_action : trigger_action
|
{ $$ = NULL; }
;
/* DROP metadata operations */
2003-09-04 23:26:15 +02:00
2001-05-23 15:26:42 +02:00
drop : DROP drop_clause
{ $$ = $2; }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
drop_clause : EXCEPTION symbol_exception_name
{ $$ = make_node (nod_del_exception, 1, $2); }
| INDEX symbol_index_name
{ $$ = make_node (nod_del_index, (int) 1, $2); }
| PROCEDURE symbol_procedure_name
{ $$ = make_node (nod_del_procedure, (int) 1, $2); }
| TABLE symbol_table_name
{ $$ = make_node (nod_del_relation, (int) 1, $2); }
| TRIGGER symbol_trigger_name
{ $$ = make_node (nod_del_trigger, (int) 1, $2); }
| VIEW symbol_view_name
2002-06-29 08:56:51 +02:00
{ $$ = make_node (nod_del_view, (int) 1, $2); }
2001-05-23 15:26:42 +02:00
| FILTER symbol_filter_name
{ $$ = make_node (nod_del_filter, (int) 1, $2); }
| DOMAIN symbol_domain_name
{ $$ = make_node (nod_del_domain, (int) 1, $2); }
| EXTERNAL FUNCTION symbol_UDF_name
{ $$ = make_node (nod_del_udf, (int) 1, $3); }
| SHADOW pos_short_integer
{ $$ = make_node (nod_del_shadow, (int) 1, $2); }
| ROLE symbol_role_name
{ $$ = make_node (nod_del_role, (int) 1, $2); }
2002-06-29 08:56:51 +02:00
| GENERATOR symbol_generator_name
{ $$ = make_node (nod_del_generator, (int) 1, $2); }
2001-05-23 15:26:42 +02:00
;
/* these are the allowable datatypes */
data_type : non_array_type
| array_type
;
non_array_type : simple_type
| blob_type
;
array_type : non_charset_simple_type '[' array_spec ']'
2003-09-04 23:26:15 +02:00
{ lex.g_field->fld_ranges = make_list ($3);
lex.g_field->fld_dimensions = lex.g_field->fld_ranges->nod_count / 2;
lex.g_field->fld_element_dtype = lex.g_field->fld_dtype;
$$ = $1; }
2001-05-23 15:26:42 +02:00
| character_type '[' array_spec ']' charset_clause
2003-09-04 23:26:15 +02:00
{ lex.g_field->fld_ranges = make_list ($3);
lex.g_field->fld_dimensions = lex.g_field->fld_ranges->nod_count / 2;
lex.g_field->fld_element_dtype = lex.g_field->fld_dtype;
$$ = $1; }
2001-05-23 15:26:42 +02:00
;
array_spec : array_range
| array_spec ',' array_range
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
;
array_range : signed_long_integer
2003-09-04 23:26:15 +02:00
{ if ((SLONG) $1 < 1)
$$ = make_node (nod_list, (int) 2,
2001-05-23 15:26:42 +02:00
MAKE_constant ((STR) $1, CONSTANT_SLONG),
MAKE_constant ((STR) 1, CONSTANT_SLONG));
2003-09-04 23:26:15 +02:00
else
$$ = make_node (nod_list, (int) 2,
2001-05-23 15:26:42 +02:00
MAKE_constant ((STR) 1, CONSTANT_SLONG),
MAKE_constant ((STR) $1, CONSTANT_SLONG) ); }
| signed_long_integer ':' signed_long_integer
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2,
2001-05-23 15:26:42 +02:00
MAKE_constant ((STR) $1, CONSTANT_SLONG),
MAKE_constant ((STR) $3, CONSTANT_SLONG)); }
;
simple_type : non_charset_simple_type
| character_type charset_clause
;
non_charset_simple_type : national_character_type
| numeric_type
| float_type
| BIGINT
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "BIGINT",
0);
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "BIGINT",
0);
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_int64;
lex.g_field->fld_length = sizeof (SINT64);
}
2001-05-23 15:26:42 +02:00
| integer_keyword
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_long;
lex.g_field->fld_length = sizeof (SLONG);
2001-05-23 15:26:42 +02:00
}
| SMALLINT
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_short;
lex.g_field->fld_length = sizeof (SSHORT);
2001-05-23 15:26:42 +02:00
}
| DATE
{
*stmt_ambiguous = TRUE;
if (client_dialect <= SQL_DIALECT_V5)
2003-09-04 23:26:15 +02:00
{
/* Post warning saying that DATE is equivalent to TIMESTAMP */
ERRD_post_warning (isc_sqlwarn, gds_arg_number, (SLONG) 301,
isc_arg_warning, isc_dtype_renamed, 0);
lex.g_field->fld_dtype = dtype_timestamp;
lex.g_field->fld_length = sizeof (GDS_TIMESTAMP);
}
2001-05-23 15:26:42 +02:00
else if (client_dialect == SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
yyabandon (-104, isc_transitional_date);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
{
lex.g_field->fld_dtype = dtype_sql_date;
lex.g_field->fld_length = sizeof (ULONG);
}
2001-05-23 15:26:42 +02:00
}
| TIME
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "TIME",
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "TIME",
0);
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_sql_time;
lex.g_field->fld_length = sizeof (SLONG);
2001-05-23 15:26:42 +02:00
}
| TIMESTAMP
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_timestamp;
lex.g_field->fld_length = sizeof (GDS_TIMESTAMP);
2001-05-23 15:26:42 +02:00
}
;
integer_keyword : INTEGER
| KW_INT
;
/* allow a blob to be specified with any combination of
segment length and subtype */
blob_type : BLOB blob_subtype blob_segsize charset_clause
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_blob;
2001-05-23 15:26:42 +02:00
}
| BLOB '(' unsigned_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_blob;
lex.g_field->fld_seg_length = (USHORT)(ULONG) $3;
2003-05-23 18:55:40 +02:00
lex.g_field->fld_sub_type = 0;
2001-05-23 15:26:42 +02:00
}
| BLOB '(' unsigned_short_integer ',' signed_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_blob;
lex.g_field->fld_seg_length = (USHORT)(ULONG) $3;
lex.g_field->fld_sub_type = (USHORT)(ULONG) $5;
2001-05-23 15:26:42 +02:00
}
| BLOB '(' ',' signed_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_blob;
lex.g_field->fld_seg_length = 80;
lex.g_field->fld_sub_type = (USHORT)(ULONG) $4;
2001-05-23 15:26:42 +02:00
}
;
blob_segsize : SEGMENT KW_SIZE unsigned_short_integer
2001-05-23 15:26:42 +02:00
{
lex.g_field->fld_seg_length = (USHORT)(ULONG) $3;
2001-05-23 15:26:42 +02:00
}
|
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_seg_length = (USHORT) 80;
2001-05-23 15:26:42 +02:00
}
;
blob_subtype : SUB_TYPE signed_short_integer
{
lex.g_field->fld_sub_type = (USHORT)(ULONG) $2;
2001-05-23 15:26:42 +02:00
}
| SUB_TYPE symbol_blob_subtype_name
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_sub_type_name = $2;
2001-05-23 15:26:42 +02:00
}
|
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_sub_type = (USHORT) 0;
2001-05-23 15:26:42 +02:00
}
;
charset_clause : CHARACTER SET symbol_character_set_name
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_character_set = $3;
2001-05-23 15:26:42 +02:00
}
|
;
/* character type */
national_character_type : national_character_keyword '(' pos_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_text;
lex.g_field->fld_character_length = (USHORT)(ULONG) $3;
2003-05-23 18:55:40 +02:00
lex.g_field->fld_flags |= FLD_national;
2001-05-23 15:26:42 +02:00
}
| national_character_keyword
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_text;
lex.g_field->fld_character_length = 1;
lex.g_field->fld_flags |= FLD_national;
2001-05-23 15:26:42 +02:00
}
| national_character_keyword VARYING '(' pos_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_varying;
lex.g_field->fld_character_length = (USHORT)(ULONG) $4;
2003-05-23 18:55:40 +02:00
lex.g_field->fld_flags |= FLD_national;
2001-05-23 15:26:42 +02:00
}
;
character_type : character_keyword '(' pos_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_text;
lex.g_field->fld_character_length = (USHORT)(ULONG) $3;
2001-05-23 15:26:42 +02:00
}
| character_keyword
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_text;
lex.g_field->fld_character_length = 1;
2001-05-23 15:26:42 +02:00
}
| varying_keyword '(' pos_short_integer ')'
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_varying;
lex.g_field->fld_character_length = (USHORT)(ULONG) $3;
2001-05-23 15:26:42 +02:00
}
;
varying_keyword : VARCHAR
| CHARACTER VARYING
| KW_CHAR VARYING
;
character_keyword : CHARACTER
| KW_CHAR
;
national_character_keyword : NCHAR
| NATIONAL CHARACTER
2003-09-04 23:26:15 +02:00
| NATIONAL KW_CHAR
;
2001-05-23 15:26:42 +02:00
/* numeric type */
numeric_type : KW_NUMERIC prec_scale
2003-09-04 23:26:15 +02:00
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_sub_type = dsc_num_type_numeric;
2001-05-23 15:26:42 +02:00
}
| decimal_keyword prec_scale
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_sub_type = dsc_num_type_decimal;
if (lex.g_field->fld_dtype == dtype_short)
2001-05-23 15:26:42 +02:00
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_long;
lex.g_field->fld_length = sizeof (SLONG);
2001-05-23 15:26:42 +02:00
};
}
;
prec_scale :
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_long;
2003-09-04 23:26:15 +02:00
lex.g_field->fld_length = sizeof (SLONG);
2003-05-23 18:55:40 +02:00
lex.g_field->fld_precision = 9;
2003-09-04 23:26:15 +02:00
}
2001-05-23 15:26:42 +02:00
| '(' signed_long_integer ')'
2003-09-04 23:26:15 +02:00
{
2001-05-23 15:26:42 +02:00
if ( ((SLONG) $2 < 1) || ((SLONG) $2 > 18) )
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_precision_err);
2001-05-23 15:26:42 +02:00
/* Precision most be between 1 and 18. */
if ((SLONG) $2 > 9)
2003-09-04 23:26:15 +02:00
{
if ( ( (client_dialect <= SQL_DIALECT_V5) &&
(db_dialect > SQL_DIALECT_V5) ) ||
2001-05-23 15:26:42 +02:00
( (client_dialect > SQL_DIALECT_V5) &&
2003-09-04 23:26:15 +02:00
(db_dialect <= SQL_DIALECT_V5) ) )
ERRD_post (gds_sqlerr,
2001-05-23 15:26:42 +02:00
gds_arg_number, (SLONG) -817,
gds_arg_gds,
isc_ddl_not_allowed_by_db_sql_dial,
gds_arg_number, (SLONG) db_dialect,
0);
2003-09-04 23:26:15 +02:00
if (client_dialect <= SQL_DIALECT_V5)
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_double;
lex.g_field->fld_length = sizeof (double);
2003-09-04 23:26:15 +02:00
}
else
{
2001-05-23 15:26:42 +02:00
if (client_dialect == SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
{
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous,
gds_arg_end );
2003-09-04 23:26:15 +02:00
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous1,
gds_arg_end );
2003-09-04 23:26:15 +02:00
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous2,
gds_arg_end );
2003-09-04 23:26:15 +02:00
}
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_int64;
lex.g_field->fld_length = sizeof (SINT64);
2003-09-04 23:26:15 +02:00
}
}
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
if ((SLONG) $2 < 5)
{
lex.g_field->fld_dtype = dtype_short;
lex.g_field->fld_length = sizeof (SSHORT);
}
else
{
lex.g_field->fld_dtype = dtype_long;
lex.g_field->fld_length = sizeof (SLONG);
}
lex.g_field->fld_precision = (USHORT)(ULONG) $2;
2001-05-23 15:26:42 +02:00
}
| '(' signed_long_integer ',' signed_long_integer ')'
{
if ( ((SLONG) $2 < 1) || ((SLONG) $2 > 18) )
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_precision_err);
2001-05-23 15:26:42 +02:00
/* Precision should be between 1 and 18 */
if (((SLONG) $4 > (SLONG) $2) || ((SLONG) $4 < 0))
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_scale_nogt);
2001-05-23 15:26:42 +02:00
/* Scale must be between 0 and precision */
if ((SLONG) $2 > 9)
2003-09-04 23:26:15 +02:00
{
if ( ( (client_dialect <= SQL_DIALECT_V5) &&
(db_dialect > SQL_DIALECT_V5) ) ||
2001-05-23 15:26:42 +02:00
( (client_dialect > SQL_DIALECT_V5) &&
2003-09-04 23:26:15 +02:00
(db_dialect <= SQL_DIALECT_V5) ) )
ERRD_post (gds_sqlerr,
2001-05-23 15:26:42 +02:00
gds_arg_number, (SLONG) -817,
gds_arg_gds,
isc_ddl_not_allowed_by_db_sql_dial,
gds_arg_number, (SLONG) db_dialect,
0);
2003-09-04 23:26:15 +02:00
if (client_dialect <= SQL_DIALECT_V5)
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_double;
lex.g_field->fld_length = sizeof (double);
2003-09-04 23:26:15 +02:00
}
else
{
2001-05-23 15:26:42 +02:00
if (client_dialect == SQL_DIALECT_V6_TRANSITION)
{
2003-09-04 23:26:15 +02:00
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous,
gds_arg_end );
2003-09-04 23:26:15 +02:00
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous1,
gds_arg_end );
2003-09-04 23:26:15 +02:00
ERRD_post_warning (
2001-05-23 15:26:42 +02:00
isc_dsql_warn_precision_ambiguous2,
gds_arg_end );
}
/* client_dialect >= SQL_DIALECT_V6 */
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_int64;
lex.g_field->fld_length = sizeof (SINT64);
2003-09-04 23:26:15 +02:00
}
}
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
{
if ((SLONG) $2 < 5)
{
lex.g_field->fld_dtype = dtype_short;
lex.g_field->fld_length = sizeof (SSHORT);
}
else
{
lex.g_field->fld_dtype = dtype_long;
lex.g_field->fld_length = sizeof (SLONG);
}
}
lex.g_field->fld_precision = (USHORT)(ULONG) $2;
lex.g_field->fld_scale = - (SSHORT)(SLONG) $4;
2001-05-23 15:26:42 +02:00
}
;
decimal_keyword : DECIMAL
| KW_DEC
;
/* floating point type */
float_type : KW_FLOAT precision_opt
{
if ((SLONG) $2 > 7)
2003-09-04 23:26:15 +02:00
{
lex.g_field->fld_dtype = dtype_double;
lex.g_field->fld_length = sizeof (double);
}
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
{
lex.g_field->fld_dtype = dtype_real;
lex.g_field->fld_length = sizeof (float);
}
2001-05-23 15:26:42 +02:00
}
| KW_LONG KW_FLOAT precision_opt
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_double;
lex.g_field->fld_length = sizeof (double);
2001-05-23 15:26:42 +02:00
}
| REAL
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_real;
lex.g_field->fld_length = sizeof (float);
2001-05-23 15:26:42 +02:00
}
| KW_DOUBLE PRECISION
{
2003-05-23 18:55:40 +02:00
lex.g_field->fld_dtype = dtype_double;
lex.g_field->fld_length = sizeof (double);
2001-05-23 15:26:42 +02:00
}
;
precision_opt : '(' nonneg_short_integer ')'
{ $$ = $2; }
|
{ $$ = 0; }
;
/* SET statements */
set : set_transaction
| set_generator
| set_statistics
;
set_generator : SET GENERATOR symbol_generator_name TO signed_long_integer
{
$$ = make_node (nod_set_generator2,e_gen_id_count,$3,
MAKE_constant ((STR) $5, CONSTANT_SLONG));
}
2003-09-04 23:26:15 +02:00
| SET GENERATOR symbol_generator_name TO NUMBER64BIT
{
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_set_generator2,e_gen_id_count,$3,
2003-09-04 23:26:15 +02:00
MAKE_constant((STR)$5, CONSTANT_SINT64));
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
| SET GENERATOR symbol_generator_name TO '-' NUMBER64BIT
{
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_set_generator2, e_gen_id_count, $3,
make_node(nod_negate, 1,
2003-09-04 23:26:15 +02:00
MAKE_constant((STR)$6, CONSTANT_SINT64)));
2001-05-23 15:26:42 +02:00
}
;
/* transaction statements */
savepoint : set_savepoint
| release_savepoint
| undo_savepoint
;
set_savepoint : SAVEPOINT symbol_savepoint_name
{ $$ = make_node (nod_user_savepoint, 1, $2); }
;
release_savepoint : RELEASE SAVEPOINT symbol_savepoint_name release_only_opt
{ $$ = make_node (nod_release_savepoint, 2, $3, $4); }
;
release_only_opt : ONLY
{ $$ = make_node (nod_flag, 0, NULL); }
|
{ $$ = 0; }
;
undo_savepoint : ROLLBACK optional_work TO optional_savepoint symbol_savepoint_name
{ $$ = make_node (nod_undo_savepoint, 1, $5); }
;
optional_savepoint : SAVEPOINT
|
;
2001-05-23 15:26:42 +02:00
commit : COMMIT optional_work optional_retain
{ $$ = make_node (nod_commit, 1, $3); }
;
rollback : ROLLBACK optional_work
{ $$ = make_node (nod_rollback, 0, NULL); }
;
optional_work : WORK
|
;
optional_retain : RETAIN opt_snapshot
{ $$ = make_node (nod_commit_retain, 0, NULL); }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
opt_snapshot : SNAPSHOT
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
set_transaction : SET TRANSACTION tran_opt_list_m
{$$ = make_node (nod_trans, 1, make_list ($3)); }
;
tran_opt_list_m : tran_opt_list
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
tran_opt_list : tran_opt
| tran_opt_list tran_opt
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $2); }
2001-05-23 15:26:42 +02:00
;
tran_opt : access_mode
| lock_wait
| isolation_mode
| tbl_reserve_options
;
access_mode : READ ONLY
{ $$ = make_flag_node (nod_access, NOD_READ_ONLY, (int) 0, NULL); }
| READ WRITE
{ $$ = make_flag_node (nod_access, NOD_READ_WRITE, (int) 0, NULL); }
;
lock_wait : WAIT
{ $$ = make_flag_node (nod_wait, NOD_WAIT, (int) 0, NULL); }
| NO WAIT
{ $$ = make_flag_node (nod_wait, NOD_NO_WAIT, (int) 0, NULL); }
;
isolation_mode : ISOLATION LEVEL iso_mode
{ $$ = $3;}
| iso_mode
;
iso_mode : snap_shot
{ $$ = $1;}
| READ UNCOMMITTED version_mode
2001-05-23 15:26:42 +02:00
{ $$ = make_flag_node (nod_isolation, NOD_READ_COMMITTED, 1, $3); }
| READ COMMITTED version_mode
2001-05-23 15:26:42 +02:00
{ $$ = make_flag_node (nod_isolation, NOD_READ_COMMITTED, 1, $3); }
;
snap_shot : SNAPSHOT
{ $$ = make_flag_node (nod_isolation, NOD_CONCURRENCY, 0, NULL); }
| SNAPSHOT TABLE
{ $$ = make_flag_node (nod_isolation, NOD_CONSISTENCY, 0, NULL); }
| SNAPSHOT TABLE STABILITY
{ $$ = make_flag_node (nod_isolation, NOD_CONSISTENCY, 0, NULL); }
;
version_mode : VERSION
{ $$ = make_flag_node (nod_version, NOD_VERSION, 0, NULL); }
| NO VERSION
{ $$ = make_flag_node (nod_version, NOD_NO_VERSION, 0, NULL); }
|
{ $$ = 0; }
;
tbl_reserve_options: RESERVING restr_list
{ $$ = make_node (nod_reserve, 1, make_list ($2)); }
;
lock_type : KW_SHARED
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) NOD_SHARED; }
2001-05-23 15:26:42 +02:00
| PROTECTED
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) NOD_PROTECTED ; }
2001-05-23 15:26:42 +02:00
|
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) 0; }
2001-05-23 15:26:42 +02:00
;
lock_mode : READ
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) NOD_READ; }
2001-05-23 15:26:42 +02:00
| WRITE
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) NOD_WRITE; }
2001-05-23 15:26:42 +02:00
;
restr_list : restr_option
| restr_list ',' restr_option
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
2001-05-23 15:26:42 +02:00
;
restr_option : table_list table_lock
{ $$ = make_node (nod_table_lock, (int) 2, make_list ($1), $2); }
;
table_lock : FOR lock_type lock_mode
{ $$ = make_flag_node (nod_lock_mode, (SSHORT) ((SSHORT)(SLONG) $2 | (SSHORT)(SLONG) $3), (SSHORT) 0, NULL); }
2001-05-23 15:26:42 +02:00
|
{ $$ = 0; }
;
table_list : simple_table_name
| table_list ',' simple_table_name
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2, $1, $3); }
2001-05-23 15:26:42 +02:00
;
set_statistics : SET STATISTICS INDEX symbol_index_name
{$$ = make_node (nod_set_statistics,
(int)e_stat_count, $4); }
2003-06-05 14:37:49 +02:00
;
2001-05-23 15:26:42 +02:00
/* SELECT statement */
select : union_expr order_clause for_update_clause lock_clause
{ $$ = make_node (nod_select, e_select_count, $1, $2, $3, $4); }
2001-05-23 15:26:42 +02:00
;
union_expr : select_expr
{ $$ = make_node (nod_list, 1, $1); }
| union_expr UNION select_expr
{ $$ = make_node (nod_list, 2, $1, $3); }
| union_expr UNION ALL select_expr
{ $$ = make_flag_node (nod_list, NOD_UNION_ALL, 2, $1, $4); }
2001-05-23 15:26:42 +02:00
;
order_clause : ORDER BY order_list
{ $$ = make_list ($3); }
|
{ $$ = 0; }
;
order_list : order_item
| order_list ',' order_item
{ $$ = make_node (nod_list, 2, $1, $3); }
;
2003-05-23 18:55:40 +02:00
order_item : value order_direction nulls_clause
{ $$ = make_node (nod_order, e_order_count, $1, $2, $3); }
2001-05-23 15:26:42 +02:00
;
order_direction : ASC
{ $$ = 0; }
| DESC
{ $$ = make_node (nod_flag, 0, NULL); }
|
{ $$ = 0; }
;
2003-05-23 18:55:40 +02:00
nulls_placement : FIRST
2002-09-10 20:28:23 +02:00
{ $$ = make_node (nod_flag, 0, NULL); }
2003-05-23 18:55:40 +02:00
| LAST
2002-09-10 20:28:23 +02:00
{ $$ = 0; }
2003-05-23 18:55:40 +02:00
;
nulls_clause : NULLS begin_first nulls_placement end_first
{ $$ = $3; }
2002-09-10 20:28:23 +02:00
|
{ $$ = 0; }
;
for_update_clause : FOR UPDATE for_update_list
{ $$ = make_node (nod_for_update, 1, $3); }
2001-05-23 15:26:42 +02:00
|
{ $$ = 0; }
;
for_update_list : OF column_list
{ $$ = $2; }
|
{ $$ = make_node (nod_flag, 0, NULL); }
;
2002-11-01 09:53:05 +01:00
lock_clause : WITH LOCK
{ $$ = make_node (nod_flag, 0, NULL); }
|
{ $$ = 0; }
;
2001-05-23 15:26:42 +02:00
/* SELECT expression */
2002-06-29 08:56:51 +02:00
select_expr : SELECT limit_clause
2003-09-04 23:26:15 +02:00
distinct_clause
2001-05-23 15:26:42 +02:00
select_list
from_clause
where_clause
group_clause
having_clause
plan_clause
{ $$ = make_node (nod_select_expr, e_sel_count,
$2, $3, $4, $5, $6, $7, $8, $9, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
ordered_select_expr : SELECT limit_clause
2003-09-04 23:26:15 +02:00
distinct_clause
select_list
from_clause
where_clause
group_clause
having_clause
plan_clause
order_clause
{ $$ = make_node (nod_select_expr, e_sel_count,
$2, $3, $4, $5, $6, $7, $8, $9, $10, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
2003-05-23 18:55:40 +02:00
begin_limit :
{ lex.limit_clause = true; }
;
end_limit :
{ lex.limit_clause = false; }
;
begin_first :
{ lex.first_detection = true; }
;
end_first :
{ lex.first_detection = false; }
;
limit_clause : first_clause skip_clause end_limit
2002-11-01 09:53:05 +01:00
{ $$ = make_node (nod_limit, e_limit_count, $2, $1); }
2003-05-23 18:55:40 +02:00
| first_clause end_limit
2002-11-01 09:53:05 +01:00
{ $$ = make_node (nod_limit, e_limit_count, NULL, $1); }
| skip_clause
{ $$ = make_node (nod_limit, e_limit_count, $1, NULL); }
|
{ $$ = 0; }
;
2003-05-23 18:55:40 +02:00
first_clause : FIRST long_integer begin_limit
2002-11-01 09:53:05 +01:00
{ $$ = MAKE_constant ((STR) $2, CONSTANT_SLONG); }
2003-05-23 18:55:40 +02:00
| FIRST '(' value ')' begin_limit
2002-11-01 09:53:05 +01:00
{ $$ = $3; }
2003-05-23 18:55:40 +02:00
| FIRST parameter begin_limit
2002-11-01 09:53:05 +01:00
{ $$ = $2; }
;
skip_clause : SKIP long_integer
{ $$ = MAKE_constant ((STR) $2, CONSTANT_SLONG); }
2003-05-23 18:55:40 +02:00
| SKIP '(' end_limit value ')'
{ $$ = $4; }
2002-11-01 09:53:05 +01:00
| SKIP parameter
{ $$ = $2; }
;
2001-05-23 15:26:42 +02:00
distinct_clause : DISTINCT
2002-06-29 08:56:51 +02:00
{ $$ = make_node (nod_flag, 0, NULL); }
2001-05-23 15:26:42 +02:00
| all_noise
{ $$ = 0; }
;
select_list : select_items
{ $$ = make_list ($1); }
| '*'
{ $$ = 0; }
;
select_items : select_item
| select_items ',' select_item
{ $$ = make_node (nod_list, 2, $1, $3); }
;
select_item : rhs
2003-08-15 02:02:18 +02:00
| rhs as_noise symbol_item_alias_name
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_alias, 2, $1, $3); }
;
2003-08-15 02:02:18 +02:00
as_noise : AS
|
;
2001-05-23 15:26:42 +02:00
/* FROM clause */
from_clause : FROM from_list
{ $$ = make_list ($2); }
;
from_list : table_reference
| from_list ',' table_reference
{ $$ = make_node (nod_list, 2, $1, $3); }
;
table_reference : joined_table
| table_proc
| derived_table
;
2003-08-15 02:02:18 +02:00
/* AB: derived table support */
derived_table :
'(' select ')' as_noise correlation_name derived_column_list
2003-08-15 02:02:18 +02:00
{ $$ = make_node(nod_derived_table, (int) e_derived_table_count, $2, $5, $6); }
;
correlation_name : symbol_table_alias_name
|
{ $$ = NULL; }
;
2003-08-15 02:02:18 +02:00
derived_column_list : '(' alias_list ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
alias_list : symbol_item_alias_name
| alias_list ',' symbol_item_alias_name
{ $$ = make_node (nod_list, 2, $1, $3); }
2001-05-23 15:26:42 +02:00
;
joined_table : table_reference join_type JOIN table_reference ON search_condition
{ $$ = make_node (nod_join, (int) e_join_count, $1, $2, $4, $6); }
| '(' joined_table ')'
{ $$ = $2; }
;
2003-08-15 02:02:18 +02:00
table_proc : symbol_procedure_name proc_table_inputs as_noise symbol_table_alias_name
{ $$ = make_node (nod_rel_proc_name,
(int) e_rpn_count, $1, $4, $2); }
2001-05-23 15:26:42 +02:00
| symbol_procedure_name proc_table_inputs
{ $$ = make_node (nod_rel_proc_name,
(int) e_rpn_count, $1, NULL, $2); }
;
proc_table_inputs : '(' null_or_value_list ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
null_or_value_list : null_or_value
| null_or_value_list ',' null_or_value
{ $$ = make_node (nod_list, 2, $1, $3); }
;
null_or_value : null_value
2003-09-04 23:26:15 +02:00
| value
;
2001-05-23 15:26:42 +02:00
table_name : simple_table_name
| symbol_table_name symbol_table_alias_name
{ $$ = make_node (nod_relation_name,
(int) e_rln_count, $1, $2); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
simple_table_name: symbol_table_name
{ $$ = make_node (nod_relation_name,
(int) e_rln_count, $1, NULL); }
;
join_type : INNER
{ $$ = make_node (nod_join_inner, (int) 0, NULL); }
| LEFT
{ $$ = make_node (nod_join_left, (int) 0, NULL); }
| LEFT OUTER
{ $$ = make_node (nod_join_left, (int) 0, NULL); }
| RIGHT
{ $$ = make_node (nod_join_right, (int) 0, NULL); }
| RIGHT OUTER
{ $$ = make_node (nod_join_right, (int) 0, NULL); }
| FULL
{ $$ = make_node (nod_join_full, (int) 0, NULL); }
| FULL OUTER
{ $$ = make_node (nod_join_full, (int) 0, NULL); }
|
{ $$ = make_node (nod_join_inner, (int) 0, NULL); }
;
/* other clauses in the select expression */
group_clause : GROUP BY group_by_list
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($3); }
|
{ $$ = 0; }
;
group_by_list : group_by_item
| group_by_list ',' group_by_item
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_list, 2, $1, $3); }
;
2003-08-16 15:14:35 +02:00
/* Except aggregate-functions are all expressions supported in group_by_item,
they are caught inside pass1.cpp */
group_by_item : value
2001-05-23 15:26:42 +02:00
;
having_clause : HAVING search_condition
{ $$ = $2; }
|
{ $$ = 0; }
;
where_clause : WHERE search_condition
{ $$ = $2; }
|
{ $$ = 0; }
;
/* PLAN clause to specify an access plan for a query */
plan_clause : PLAN plan_expression
{ $$ = $2; }
|
{ $$ = 0; }
;
plan_expression : plan_type '(' plan_item_list ')'
{ $$ = make_node (nod_plan_expr, 2, $1, make_list ($3)); }
;
plan_type : JOIN
{ $$ = 0; }
| SORT MERGE
{ $$ = make_node (nod_merge, (int) 0, NULL); }
| MERGE
{ $$ = make_node (nod_merge, (int) 0, NULL); }
/* for now the SORT operator is a no-op; it does not
change the place where a sort happens, but is just intended
to read the output from a SET PLAN */
| SORT
{ $$ = 0; }
|
{ $$ = 0; }
;
plan_item_list : plan_item
| plan_item ',' plan_item_list
{ $$ = make_node (nod_list, 2, $1, $3); }
;
plan_item : table_or_alias_list access_type
{ $$ = make_node (nod_plan_item, 2, make_list ($1), $2); }
| plan_expression
;
table_or_alias_list : symbol_table_name
| symbol_table_name table_or_alias_list
{ $$ = make_node (nod_list, 2, $1, $2); }
;
access_type : NATURAL
{ $$ = make_node (nod_natural, (int) 0, NULL); }
| INDEX '(' index_list ')'
{ $$ = make_node (nod_index, 1, make_list ($3)); }
| ORDER symbol_index_name extra_indices_opt
{ $$ = make_node (nod_index_order, 2, $2, $3); }
2001-05-23 15:26:42 +02:00
;
index_list : symbol_index_name
| symbol_index_name ',' index_list
{ $$ = make_node (nod_list, 2, $1, $3); }
;
extra_indices_opt : INDEX '(' index_list ')'
{ $$ = make_list ($3); }
|
{ $$ = 0; }
;
2001-05-23 15:26:42 +02:00
/* INSERT statement */
/* IBO hack: replace column_parens_opt by ins_column_parens_opt. */
insert : INSERT INTO simple_table_name ins_column_parens_opt VALUES '(' insert_value_list ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_insert, e_ins_count,
$3, make_list ($4), make_list ($7), NULL); }
| INSERT INTO simple_table_name ins_column_parens_opt ordered_select_expr
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_insert, e_ins_count, $3, $4, NULL, $5); }
;
insert_value_list : rhs
| insert_value_list ',' rhs
{ $$ = make_node (nod_list, 2, $1, $3); }
;
/* DELETE statement */
delete : delete_searched
| delete_positioned
;
delete_searched : KW_DELETE FROM table_name where_clause
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_delete, e_del_count, $3, $4, NULL); }
;
delete_positioned : KW_DELETE FROM table_name cursor_clause
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_delete, e_del_count, $3, NULL, $4); }
;
cursor_clause : WHERE CURRENT OF symbol_cursor_name
{ $$ = make_node (nod_cursor, e_cur_count, $4, NULL, NULL); }
;
/* UPDATE statement */
update : update_searched
| update_positioned
;
update_searched : UPDATE table_name SET assignments where_clause
{ $$ = make_node (nod_update, e_upd_count,
$2, make_list ($4), $5, NULL); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
update_positioned : UPDATE table_name SET assignments cursor_clause
{ $$ = make_node (nod_update, e_upd_count,
2003-09-04 23:26:15 +02:00
$2, make_list ($4), NULL, $5); }
2001-05-23 15:26:42 +02:00
;
assignments : assignment
| assignments ',' assignment
{ $$ = make_node (nod_list, 2, $1, $3); }
;
2002-06-29 08:56:51 +02:00
assignment : update_column_name '=' rhs
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_assign, 2, $3, $1); }
;
rhs : value
| null_value
;
/* BLOB get and put */
2003-09-04 23:26:15 +02:00
blob : READ BLOB simple_column_name FROM simple_table_name filter_clause segment_clause
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_get_segment, e_blb_count, $3, $5, $6, $7); }
2003-09-04 23:26:15 +02:00
| INSERT BLOB simple_column_name INTO simple_table_name filter_clause segment_clause
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_put_segment, e_blb_count, $3, $5, $6, $7); }
;
filter_clause : FILTER FROM blob_subtype_value TO blob_subtype_value
{ $$ = make_node (nod_list, 2, $3, $5); }
| FILTER TO blob_subtype_value
{ $$ = make_node (nod_list, 2, NULL, $3); }
|
;
blob_subtype_value : blob_subtype
| parameter
;
blob_subtype : signed_short_integer
{ $$ = MAKE_constant ((STR) $1, CONSTANT_SLONG); }
;
segment_clause : MAX_SEGMENT segment_length
{ $$ = $2; }
|
;
segment_length : unsigned_short_integer
{ $$ = MAKE_constant ((STR) $1, CONSTANT_SLONG); }
| parameter
;
/* column specifications */
column_parens_opt : column_parens
|
{ $$ = NULL; }
;
column_parens : '(' column_list ')'
{ $$ = make_list ($2); }
;
2002-06-29 08:56:51 +02:00
column_list : simple_column_name
| column_list ',' simple_column_name
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_list, 2, $1, $3); }
;
/* begin IBO hack */
ins_column_parens_opt : ins_column_parens
|
{ $$ = NULL; }
;
ins_column_parens : '(' ins_column_list ')'
{ $$ = make_list ($2); }
;
ins_column_list : update_column_name
| ins_column_list ',' update_column_name
{ $$ = make_node (nod_list, 2, $1, $3); }
;
/* end IBO hack */
2003-09-04 23:26:15 +02:00
column_name : simple_column_name
2001-05-23 15:26:42 +02:00
| symbol_table_alias_name '.' symbol_column_name
{ $$ = make_node (nod_field_name, (int) e_fln_count,
$1, $3); }
| symbol_table_alias_name '.' '*'
{ $$ = make_node (nod_field_name, (int) e_fln_count,
$1, NULL); }
;
simple_column_name : symbol_column_name
{ $$ = make_node (nod_field_name, (int) e_fln_count,
NULL, $1); }
;
2002-06-29 08:56:51 +02:00
update_column_name : simple_column_name
/* CVC: This option should be deprecated! The only allowed syntax should be
Update...set column = expr, without qualifier for the column. */
| symbol_table_alias_name '.' symbol_column_name
{ $$ = make_node (nod_field_name, (int) e_fln_count,
$1, $3); }
;
2001-05-23 15:26:42 +02:00
/* boolean expressions */
search_condition : trigger_action_predicate
| NOT trigger_action_predicate
{ $$ = make_node (nod_not, 1, $2); }
| simple_search_condition
2001-05-23 15:26:42 +02:00
| search_condition OR search_condition
{ $$ = make_node (nod_or, 2, $1, $3); }
| search_condition AND search_condition
{ $$ = make_node (nod_and, 2, $1, $3); }
;
bracable_search_condition : simple_search_condition
| NOT trigger_action_predicate
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_not, 1, $2); }
| bracable_search_condition OR search_condition
{ $$ = make_node (nod_or, 2, $1, $3); }
| bracable_search_condition AND search_condition
{ $$ = make_node (nod_and, 2, $1, $3); }
/* Special cases. Need help from lexer to parse the grammar */
/*| special_trigger_action_predicate -- handled by lexer */
| special_trigger_action_predicate OR search_condition
{ $$ = make_node (nod_or, 2, $1, $3); }
| special_trigger_action_predicate AND search_condition
{ $$ = make_node (nod_and, 2, $1, $3); }
2001-05-23 15:26:42 +02:00
;
simple_search_condition : predicate
| '(' bracable_search_condition ')'
{ $$ = $2; }
| NOT simple_search_condition
{ $$ = make_node (nod_not, 1, $2); }
;
predicate : comparison_predicate
2001-05-23 15:26:42 +02:00
| between_predicate
| like_predicate
| in_predicate
| null_predicate
| quantified_predicate
| exists_predicate
| containing_predicate
| starting_predicate
| unique_predicate;
2001-05-23 15:26:42 +02:00
/* comparisons */
comparison_predicate : value '=' value
{ $$ = make_node (nod_eql, 2, $1, $3); }
| value '<' value
{ $$ = make_node (nod_lss, 2, $1, $3); }
| value '>' value
{ $$ = make_node (nod_gtr, 2, $1, $3); }
| value GEQ value
{ $$ = make_node (nod_geq, 2, $1, $3); }
| value LEQ value
{ $$ = make_node (nod_leq, 2, $1, $3); }
| value NOT_GTR value
{ $$ = make_node (nod_leq, 2, $1, $3); }
| value NOT_LSS value
{ $$ = make_node (nod_geq, 2, $1, $3); }
| value NEQ value
{ $$ = make_node (nod_neq, 2, $1, $3); }
;
/* quantified comparisons */
quantified_predicate : value '=' ALL '(' column_select ')'
{ $$ = make_node (nod_eql_all, 2, $1, $5); }
| value '<' ALL '(' column_select ')'
{ $$ = make_node (nod_lss_all, 2, $1, $5); }
| value '>' ALL '(' column_select ')'
{ $$ = make_node (nod_gtr_all, 2, $1, $5); }
| value GEQ ALL '(' column_select ')'
{ $$ = make_node (nod_geq_all, 2, $1, $5); }
| value LEQ ALL '(' column_select ')'
{ $$ = make_node (nod_leq_all, 2, $1, $5); }
| value NOT_GTR ALL '(' column_select ')'
{ $$ = make_node (nod_leq_all, 2, $1, $5); }
| value NOT_LSS ALL '(' column_select ')'
{ $$ = make_node (nod_geq_all, 2, $1, $5); }
| value NEQ ALL '(' column_select ')'
{ $$ = make_node (nod_neq_all, 2, $1, $5); }
| value '=' some '(' column_select ')'
{ $$ = make_node (nod_eql_any, 2, $1, $5); }
| value '<' some '(' column_select ')'
{ $$ = make_node (nod_lss_any, 2, $1, $5); }
| value '>' some '(' column_select ')'
{ $$ = make_node (nod_gtr_any, 2, $1, $5); }
| value GEQ some '(' column_select ')'
{ $$ = make_node (nod_geq_any, 2, $1, $5); }
| value LEQ some '(' column_select ')'
{ $$ = make_node (nod_leq_any, 2, $1, $5); }
| value NOT_GTR some '(' column_select ')'
{ $$ = make_node (nod_leq_any, 2, $1, $5); }
| value NOT_LSS some '(' column_select ')'
{ $$ = make_node (nod_geq_any, 2, $1, $5); }
| value NEQ some '(' column_select ')'
{ $$ = make_node (nod_neq_any, 2, $1, $5); }
;
some : SOME
| ANY
;
/* other predicates */
between_predicate : value BETWEEN value AND value
{ $$ = make_node (nod_between, 3, $1, $3, $5); }
| value NOT BETWEEN value AND value
{ $$ = make_node (nod_not, 1, make_node (nod_between,
3, $1, $4, $6)); }
;
like_predicate : value LIKE value
{ $$ = make_node (nod_like, 2, $1, $3); }
| value NOT LIKE value
{ $$ = make_node (nod_not, 1, make_node (nod_like, 2, $1, $4)); }
| value LIKE value ESCAPE value
{ $$ = make_node (nod_like, 3, $1, $3, $5); }
| value NOT LIKE value ESCAPE value
{ $$ = make_node (nod_not, 1, make_node (nod_like,
3, $1, $4, $6)); }
;
in_predicate : value KW_IN in_predicate_value
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_eql_any, 2, $1, $3); }
| value NOT KW_IN in_predicate_value
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_not, 1, make_node (nod_eql_any, 2, $1, $4)); }
;
containing_predicate : value CONTAINING value
{ $$ = make_node (nod_containing, 2, $1, $3); }
| value NOT CONTAINING value
{ $$ = make_node (nod_not, 1, make_node (nod_containing, 2, $1, $4)); }
;
starting_predicate : value STARTING value
{ $$ = make_node (nod_starting, 2, $1, $3); }
| value NOT STARTING value
{ $$ = make_node (nod_not, 1, make_node (nod_starting, 2, $1, $4)); }
| value STARTING WITH value
{ $$ = make_node (nod_starting, 2, $1, $4); }
| value NOT STARTING WITH value
{ $$ = make_node (nod_not, 1, make_node (nod_starting, 2, $1, $5)); }
;
exists_predicate : EXISTS '(' ordered_select_expr ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_exists, 1, $3); }
;
unique_predicate : SINGULAR '(' ordered_select_expr ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_singular, 1, $3); }
;
null_predicate : value IS KW_NULL
{ $$ = make_node (nod_missing, 1, $1); }
| value IS NOT KW_NULL
{ $$ = make_node (nod_not, 1, make_node (nod_missing, 1, $1)); }
;
trigger_action_predicate : INSERTING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 1, CONSTANT_SLONG)); }
| UPDATING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 2, CONSTANT_SLONG)); }
| DELETING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 3, CONSTANT_SLONG)); }
;
special_trigger_action_predicate : KW_INSERTING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 1, CONSTANT_SLONG)); }
| KW_UPDATING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 2, CONSTANT_SLONG)); }
| KW_DELETING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((STR) 3, CONSTANT_SLONG)); }
;
2001-05-23 15:26:42 +02:00
/* set values */
in_predicate_value : table_subquery
| '(' value_list ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($2); }
;
table_subquery : '(' column_select ')'
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
;
2002-06-29 08:56:51 +02:00
column_select : SELECT limit_clause
2003-09-04 23:26:15 +02:00
distinct_clause
2001-05-23 15:26:42 +02:00
value
from_clause
where_clause
group_clause
having_clause
plan_clause
order_clause
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_select_expr, e_sel_count,
$2, $3, make_list ($4), $5, $6, $7, $8, $9, $10, NULL); }
2001-05-23 15:26:42 +02:00
;
2002-06-29 08:56:51 +02:00
column_singleton : SELECT limit_clause
2003-09-04 23:26:15 +02:00
distinct_clause
2001-05-23 15:26:42 +02:00
value
from_clause
where_clause
group_clause
having_clause
plan_clause
order_clause
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_select_expr, e_sel_count,
$2, $3, make_list ($4), $5, $6, $7, $8, $9, $10,
2001-05-23 15:26:42 +02:00
MAKE_constant ((STR) 1, CONSTANT_SLONG)); }
;
/* value types */
value : column_name
2001-05-23 15:26:42 +02:00
| array_element
| function
| u_constant
| parameter
| variable
| cast_specification
| case_expression
2001-05-23 15:26:42 +02:00
| udf
| '-' value
{ $$ = make_node (nod_negate, 1, $2); }
2003-09-04 23:26:15 +02:00
| '+' value
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| value '+' value
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_add2, 2, $1, $3);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_add, 2, $1, $3);
2001-05-23 15:26:42 +02:00
}
| value CONCATENATE value
{ $$ = make_node (nod_concatenate, 2, $1, $3); }
| value COLLATE symbol_collation_name
2002-11-11 20:08:37 +01:00
{ $$ = make_node (nod_collate, e_coll_count, (DSQL_NOD) $3, $1); }
2001-05-23 15:26:42 +02:00
| value '-' value
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_subtract2, 2, $1, $3);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_subtract, 2, $1, $3);
2001-05-23 15:26:42 +02:00
}
| value '*' value
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_multiply2, 2, $1, $3);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_multiply, 2, $1, $3);
2001-05-23 15:26:42 +02:00
}
| value '/' value
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_divide2, 2, $1, $3);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_divide, 2, $1, $3);
2001-05-23 15:26:42 +02:00
}
| '(' value ')'
{ $$ = $2; }
| '(' column_singleton ')'
{ $$ = $2; }
2002-06-29 08:56:51 +02:00
| current_user
| current_role
| internal_info
2001-05-23 15:26:42 +02:00
| DB_KEY
{ $$ = make_node (nod_dbkey, 1, NULL); }
| symbol_table_alias_name '.' DB_KEY
{ $$ = make_node (nod_dbkey, 1, $1); }
2003-09-04 23:26:15 +02:00
| KW_VALUE
{
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_dom_value, 0, NULL);
2003-09-04 23:26:15 +02:00
}
2001-05-23 15:26:42 +02:00
| datetime_value_expression
{ $$ = $1; }
;
datetime_value_expression : CURRENT_DATE
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "DATE",
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "DATE",
0);
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_current_date, 0, NULL);
}
| CURRENT_TIME
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "TIME",
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "TIME",
0);
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_current_time, 0, NULL);
}
| CURRENT_TIMESTAMP
{ $$ = make_node (nod_current_timestamp, 0, NULL); }
;
array_element : column_name '[' value_list ']'
{ $$ = make_node (nod_array, 2, $1, make_list ($3)); }
;
value_list : value
| value_list ',' value
{ $$ = make_node (nod_list, 2, $1, $3); }
;
constant : u_constant
| '-' u_numeric_constant
{ $$ = make_node (nod_negate, 1, $2); }
;
u_numeric_constant : NUMERIC
{ $$ = MAKE_constant ((STR) $1, CONSTANT_STRING); }
| NUMBER
{ $$ = MAKE_constant ((STR) $1, CONSTANT_SLONG); }
| FLOAT_NUMBER
2001-05-23 15:26:42 +02:00
{ $$ = MAKE_constant ((STR) $1, CONSTANT_DOUBLE); }
| NUMBER64BIT
{ $$ = MAKE_constant ((STR) $1, CONSTANT_SINT64); }
| SCALEDINT
{ $$ = MAKE_constant ((STR) $1, CONSTANT_SINT64); }
;
u_constant : u_numeric_constant
| sql_string
2003-05-23 18:55:40 +02:00
{ $$ = MAKE_str_constant ((STR) $1, lex.att_charset); }
2001-05-23 15:26:42 +02:00
| DATE STRING
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "DATE",
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "DATE",
0);
2001-05-23 15:26:42 +02:00
$$ = MAKE_constant ((STR) $2, CONSTANT_DATE);
}
| TIME STRING
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_dialect_datatype_unsupport,
gds_arg_number, client_dialect,
gds_arg_string, "TIME",
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, isc_sql_db_dialect_dtype_unsupport,
gds_arg_number, db_dialect,
gds_arg_string, "TIME",
0);
2001-05-23 15:26:42 +02:00
$$ = MAKE_constant ((STR) $2, CONSTANT_TIME);
}
| TIMESTAMP STRING
{ $$ = MAKE_constant ((STR) $2, CONSTANT_TIMESTAMP); }
;
parameter : '?'
{ $$ = make_parameter (); }
2001-05-23 15:26:42 +02:00
;
current_user : USER
{ $$ = make_node (nod_user_name, 0, NULL); }
2002-06-29 08:56:51 +02:00
| CURRENT_USER
{ $$ = make_node (nod_user_name, 0, NULL); }
;
current_role : CURRENT_ROLE
{ $$ = make_node (nod_current_role, 0, NULL); }
2001-05-23 15:26:42 +02:00
;
internal_info : CURRENT_CONNECTION
{ $$ = make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_connection_id, CONSTANT_SLONG)); }
| CURRENT_TRANSACTION
{ $$ = make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_transaction_id, CONSTANT_SLONG)); }
| GDSCODE
{ $$ = make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_gdscode, CONSTANT_SLONG)); }
| SQLCODE
{ $$ = make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_sqlcode, CONSTANT_SLONG)); }
| ROW_COUNT
{ $$ = make_node (nod_internal_info, e_internal_info_count,
MAKE_constant ((STR) internal_rows_affected, CONSTANT_SLONG)); }
;
2001-05-23 15:26:42 +02:00
sql_string : STRING /* string in current charset */
{ $$ = $1; }
| INTRODUCER STRING /* string in specific charset */
{ ((STR) $2)->str_charset = (TEXT *) $1;
$$ = $2; }
;
signed_short_integer : nonneg_short_integer
| '-' neg_short_integer
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) - (SLONG) $2; }
2001-05-23 15:26:42 +02:00
;
nonneg_short_integer : NUMBER
{ if ((SLONG) $1 > SHRT_POS_MAX)
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_expec_short);
2001-05-23 15:26:42 +02:00
/* Short integer expected */
$$ = $1;}
;
neg_short_integer : NUMBER
{ if ((SLONG) $1 > SHRT_NEG_MAX)
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_expec_short);
2001-05-23 15:26:42 +02:00
/* Short integer expected */
$$ = $1;}
;
pos_short_integer : nonneg_short_integer
{ if ((SLONG) $1 == 0)
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_expec_positive);
2001-05-23 15:26:42 +02:00
/* Positive number expected */
$$ = $1;}
;
unsigned_short_integer : NUMBER
{ if ((SLONG) $1 > SHRT_UNSIGNED_MAX)
2003-09-04 23:26:15 +02:00
yyabandon (-842, isc_expec_ushort);
2001-05-23 15:26:42 +02:00
/* Unsigned short integer expected */
$$ = $1;}
;
signed_long_integer : long_integer
| '-' long_integer
2002-11-11 20:08:37 +01:00
{ $$ = (DSQL_NOD) - (SLONG) $2; }
2001-05-23 15:26:42 +02:00
;
long_integer : NUMBER
{ $$ = $1;}
;
/* functions */
function : aggregate_function
| generate_value_function
| numeric_value_function
| string_value_function
;
aggregate_function : COUNT '(' '*' ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_agg_count, 0, NULL); }
| COUNT '(' all_noise value ')'
{ $$ = make_node (nod_agg_count, 1, $4); }
| COUNT '(' DISTINCT value ')'
{ $$ = make_flag_node (nod_agg_count,
2003-09-04 23:26:15 +02:00
NOD_AGG_DISTINCT, 1, $4); }
2001-05-23 15:26:42 +02:00
| SUM '(' all_noise value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_agg_total2, 1, $4);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_agg_total, 1, $4);
2001-05-23 15:26:42 +02:00
}
| SUM '(' DISTINCT value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_flag_node (nod_agg_total2,
2001-05-23 15:26:42 +02:00
NOD_AGG_DISTINCT, 1, $4);
else
2003-09-04 23:26:15 +02:00
$$ = make_flag_node (nod_agg_total,
2001-05-23 15:26:42 +02:00
NOD_AGG_DISTINCT, 1, $4);
}
| AVG '(' all_noise value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_agg_average2, 1, $4);
2001-05-23 15:26:42 +02:00
else
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_agg_average, 1, $4);
2001-05-23 15:26:42 +02:00
}
| AVG '(' DISTINCT value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_flag_node (nod_agg_average2,
2001-05-23 15:26:42 +02:00
NOD_AGG_DISTINCT, 1, $4);
else
2003-09-04 23:26:15 +02:00
$$ = make_flag_node (nod_agg_average,
2001-05-23 15:26:42 +02:00
NOD_AGG_DISTINCT, 1, $4);
}
| MINIMUM '(' all_noise value ')'
{ $$ = make_node (nod_agg_min, 1, $4); }
| MINIMUM '(' DISTINCT value ')'
{ $$ = make_node (nod_agg_min, 1, $4); }
| MAXIMUM '(' all_noise value ')'
{ $$ = make_node (nod_agg_max, 1, $4); }
| MAXIMUM '(' DISTINCT value ')'
{ $$ = make_node (nod_agg_max, 1, $4); }
;
/* Firebird specific functions into 'generate_value_function' */
generate_value_function : GEN_ID '(' symbol_generator_name ',' value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_gen_id2, 2, $3, $5);
else
$$ = make_node (nod_gen_id, 2, $3, $5);
}
;
numeric_value_function : extract_expression
;
extract_expression : EXTRACT '(' timestamp_part FROM value ')'
{ $$ = make_node (nod_extract, e_extract_count, $3, $5); }
;
string_value_function : substring_function
| KW_UPPER '(' value ')'
{ $$ = make_node (nod_upcase, 1, $3); }
;
substring_function : SUBSTRING '(' value FROM value string_length_opt ')'
/* SQL spec requires numbering to start with 1,
hence we decrement the first parameter to make it
compatible with the engine's implementation */
{ $$ = make_node (nod_substr, e_substr_count, $3,
make_node (nod_subtract, 2, $5,
MAKE_constant ((STR) 1, CONSTANT_SLONG)), $6); }
;
string_length_opt : FOR value
{ $$ = $2; }
|
{ $$ = MAKE_constant ((STR) SHRT_POS_MAX, CONSTANT_SLONG); }
;
2002-06-29 08:56:51 +02:00
2001-05-23 15:26:42 +02:00
udf : symbol_UDF_name '(' value_list ')'
{ $$ = make_node (nod_udf, 2, $1, $3); }
| symbol_UDF_name '(' ')'
{ $$ = make_node (nod_udf, 1, $1); }
;
cast_specification : CAST '(' rhs AS data_type_descriptor ')'
{ $$ = make_node (nod_cast, e_cast_count, $5, $3); }
;
/* case expressions */
case_expression : case_abbreviation
| case_specification
;
case_abbreviation : NULLIF '(' value ',' value ')'
{ $$ = make_node (nod_searched_case, 2,
make_node (nod_list, 2, make_node (nod_eql, 2, $3, $5),
make_node (nod_null, 0, NULL)), $3); }
| COALESCE '(' null_or_value ',' null_or_value_list ')'
{ $$ = make_node (nod_coalesce, 2, $3, $5); }
;
case_specification : simple_case
| searched_case
;
simple_case : CASE case_operand simple_when_clause END
{ $$ = make_node (nod_simple_case, 3, $2, make_list($3), make_node (nod_null, 0, NULL)); }
| CASE case_operand simple_when_clause ELSE case_result END
{ $$ = make_node (nod_simple_case, 3, $2, make_list($3), $5); }
;
simple_when_clause : WHEN when_operand THEN case_result
{ $$ = make_node (nod_list, 2, $2, $4); }
| simple_when_clause WHEN when_operand THEN case_result
{ $$ = make_node (nod_list, 2, $1, make_node (nod_list, 2, $3, $5)); }
;
searched_case : CASE searched_when_clause END
{ $$ = make_node (nod_searched_case, 2, make_list($2), make_node (nod_null, 0, NULL)); }
| CASE searched_when_clause ELSE case_result END
{ $$ = make_node (nod_searched_case, 2, make_list($2), $4); }
;
searched_when_clause : WHEN search_condition THEN case_result
{ $$ = make_node (nod_list, 2, $2, $4); }
| searched_when_clause WHEN search_condition THEN case_result
{ $$ = make_node (nod_list, 2, $1, make_node (nod_list, 2, $3, $5)); }
;
when_operand : value
;
case_operand : value
;
case_result : null_or_value
;
2001-05-23 15:26:42 +02:00
timestamp_part : YEAR
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_year, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| MONTH
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_month, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| DAY
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_day, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| HOUR
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_hour, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| MINUTE
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_minute, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| SECOND
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_second, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| WEEKDAY
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_weekday, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| YEARDAY
2002-06-29 08:56:51 +02:00
{ $$ = MAKE_constant ((STR)blr_extract_yearday, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
all_noise : ALL
|
;
null_value : KW_NULL
{ $$ = make_node (nod_null, 0, NULL); }
;
/* Performs special mapping of keywords into symbols */
symbol_UDF_name : SYMBOL
;
2001-05-23 15:26:42 +02:00
symbol_blob_subtype_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_character_set_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_collation_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_column_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_constraint_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_cursor_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_domain_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_exception_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_filter_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_gdscode_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_generator_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_index_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_item_alias_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_label_name : valid_symbol_name
;
symbol_procedure_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_role_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_table_alias_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_table_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_trigger_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_user_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_variable_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_view_name : valid_symbol_name
;
2001-05-23 15:26:42 +02:00
symbol_savepoint_name : valid_symbol_name
;
/* symbols */
valid_symbol_name : SYMBOL
| non_reserved_word
;
/* list of non-reserved words */
non_reserved_word :
KW_BREAK /* added in FB 1.0 */
| KW_DESCRIPTOR
| SUBSTRING
| COALESCE /* added in FB 1.5 */
| LAST
| LEAVE
| LOCK
| NULLIF
| NULLS
| STATEMENT
| USING
2003-09-04 23:26:15 +02:00
| INSERTING
| UPDATING
| DELETING
/* | FIRST | SKIP -- this is handled by the lexer. */
2003-09-04 23:26:15 +02:00
| BACKUP /* added in FB 2.0 */
| KW_DIFFERENCE
;
2001-05-23 15:26:42 +02:00
%%
/*
* PROGRAM: Dynamic SQL runtime support
* MODULE: lex.c
* DESCRIPTION: Lexical routine
*
*/
2001-05-23 15:26:42 +02:00
void LEX_dsql_init (void)
{
/**************************************
*
* L E X _ d s q l _ i n i t
*
**************************************
*
* Functional description
* Initialize LEX for processing. This is called only once
* per session.
*
**************************************/
for (const TOK *token = KEYWORD_getTokens(); token->tok_string; ++token)
{
DSQL_SYM symbol = FB_NEW_RPT(*DSQL_permanent_pool, 0) dsql_sym;
symbol->sym_string = (TEXT *) token->tok_string;
symbol->sym_length = strlen(token->tok_string);
symbol->sym_type = SYM_keyword;
symbol->sym_keyword = token->tok_ident;
symbol->sym_version = token->tok_version;
STR str_ = FB_NEW_RPT(*DSQL_permanent_pool, symbol->sym_length) str;
str_->str_length = symbol->sym_length;
strncpy((char*)str_->str_data, (char*)symbol->sym_string, symbol->sym_length);
symbol->sym_object = (void *) str_;
HSHD_insert(symbol);
}
2001-05-23 15:26:42 +02:00
}
void LEX_string (
2003-09-04 23:26:15 +02:00
TEXT *string,
USHORT length,
SSHORT character_set)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L E X _ s t r i n g
*
**************************************
*
* Functional description
* Initialize LEX to process a string.
*
**************************************/
2003-09-04 23:26:15 +02:00
lex.line_start = lex.ptr = string;
lex.end = string + length;
lex.lines = 1;
lex.att_charset = character_set;
lex.line_start_bk = lex.line_start;
lex.lines_bk = lex.lines;
2003-05-23 18:55:40 +02:00
lex.param_number = 1;
lex.prev_keyword = -1;
lex.prev_prev_keyword = -1;
2003-05-23 18:55:40 +02:00
lex.limit_clause = false;
lex.first_detection = false;
lex.brace_analysis = false;
#ifdef DSQL_DEBUG
2003-09-04 23:26:15 +02:00
if (DSQL_debug & 32)
dsql_trace("Source DSQL string:\n%.*s", (int)length, string);
#endif
2001-05-23 15:26:42 +02:00
}
static void check_log_file_attrs (void)
{
/**********************************************
*
* c h e c k _ l o g _ f i l e _ a t t r s
*
**********************************************
*
* Functional description
* Check if log file attributes are valid
*
*********************************************/
2003-09-04 23:26:15 +02:00
if (lex.g_file->fil_partitions) {
if (!lex.g_file->fil_length) {
yyabandon (-261, isc_log_length_spec);
/* Total length of a partitioned log must be specified */
}
if (PARTITION_SIZE (OneK * lex.g_file->fil_length, lex.g_file->fil_partitions) <
(OneK*MIN_LOG_LENGTH)) {
yyabandon (-239, isc_partition_too_small);
/* Log partition size too small */
}
}
else {
if ((lex.g_file->fil_length) && (lex.g_file->fil_length < MIN_LOG_LENGTH)) {
yyabandon (-239, isc_log_too_small); /* Log size too small */
}
}
2001-05-23 15:26:42 +02:00
}
static TEXT *lex_position (void)
{
/**************************************
*
* l e x _ p o s i t i o n
*
**************************************
*
* Functional description
* Return the current position of LEX
* in the input string.
*
**************************************/
2003-09-04 23:26:15 +02:00
return lex.ptr;
2001-05-23 15:26:42 +02:00
}
2003-04-03 03:36:52 +02:00
#ifdef NOT_USED_OR_REPLACED
2003-09-04 23:26:15 +02:00
static bool long_int(DSQL_NOD string,
SLONG *long_value)
2001-05-23 15:26:42 +02:00
{
/*************************************
*
* l o n g _ i n t
*
*************************************
*
* Functional description
* checks for all digits in the
* number and return an atol().
*
*************************************/
for (const char* p = ((STR) string)->str_data; classes [*p] & CHR_DIGIT; p++)
{
if (!(classes [*p] & CHR_DIGIT)) {
2003-09-04 23:26:15 +02:00
return false;
}
}
2001-05-23 15:26:42 +02:00
*long_value = atol ((char *)((STR) string)->str_data);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return true;
2001-05-23 15:26:42 +02:00
}
2003-04-03 03:36:52 +02:00
#endif
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
static DSQL_FLD make_field (DSQL_NOD field_name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a k e _ f i e l d
*
**************************************
*
* Functional description
* Make a field block of given name.
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_FLD field;
STR string;
TSQL tdsql;
tdsql = GET_THREAD_DATA;
if (field_name == NULL)
{
field = FB_NEW_RPT(*tdsql->tsql_default, sizeof (INTERNAL_FIELD_NAME)) dsql_fld;
strcpy (field->fld_name, (TEXT*) INTERNAL_FIELD_NAME);
return field;
}
string = (STR) field_name->nod_arg [1];
field = FB_NEW_RPT(*tdsql->tsql_default, strlen ((SCHAR*) string->str_data)) dsql_fld;
strcpy (field->fld_name, (TEXT*) string->str_data);
return field;
2001-05-23 15:26:42 +02:00
}
static FIL make_file (void)
{
/**************************************
*
* m a k e _ f i l e
*
**************************************
*
* Functional description
* Make a file block
*
**************************************/
2003-09-04 23:26:15 +02:00
FIL temp_file;
TSQL tdsql;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
tdsql = GET_THREAD_DATA;
temp_file = FB_NEW(*tdsql->tsql_default) fil;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return temp_file;
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
static DSQL_NOD make_list (DSQL_NOD node)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a k e _ l i s t
*
**************************************
*
* Functional description
* Collapse nested list nodes into single list.
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_NOD *ptr;
DLLS stack, temp;
USHORT l;
DSQL_NOD old;
TSQL tdsql;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
tdsql = GET_THREAD_DATA;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (!node)
return node;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
stack = 0;
stack_nodes (node, &stack);
for (l = 0, temp = stack; temp; temp = temp->lls_next)
l++;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
old = node;
node = FB_NEW_RPT(*tdsql->tsql_default, l) dsql_nod;
node->nod_count = l;
node->nod_type = nod_list;
node->nod_flags = old->nod_flags;
ptr = node->nod_arg + node->nod_count;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
while (stack)
*--ptr = (DSQL_NOD) LLS_POP (&stack);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return node;
2001-05-23 15:26:42 +02:00
}
static DSQL_NOD make_parameter (void)
{
/**************************************
*
* m a k e _ p a r a m e t e r
*
**************************************
*
* Functional description
* Make parameter node
* Any change should also be made to function below
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_NOD node;
TSQL tdsql;
2003-09-04 23:26:15 +02:00
tdsql = GET_THREAD_DATA;
2003-09-04 23:26:15 +02:00
node = FB_NEW_RPT(*tdsql->tsql_default, 1) dsql_nod;
node->nod_type = nod_parameter;
node->nod_line = (USHORT) lex.lines_bk;
node->nod_column = (USHORT) (lex.last_token_bk - lex.line_start_bk + 1);
node->nod_count = 1;
node->nod_arg[0] = (DSQL_NOD)(ULONG) lex.param_number++;
2003-09-04 23:26:15 +02:00
return node;
}
2003-09-04 23:26:15 +02:00
static DSQL_NOD make_node (NOD_TYPE type,
int count,
...)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a k e _ n o d e
*
**************************************
*
* Functional description
* Make a node of given type.
* Any change should also be made to function below
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_NOD node, *p;
va_list ptr;
TSQL tdsql;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
tdsql = GET_THREAD_DATA;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
node = FB_NEW_RPT(*tdsql->tsql_default, count) dsql_nod;
node->nod_type = type;
node->nod_line = (USHORT) lex.lines_bk;
node->nod_column = (USHORT) (lex.last_token_bk - lex.line_start_bk + 1);
node->nod_count = count;
p = node->nod_arg;
VA_START (ptr, count);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
while (--count >= 0)
*p++ = va_arg (ptr, DSQL_NOD);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return node;
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
static DSQL_NOD make_flag_node (NOD_TYPE type,
SSHORT flag,
int count,
...)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a k e _ f l a g _ n o d e
*
**************************************
*
* Functional description
* Make a node of given type. Set flag field
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_NOD node, *p;
va_list ptr;
TSQL tdsql;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
tdsql = GET_THREAD_DATA;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
node = FB_NEW_RPT(*tdsql->tsql_default, count) dsql_nod;
node->nod_type = type;
node->nod_flags = flag;
node->nod_line = (USHORT) lex.lines_bk;
node->nod_column = (USHORT) (lex.last_token_bk - lex.line_start_bk + 1);
node->nod_count = count;
p = node->nod_arg;
VA_START (ptr, count);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
while (--count >= 0)
*p++ = va_arg (ptr, DSQL_NOD);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return node;
2001-05-23 15:26:42 +02:00
}
2002-06-29 08:56:51 +02:00
static void prepare_console_debug (int level, int *yydeb)
{
/*************************************
*
* p r e p a r e _ c o n s o l e _ d e b u g
*
*************************************
*
* Functional description
* Activate debug info. In WinNT, redirect the standard
* output so one can see the generated information.
* Feel free to add your platform specific code.
*
*************************************/
#ifdef DSQL_DEBUG
2003-09-04 23:26:15 +02:00
DSQL_debug = level;
#endif
2003-09-04 23:26:15 +02:00
if (level >> 8)
*yydeb = level >> 8;
2002-06-29 08:56:51 +02:00
}
2003-04-03 03:36:52 +02:00
#ifdef NOT_USED_OR_REPLACED
2003-09-04 23:26:15 +02:00
static bool short_int(DSQL_NOD string,
SLONG *long_value,
SSHORT range)
2001-05-23 15:26:42 +02:00
{
/*************************************
*
* s h o r t _ i n t
*
*************************************
*
* Functional description
* is the string a valid representation
* of a positive short int?
*
*************************************/
if (((STR) string)->str_length > 5) {
2003-09-04 23:26:15 +02:00
return false;
}
2001-05-23 15:26:42 +02:00
for (char* p = ((STR) string)->str_data; classes [*p] & CHR_DIGIT; p++)
{
if (!(classes [*p] & CHR_DIGIT)) {
2003-09-04 23:26:15 +02:00
return false;
}
}
2001-05-23 15:26:42 +02:00
/* there are 5 or fewer digits, it's value may still be greater
* than 32767... */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
SCHAR buf[10];
buf [0] = ((STR) string)->str_data[0];
buf [1] = ((STR) string)->str_data[1];
buf [2] = ((STR) string)->str_data[2];
buf [3] = ((STR) string)->str_data[3];
buf [4] = ((STR) string)->str_data[4];
buf [5] = '\0';
2001-05-23 15:26:42 +02:00
*long_value = atoi (buf);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
bool return_value;
switch (range)
{
case POSITIVE:
2003-09-04 23:26:15 +02:00
return_value = *long_value > SHRT_POS_MAX;
break;
case NEGATIVE:
return_value = *long_value > SHRT_NEG_MAX;
break;
case UNSIGNED:
2003-09-04 23:26:15 +02:00
return_value = *long_value > SHRT_UNSIGNED_MAX;
break;
}
return !return_value;
2001-05-23 15:26:42 +02:00
}
2003-04-03 03:36:52 +02:00
#endif
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
static void stack_nodes (DSQL_NOD node,
DLLS *stack)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* s t a c k _ n o d e s
*
**************************************
*
* Functional description
* Assist in turning a tree of misc nodes into a clean list.
*
**************************************/
2003-09-04 23:26:15 +02:00
DSQL_NOD *ptr, *end;
DSQL_NOD curr_node, next_node, start_chain, end_chain, save_link;
if (node->nod_type != nod_list)
{
LLS_PUSH (node, stack);
return;
}
/* To take care of cases where long lists of nodes are in a chain
of list nodes with exactly one entry, this algorithm will look
for a pattern of repeated list nodes with two entries, the first
being a list node and the second being a non-list node. Such
a list will be reverse linked, and then re-reversed, stacking the
non-list nodes in the process. The purpose of this is to avoid
massive recursion of this function. */
start_chain = node;
end_chain = NULL;
curr_node = node;
next_node = node->nod_arg[0];
while ( curr_node->nod_count == 2 &&
curr_node->nod_arg[0]->nod_type == nod_list &&
curr_node->nod_arg[1]->nod_type != nod_list &&
next_node->nod_arg[0]->nod_type == nod_list &&
next_node->nod_arg[1]->nod_type != nod_list)
{
/* pattern was found so reverse the links and go to next node */
save_link = next_node->nod_arg[0];
next_node->nod_arg[0] = curr_node;
curr_node = next_node;
next_node = save_link;
end_chain = curr_node;
}
/* see if any chain was found */
if ( end_chain)
{
/* first, handle the rest of the nodes */
/* note that next_node still points to the first non-pattern node */
stack_nodes( next_node, stack);
/* stack the non-list nodes and reverse the chain on the way back */
curr_node = end_chain;
while (true)
{
LLS_PUSH( curr_node->nod_arg[1], stack);
if ( curr_node == start_chain)
break;
save_link = curr_node->nod_arg[0];
curr_node->nod_arg[0] = next_node;
next_node = curr_node;
curr_node = save_link;
}
return;
}
for (ptr = node->nod_arg, end = ptr + node->nod_count; ptr < end; ptr++)
stack_nodes (*ptr, stack);
2001-05-23 15:26:42 +02:00
}
2003-05-23 18:55:40 +02:00
inline static int yylex (
2003-09-04 23:26:15 +02:00
USHORT client_dialect,
USHORT db_dialect,
USHORT parser_version,
BOOLEAN *stmt_ambiguous)
2003-05-23 18:55:40 +02:00
{
int temp = lex.yylex(client_dialect, db_dialect, parser_version, stmt_ambiguous);
lex.prev_prev_keyword = lex.prev_keyword;
lex.prev_keyword = temp;
return temp;
2003-05-23 18:55:40 +02:00
}
2001-05-23 15:26:42 +02:00
2003-05-23 18:55:40 +02:00
int LexerState::yylex (
2003-09-04 23:26:15 +02:00
USHORT client_dialect,
USHORT db_dialect,
USHORT parser_version,
BOOLEAN *stmt_ambiguous)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* y y l e x
*
**************************************
*
2002-06-29 08:56:51 +02:00
* Functional description: lexer.
2001-05-23 15:26:42 +02:00
*
**************************************/
2003-09-04 23:26:15 +02:00
UCHAR tok_class;
char string[MAX_TOKEN_LEN];
char* p;
char* buffer;
char* buffer_end;
char* new_buffer;
DSQL_SYM sym;
2003-09-04 23:26:15 +02:00
SSHORT c;
USHORT buffer_len;
STR delimited_id_str;
/* Find end of white space and skip comments */
for (;;)
{
if (ptr >= end)
2002-06-29 08:56:51 +02:00
return -1;
2003-09-04 23:26:15 +02:00
c = *ptr++;
/* Process comments */
if (c == '\n') {
lines++;
line_start = ptr;
continue;
}
if ((c == '-') && (*ptr == '-')) {
/* single-line */
ptr++;
while (ptr < end) {
if ((c = *ptr++) == '\n') {
lines++;
line_start = ptr /* + 1*/; /* CVC: +1 left out. */
break;
2003-09-04 23:26:15 +02:00
}
}
2003-09-04 23:26:15 +02:00
if (ptr >= end)
return -1;
continue;
}
else if ((c == '/') && (*ptr == '*')) {
/* multi-line */
ptr++;
while (ptr < end) {
if ((c = *ptr++) == '*') {
if (*ptr == '/')
break;
}
if (c == '\n') {
lines++;
line_start = ptr /* + 1*/; /* CVC: +1 left out. */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
}
}
2003-09-04 23:26:15 +02:00
if (ptr >= end)
return -1;
ptr++;
continue;
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
tok_class = classes [c];
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (!(tok_class & CHR_WHITE))
break;
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* Depending on tok_class of token, parse token */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
last_token = ptr - 1;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (tok_class & CHR_INTRODUCER)
{
/* The Introducer (_) is skipped, all other idents are copied
* to become the name of the character set
*/
p = string;
for (; ptr < end && classes [*ptr] & CHR_IDENT; ptr++)
{
if (ptr >= end)
return -1;
2003-10-15 00:22:32 +02:00
check_copy_incr(p, UPPER7(*ptr), string);
2003-09-04 23:26:15 +02:00
}
2003-10-15 00:22:32 +02:00
check_bound(p, string);
2003-09-04 23:26:15 +02:00
*p = 0;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* make a string value to hold the name, the name
* is resolved in pass1_constant */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) (MAKE_string(string, p - string))->str_data;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
return INTRODUCER;
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* parse a quoted string, being sure to look for double quotes */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (tok_class & CHR_QUOTE)
2001-05-23 15:26:42 +02:00
{
2003-09-04 23:26:15 +02:00
buffer = string;
buffer_len = sizeof (string);
buffer_end = buffer + buffer_len - 1;
for (p = buffer; ; p++)
{
if (ptr >= end)
{
if (buffer != string)
gds__free (buffer);
return -1;
}
/* *ptr is quote - if next != quote we're at the end */
if ((*ptr == c) && ((++ptr == end) || (*ptr != c)))
break;
if (p > buffer_end)
{
new_buffer = (char*)gds__alloc (2 * buffer_len);
/* FREE: at outer block */
if (!new_buffer) /* NOMEM: */
{
if (buffer != string)
gds__free (buffer);
return -1;
}
memcpy (new_buffer, buffer, buffer_len);
if (buffer != string)
gds__free (buffer);
buffer = new_buffer;
p = buffer + buffer_len;
buffer_len = 2 * buffer_len;
buffer_end = buffer + buffer_len - 1;
}
*p = *ptr++;
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
if (c == '"')
2001-05-23 15:26:42 +02:00
{
2003-09-04 23:26:15 +02:00
*stmt_ambiguous = TRUE; /* string delimited by double quotes could be
** either a string constant or a SQL delimited
** identifier, therefore marks the SQL
** statement as ambiguous */
if (client_dialect == SQL_DIALECT_V6_TRANSITION)
{
if (buffer != string)
gds__free (buffer);
yyabandon (-104, isc_invalid_string_constant);
}
else if (client_dialect >= SQL_DIALECT_V6)
{
if ((p - buffer) >= MAX_TOKEN_LEN)
{
if (buffer != string)
gds__free (buffer);
yyabandon (-104, isc_token_too_long);
}
yylval = (DSQL_NOD) MAKE_string(buffer, p - buffer);
delimited_id_str = (STR) yylval;
delimited_id_str->str_flags |= STR_delimited_id;
if (buffer != string)
gds__free (buffer);
return SYMBOL;
}
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) MAKE_string(buffer, p - buffer);
if (buffer != string)
2001-05-23 15:26:42 +02:00
gds__free (buffer);
2003-09-04 23:26:15 +02:00
return STRING;
}
2001-05-23 15:26:42 +02:00
/*
* Check for a numeric constant, which starts either with a digit or with
* a decimal point followed by a digit.
*
* This code recognizes the following token types:
*
* NUMBER: string of digits which fits into a 32-bit integer
*
* NUMBER64BIT: string of digits whose value might fit into an SINT64,
* depending on whether or not there is a preceding '-', which is to
* say that "9223372036854775808" is accepted here.
*
* SCALEDINT: string of digits and a single '.', where the digits
* represent a value which might fit into an SINT64, depending on
* whether or not there is a preceding '-'.
*
* FLOAT: string of digits with an optional '.', and followed by an "e"
* or "E" and an optionally-signed exponent.
*
* NOTE: we swallow leading or trailing blanks, but we do NOT accept
* embedded blanks:
*
* Another note: c is the first character which need to be considered,
* ptr points to the next character.
*/
2003-09-04 23:26:15 +02:00
assert(ptr <= end);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if ((tok_class & CHR_DIGIT) ||
((c == '.') && (ptr < end) && (classes [*ptr] & CHR_DIGIT)))
2001-05-23 15:26:42 +02:00
{
2003-09-04 23:26:15 +02:00
/* The following variables are used to recognize kinds of numbers. */
bool have_error = false; /* syntax error or value too large */
bool have_digit = false; /* we've seen a digit */
bool have_decimal = false; /* we've seen a '.' */
bool have_exp = false; /* digit ... [eE] */
bool have_exp_sign = false; /* digit ... [eE] {+-] */
bool have_exp_digit = false; /* digit ... [eE] ... digit */
UINT64 number = 0;
UINT64 limit_by_10 = MAX_SINT64 / 10;
for (--ptr ; ptr < end ; ptr++)
{
c = *ptr;
if (have_exp_digit && (! (classes [c] & CHR_DIGIT)))
/* First non-digit after exponent and digit terminates
the token. */
break;
else if (have_exp_sign && (! (classes [c] & CHR_DIGIT)))
{
/* only digits can be accepted after "1E-" */
have_error = true;
break;
}
else if (have_exp)
{
/* We've seen e or E, but nothing beyond that. */
if ( ('-' == c) || ('+' == c) )
have_exp_sign = true;
else if ( classes [c] & CHR_DIGIT )
/* We have a digit: we haven't seen a sign yet,
but it's too late now. */
have_exp_digit = have_exp_sign = true;
else
{
/* end of the token */
have_error = true;
break;
}
}
else if ('.' == c)
{
if (!have_decimal)
have_decimal = true;
else
{
have_error = true;
break;
}
}
else if (classes [c] & CHR_DIGIT)
{
/* Before computing the next value, make sure there will be
no overflow. */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
have_digit = true;
2002-06-29 08:56:51 +02:00
2003-09-04 23:26:15 +02:00
if (number >= limit_by_10)
/* possibility of an overflow */
if ((number > limit_by_10) || (c > '8'))
{
have_error = true;
break;
}
number = number * 10 + (c - '0');
}
else if ( (('E' == c) || ('e' == c)) && have_digit )
have_exp = true;
else
/* Unexpected character: this is the end of the number. */
break;
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* We're done scanning the characters: now return the right kind
of number token, if any fits the bill. */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (!have_error)
2001-05-23 15:26:42 +02:00
{
2003-09-04 23:26:15 +02:00
assert(have_digit);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (have_exp_digit)
{
yylval = (DSQL_NOD) MAKE_string(last_token, ptr - last_token);
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
2002-06-29 08:56:51 +02:00
2003-09-04 23:26:15 +02:00
return FLOAT_NUMBER;
}
else if (!have_exp)
{
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* We should return some kind (scaled-) integer type
except perhaps in dialect 1. */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (!have_decimal && (number <= MAX_SLONG))
{
yylval = (DSQL_NOD) (ULONG) number;
return NUMBER;
}
else
{
/* We have either a decimal point with no exponent
or a string of digits whose value exceeds MAX_SLONG:
the returned type depends on the client dialect,
so warn of the difference if the client dialect is
SQL_DIALECT_V6_TRANSITION.
*/
if (SQL_DIALECT_V6_TRANSITION == client_dialect)
{
/* Issue a warning about the ambiguity of the numeric
* numeric literal. There are multiple calls because
* the message text exceeds the 119-character limit
* of our message database.
*/
ERRD_post_warning( isc_dsql_warning_number_ambiguous,
gds_arg_string,
ERR_string( last_token,
ptr - last_token ),
gds_arg_end );
ERRD_post_warning( isc_dsql_warning_number_ambiguous1,
gds_arg_end );
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) MAKE_string(last_token, ptr - last_token);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
return FLOAT_NUMBER;
else if (have_decimal)
return SCALEDINT;
else
return NUMBER64BIT;
}
2003-09-04 23:26:15 +02:00
} /* else if (!have_exp) */
} /* if (!have_error) */
/* we got some kind of error or overflow, so don't recognize this
* as a number: just pass it through to the next part of the lexer.
*/
2002-06-29 08:56:51 +02:00
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* Restore the status quo ante, before we started our unsuccessful
attempt to recognize a number. */
ptr = last_token;
c = *ptr++;
/* We never touched tok_class, so it doesn't need to be restored. */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* end of number-recognition code */
if (tok_class & CHR_LETTER)
{
2003-09-04 23:26:15 +02:00
p = string;
2003-10-15 00:22:32 +02:00
check_copy_incr(p, UPPER (c), string);
2003-09-04 23:26:15 +02:00
for (; ptr < end && classes [*ptr] & CHR_IDENT; ptr++)
{
2003-09-04 23:26:15 +02:00
if (ptr >= end)
return -1;
2003-10-15 00:22:32 +02:00
check_copy_incr(p, UPPER (*ptr), string);
}
2003-09-04 23:26:15 +02:00
2003-10-15 00:22:32 +02:00
check_bound(p, string);
2003-09-04 23:26:15 +02:00
*p = 0;
sym = HSHD_lookup (NULL, (TEXT *) string, (SSHORT)(p - string), SYM_keyword, parser_version);
if (sym)
{
/* 13 June 2003. Nickolay Samofatov
* Detect INSERTING/UPDATING/DELETING as non-reserved keywords.
* We need to help parser from lexer because our grammar is not LARL(1) in this case
*/
if (prev_keyword == '(' && !brace_analysis &&
(sym->sym_keyword == INSERTING ||
sym->sym_keyword == UPDATING ||
sym->sym_keyword == DELETING
) &&
/* Produce special_trigger_action_predicate only where we can handle it -
in search conditions */
(prev_prev_keyword=='(' || prev_prev_keyword==NOT || prev_prev_keyword==AND ||
prev_prev_keyword==OR || prev_prev_keyword==ON || prev_prev_keyword==HAVING ||
prev_prev_keyword==WHERE || prev_prev_keyword==WHEN) )
{
2003-09-04 23:26:15 +02:00
LexerState savedState = lex;
int nextToken = yylex(client_dialect,db_dialect,parser_version,stmt_ambiguous);
lex = savedState;
2003-09-04 23:26:15 +02:00
if (nextToken==OR || nextToken==AND) {
switch(sym->sym_keyword) {
case INSERTING:
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) sym->sym_object;
return KW_INSERTING;
case UPDATING:
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) sym->sym_object;
return KW_UPDATING;
case DELETING:
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) sym->sym_object;
return KW_DELETING;
}
}
}
2003-09-04 23:26:15 +02:00
/* 23 May 2003. Nickolay Samofatov
* Detect FIRST/SKIP as non-reserved keywords
* 1. We detect FIRST or SKIP as keywords if they appear just after SELECT and
* immediately before parameter mark ('?'), opening brace ('(') or number
* 2. We detect SKIP as a part of FIRST/SKIP clause the same way
* 3. We detect FIRST if we are explicitly asked for (such as in NULLS FIRST/LAST clause)
* 4. In all other cases we return them as SYMBOL
*/
if ((sym->sym_keyword == FIRST && !first_detection) || sym->sym_keyword == SKIP) {
if (prev_keyword == SELECT || limit_clause) {
LexerState savedState = lex;
int nextToken = yylex(client_dialect,db_dialect,parser_version,stmt_ambiguous);
lex = savedState;
if (nextToken != NUMBER && nextToken != '?' && nextToken != '(') {
yylval = (DSQL_NOD) MAKE_string(string, p - string);
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return SYMBOL;
} else {
yylval = (DSQL_NOD) sym->sym_object;
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return sym->sym_keyword;
}
} /* else fall down and return token as SYMBOL */
} else {
yylval = (DSQL_NOD) sym->sym_object;
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return sym->sym_keyword;
}
}
2003-09-04 23:26:15 +02:00
yylval = (DSQL_NOD) MAKE_string(string, p - string);
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return SYMBOL;
}
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
/* Must be punctuation -- test for double character punctuation */
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (last_token + 1 < end)
{
sym = HSHD_lookup (NULL, last_token, (SSHORT) 2, SYM_keyword, (USHORT) parser_version);
if (sym)
{
++ptr;
return sym->sym_keyword;
}
}
/* We need to swallow braces around INSERTING/UPDATING/DELETING keywords */
/* This algorithm is not perfect, but it is ok for now.
It should be dropped when BOOLEAN datatype is introduced in Firebird */
if ( c == '(' && !brace_analysis &&
/* 1) We need to swallow braces in all boolean expressions
2) We may swallow braces in ordinary expressions
3) We should not swallow braces after special tokens
like IF, FIRST, SKIP, VALUES and 30 more other
*/
(prev_keyword=='(' || prev_keyword==NOT || prev_keyword==AND || prev_keyword==OR ||
prev_keyword==ON || prev_keyword==HAVING || prev_keyword==WHERE || prev_keyword==WHEN) )
{
LexerState savedState = lex;
brace_analysis = true;
int openCount = 0;
int nextToken;
do {
openCount++;
nextToken = yylex(client_dialect,db_dialect,parser_version,stmt_ambiguous);
} while (nextToken == '(');
DSQL_NOD temp_val = yylval;
if (nextToken == INSERTING || nextToken == UPDATING || nextToken == DELETING)
{
/* Skip closing braces. */
while ( openCount &&
yylex(client_dialect,db_dialect,
parser_version,stmt_ambiguous) == ')')
{
openCount--;
}
if (openCount) {
/* Not enough closing braces. Restore status quo. */
lex = savedState;
}
else {
/* Cool! We successfully swallowed braces ! */
brace_analysis = false;
yylval = temp_val;
/* Check if we need to handle LR(2) grammar case */
if (prev_keyword == '(' &&
/* Produce special_trigger_action_predicate only where we can handle it -
in search conditions */
(prev_prev_keyword=='(' || prev_prev_keyword==NOT || prev_prev_keyword==AND ||
prev_prev_keyword==OR || prev_prev_keyword==ON || prev_prev_keyword==HAVING ||
prev_prev_keyword==WHERE || prev_prev_keyword==WHEN) )
{
savedState = lex;
int token = yylex(client_dialect,db_dialect,parser_version,stmt_ambiguous);
lex = savedState;
if (token==OR || token==AND) {
switch(nextToken) {
case INSERTING:
return KW_INSERTING;
case UPDATING:
return KW_UPDATING;
case DELETING:
return KW_DELETING;
}
}
}
return nextToken;
}
} else {
/* Restore status quo. */
lex = savedState;
}
}
/* Single character punctuation are simply passed on */
return c;
2001-05-23 15:26:42 +02:00
}
static void yyerror (
2003-09-04 23:26:15 +02:00
TEXT *error_string)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* y y e r r o r
*
**************************************
*
* Functional description
* Print a syntax error.
*
**************************************/
2003-09-04 23:26:15 +02:00
if (yychar < 1)
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, gds_command_end_err, /* Unexpected end of command */
0);
else
{
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) -104,
gds_arg_gds, gds_dsql_token_unk_err,
gds_arg_number, (SLONG) lex.lines,
gds_arg_number, (SLONG) (lex.last_token - lex.line_start + 1), /*CVC: +1*/
/* Token unknown - line %d, char %d */
gds_arg_gds, gds_random,
gds_arg_cstring, (int) (lex.ptr - lex.last_token), lex.last_token, 0);
}
2001-05-23 15:26:42 +02:00
}
2003-09-04 23:26:15 +02:00
static void yyabandon (SSHORT sql_code,
ISC_STATUS error_symbol)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* y y a b a n d o n
*
**************************************
*
* Functional description
* Abandon the parsing outputting the supplied string
*
**************************************/
2003-09-04 23:26:15 +02:00
ERRD_post (gds_sqlerr, gds_arg_number, (SLONG) sql_code,
gds_arg_gds, error_symbol, 0);
2001-05-23 15:26:42 +02:00
}