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

5294 lines
134 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.
* 2003.10.05 Dmitry Yemanov: Added support for explicit cursors in PSQL.
* 2004.01.16 Vlad Horsun: added support for default parameters and
* EXECUTE BLOCK statement
2001-05-23 15:26:42 +02:00
*/
#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"
2003-11-08 00:27:24 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/flags.h"
#include "../dsql/errd_proto.h"
#include "../dsql/hsh_proto.h"
#include "../dsql/make_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.h"
2003-09-12 03:41:03 +02:00
#include "../jrd/err_proto.h"
2001-07-12 07:46:06 +02:00
2003-11-10 10:16:38 +01:00
static void yyerror(const TEXT*);
2001-05-23 15:26:42 +02:00
/* 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
//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
/* Make bison allocate static stack */
#define YYINITDEPTH 2048
2004-05-03 03:53:24 +02:00
// Using this option causes build problems on Win32 with bison 1.28
//#define YYSTACK_USE_ALLOCA 1
typedef dsql_nod* YYSTYPE;
#define YYSTYPE YYSTYPE
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
2003-11-10 10:16:38 +01:00
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-16 10:51:06 +02:00
inline SLONG trigger_type_suffix(const int slot1, const int slot2, const int slot3)
{
2003-10-15 00:22:32 +02:00
return ((slot1 << 1) | (slot2 << 3) | (slot3 << 5));
}
2003-11-10 10:16:38 +01:00
dsql_nod* DSQL_parse;
2001-05-23 15:26:42 +02:00
2003-11-10 10:16:38 +01:00
//static void yyerror(const TEXT*); redeclaration.
2001-05-23 15:26:42 +02:00
#define YYPARSE_PARAM_TYPE
#define YYPARSE_PARAM USHORT client_dialect, USHORT db_dialect, USHORT parser_version, bool* stmt_ambiguous
#include "../dsql/chars.h"
2003-10-15 00:22:32 +02:00
const int MAX_TOKEN_LEN = 256;
static const TEXT* lex_position();
2003-04-03 03:36:52 +02:00
#ifdef NOT_USED_OR_REPLACED
static bool long_int(dsql_nod*, SLONG*);
2003-04-03 03:36:52 +02:00
#endif
static dsql_fld* make_field (dsql_nod*);
static dsql_fil* make_file();
2004-04-10 02:25:22 +02:00
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
static bool short_int(dsql_nod*, SLONG*, SSHORT);
2003-04-03 03:36:52 +02:00
#endif
static void stack_nodes (dsql_nod*, DsqlNodStack&);
inline static int yylex (USHORT, USHORT, USHORT, bool*);
2003-04-10 08:32:58 +02:00
static void yyabandon (SSHORT, ISC_STATUS);
2003-10-16 10:51:06 +02:00
inline void check_bound(const char* const to, const char* const string)
2003-10-15 00:22:32 +02:00
{
if ((to - string) >= MAX_TOKEN_LEN)
yyabandon (-104, isc_token_too_long);
}
2003-10-16 10:51:06 +02:00
inline void check_copy_incr(char*& to, const char ch, const char* const string)
2003-10-15 00:22:32 +02:00
{
check_bound(to, string);
2003-10-16 10:51:06 +02:00
*to++ = ch;
2003-10-15 00:22:32 +02:00
}
2003-05-23 18:55:40 +02:00
struct LexerState {
/* This is, in fact, parser state. Not used in lexer itself */
2003-11-10 10:16:38 +01:00
dsql_fld* g_field;
dsql_fil* g_file;
dsql_nod* g_field_name;
bool log_defined, cache_defined;
2003-05-23 18:55:40 +02:00
int dsql_debug;
/* Actual lexer state begins from here */
const TEXT* beginning;
const TEXT* ptr;
const TEXT* end;
const TEXT* last_token;
const TEXT* line_start;
const TEXT* last_token_bk;
const TEXT* line_start_bk;
2003-05-23 18:55:40 +02:00
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,
bool* stmt_ambiguous);
2003-05-23 18:55:40 +02:00
};
/* 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
2001-05-23 15:26:42 +02:00
%token BEFORE
%token BEGIN
%token BETWEEN
%token BLOB
%token BY
//%token CACHE
2001-05-23 15:26:42 +02:00
%token CAST
%token CHARACTER
%token CHECK
//%token CHECK_POINT_LEN
2001-05-23 15:26:42 +02:00
%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
2001-05-23 15:26:42 +02:00
%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
2001-05-23 15:26:42 +02:00
%token LPAREN
%token LEFT
%token LEQ
%token LEVEL
%token LIKE
//%token LOG_BUF_SIZE
2001-05-23 15:26:42 +02:00
%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
2001-05-23 15:26:42 +02:00
%token OF
%token ON
%token ONLY
%token OPTION
%token OR
%token ORDER
%token OUTER
%token OUTPUT_TYPE
%token OVERFLOW
%token PAGE
%token PAGES
%token KW_PAGE_SIZE
2001-05-23 15:26:42 +02:00
%token PARAMETER
%token PASSWORD
%token PLAN
%token POSITION
%token POST_EVENT
%token PRECISION
%token PRIMARY
%token PRIVILEGES
%token PROCEDURE
%token PROTECTED
//%token RAW_PARTITIONS
2001-05-23 15:26:42 +02:00
%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
%token OPEN
%token CLOSE
%token FETCH
%token ROWS
%token BLOCK
%token IIF
%token SCALAR_ARRAY
%token CROSS
%token NEXT
%token SEQUENCE
%token RESTART
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
2001-05-23 15:26:42 +02:00
%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
| exec_procedure
| exec_block
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
{ prepare_console_debug ((IPTR) $2, &yydebug);
2001-05-23 15:26:42 +02:00
$$ = make_node (nod_null, (int) 0, NULL); }
;
/* GRANT statement */
grant : GRANT privileges ON table_noise simple_table_name
TO non_role_grantee_list grant_option
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $5, make_list($7), $8); }
| GRANT proc_privileges ON PROCEDURE simple_proc_name
TO non_role_grantee_list grant_option
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_grant, (int) e_grant_count,
$2, $5, make_list($7), $8); }
2001-05-23 15:26:42 +02:00
| 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); }
;
2001-05-23 15:26:42 +02:00
table_noise : TABLE
|
2001-05-23 15:26:42 +02:00
;
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); }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
role_admin_option : WITH ADMIN OPTION
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_grant_admin, (int) 0, NULL); }
|
{ $$ = NULL; }
2003-09-04 23:26:15 +02:00
;
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 table_noise simple_table_name
FROM non_role_grantee_list
{ $$ = make_node (nod_revoke, (int) e_grant_count,
$3, $6, make_list($8), $2); }
| REVOKE rev_grant_option proc_privileges ON PROCEDURE simple_proc_name
FROM non_role_grantee_list
{ $$ = make_node (nod_revoke, (int) e_grant_count,
$3, $6, make_list($8), $2); }
| REVOKE rev_admin_option role_name_list FROM role_grantee_list
{ $$ = make_node (nod_revoke, (int) e_grant_count,
make_list($3), make_list($5), NULL, $2); }
;
2001-05-23 15:26:42 +02:00
rev_grant_option : GRANT OPTION FOR
{ $$ = make_node (nod_grant, (int) 0, NULL); }
|
{ $$ = NULL; }
;
rev_admin_option : ADMIN OPTION FOR
{ $$ = make_node (nod_grant_admin, (int) 0, NULL); }
|
{ $$ = NULL; }
;
non_role_grantee_list : grantee_list
| user_grantee_list
2001-05-23 15:26:42 +02:00
;
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)(IPTR) $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; } */
arg_desc : init_data_type udf_data_type param_mechanism
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_udf_param, (int) e_udf_param_count,
$1, $3); }
2001-05-23 15:26:42 +02:00
;
param_mechanism :
{ $$ = NULL; } /* Beware: ddl.cpp converts this to mean FUN_reference. */
| BY KW_DESCRIPTOR
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_descriptor, CONSTANT_SLONG); }
| BY SCALAR_ARRAY
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_scalar_array, CONSTANT_SLONG); }
| KW_NULL
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_ref_with_null, CONSTANT_SLONG); }
;
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 return_mechanism
{ $$ = make_node (nod_udf_return_value, (int) e_udf_param_count,
$1, $3); }
2001-05-23 15:26:42 +02:00
| PARAMETER pos_short_integer
{ $$ = make_node (nod_udf_return_value, (int) e_udf_param_count,
2003-11-10 10:16:38 +01:00
NULL, MAKE_constant ((dsql_str*) $2, CONSTANT_SLONG));}
2001-05-23 15:26:42 +02:00
;
return_mechanism :
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_reference, CONSTANT_SLONG); }
| BY KW_VALUE
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_value, CONSTANT_SLONG); }
| BY KW_DESCRIPTOR
{ $$ = MAKE_constant ((dsql_str*) Jrd::FUN_descriptor, CONSTANT_SLONG); }
| FREE_IT
{ $$ = MAKE_constant ((dsql_str*) (-1 * Jrd::FUN_reference), CONSTANT_SLONG); }
/* FUN_refrence with FREE_IT is -ve */
| BY KW_DESCRIPTOR FREE_IT
{ $$ = MAKE_constant ((dsql_str*) (-1 * Jrd::FUN_descriptor), CONSTANT_SLONG); }
;
2004-09-08 14:01:30 +02:00
filter_decl_clause : symbol_filter_name INPUT_TYPE blob_filter_subtype OUTPUT_TYPE blob_filter_subtype
2001-05-23 15:26:42 +02:00
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); }
;
2004-09-08 14:01:30 +02:00
blob_filter_subtype : symbol_blob_subtype_name
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_STRING); }
|
signed_short_integer
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG); }
;
2001-05-23 15:26:42 +02:00
/* 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 exception_clause
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| 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; }
| SEQUENCE generator_clause
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| 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
*/
| EXCEPTION rexception_clause
{ $$ = $2; }
;
replace : CREATE OR ALTER replace_clause
{ $$ = $4; }
;
replace_clause : PROCEDURE replace_procedure_clause
{ $$ = $2; }
| TRIGGER replace_trigger_clause
{ $$ = $2; }
/*
| VIEW replace_view_clause
{ $$ = $2; }
*/
| EXCEPTION replace_exception_clause
{ $$ = $2; }
;
/* CREATE EXCEPTION */
exception_clause : symbol_exception_name sql_string
{ $$ = make_node (nod_def_exception, (int) e_xcp_count,
$1, $2); }
;
rexception_clause : symbol_exception_name sql_string
{ $$ = make_node (nod_redef_exception, (int) e_xcp_count,
$1, $2); }
;
replace_exception_clause : symbol_exception_name sql_string
{ $$ = make_node (nod_replace_exception, (int) e_xcp_count,
$1, $2); }
;
alter_exception_clause : symbol_exception_name sql_string
{ $$ = make_node (nod_mod_exception, (int) e_xcp_count,
$1, $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
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| AUTO
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
|
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
conditional :
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| CONDITIONAL
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
first_file_length :
{ $$ = (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 SEQUENCE/GENERATOR */
2001-05-23 15:26:42 +02:00
generator_clause : symbol_generator_name
{ $$ = make_node (nod_def_generator, (int) e_gen_count, $1); }
2001-05-23 15:26:42 +02:00
;
/* 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
{ lex.log_defined = false;
lex.cache_defined = false;
$$ = (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: KW_PAGE_SIZE equals pos_short_integer
2001-05-23 15:26:42 +02:00
{ $$ = 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
2001-05-23 15:26:42 +02:00
| db_log
| db_log_option
*/
2001-05-23 15:26:42 +02:00
| 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
;
/*
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)
yyabandon (-260, isc_log_redef);
*/ /* Log redefined */ /*
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);
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);
*/ /* Partitions not supported in series of log file specification */ /*
2001-05-23 15:26:42 +02:00
$$ = 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,
(dsql_nod*) lex.g_file);}
2001-05-23 15:26:42 +02:00
;
*/
2001-05-23 15:26:42 +02:00
db_file : file1 sql_string file_desc1
2003-11-10 10:16:38 +01:00
{ lex.g_file->fil_name = (dsql_str*) $2;
$$ = (dsql_nod*) make_node (nod_file_desc, (int) 1,
(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 = (IPTR) $3;
2003-11-10 10:16:38 +01:00
lex.g_file->fil_name = (dsql_str*) $2;
lex.cache_defined = true;
$$ = (dsql_nod*) make_node (nod_cache_file_desc, (int) 1,
(dsql_nod*) lex.g_file); }
2001-05-23 15:26:42 +02:00
;
*/
/*
cache_length :
{ $$ = (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
$$ = (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
{
$$ = (dsql_nod*) make_node (nod_log_file_desc, (int) 1,
(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();
2003-11-10 10:16:38 +01:00
lex.g_file->fil_name = (dsql_str*) $1; }
2001-05-23 15:26:42 +02:00
;
logfile_attrs :
| logfile_attrs logfile_attr
;
logfile_attr : KW_SIZE equals long_integer
{ lex.g_file->fil_length = (IPTR) $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
;
*/
2001-05-23 15:26:42 +02:00
file1 : KW_FILE
{ 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
{ lex.g_file->fil_start = (IPTR) $3;}
2001-05-23 15:26:42 +02:00
| LENGTH equals long_integer page_noise
{ lex.g_file->fil_length = (IPTR) $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 domain_default_opt
2001-05-23 15:26:42 +02:00
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
| current_user
| current_role
| internal_info
2001-05-23 15:26:42 +02:00
| null_value
| datetime_value_expression
;
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
{ $$ = make_node (nod_foreign, (int) e_for_count,
2003-09-04 23:26:15 +02:00
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, (int) 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, (int) 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
{ $$ = make_node (nod_foreign, (int) 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, (int) e_ref_upd_del_count, $1, NULL);}
2001-05-23 15:26:42 +02:00
| delete_rule
{ $$ = make_node (nod_ref_upd_del, (int) e_ref_upd_del_count, NULL, $1);}
2001-05-23 15:26:42 +02:00
| delete_rule update_rule
{ $$ = make_node (nod_ref_upd_del, (int) e_ref_upd_del_count, $2, $1); }
2001-05-23 15:26:42 +02:00
| update_rule delete_rule
{ $$ = make_node (nod_ref_upd_del, (int) e_ref_upd_del_count, $1, $2);}
2001-05-23 15:26:42 +02:00
| /* 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, (int) e_ref_trig_action_count, NULL);}
| SET DEFAULT
2001-05-23 15:26:42 +02:00
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_SET_DEFAULT, (int) e_ref_trig_action_count, NULL);}
2001-05-23 15:26:42 +02:00
| SET KW_NULL
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_SET_NULL, (int) e_ref_trig_action_count, NULL);}
2001-05-23 15:26:42 +02:00
| NO ACTION
{ $$ = make_flag_node (nod_ref_trig_action,
REF_ACTION_NONE, (int) e_ref_trig_action_count, NULL);}
2001-05-23 15:26:42 +02:00
;
/* PROCEDURE */
procedure_clause : symbol_procedure_name input_parameters
2003-09-04 23:26:15 +02:00
output_parameters
AS begin_string
local_declaration_list
2001-05-23 15:26:42 +02:00
full_proc_block
end_trigger
{ $$ = make_node (nod_def_procedure,
(int) e_prc_count, $1, $2, $3, $6, $7, $8); }
2003-09-04 23:26:15 +02:00
;
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
local_declaration_list
2002-06-29 08:56:51 +02:00
full_proc_block
end_trigger
{ $$ = make_node (nod_redef_procedure,
(int) e_prc_count, $1, $2, $3, $6, $7, $8); }
2003-09-04 23:26:15 +02:00
;
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
local_declaration_list
full_proc_block
end_trigger
{ $$ = make_node (nod_replace_procedure,
(int) e_prc_count, $1, $2, $3, $6, $7, $8); }
2003-09-04 23:26:15 +02:00
;
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
local_declaration_list
2001-05-23 15:26:42 +02:00
full_proc_block
end_trigger
{ $$ = make_node (nod_mod_procedure,
(int) e_prc_count, $1, $2, $3, $6, $7, $8); }
2003-09-04 23:26:15 +02:00
;
2001-05-23 15:26:42 +02:00
input_parameters : '(' input_proc_parameters ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
2001-05-23 15:26:42 +02:00
output_parameters : RETURNS '(' output_proc_parameters ')'
{ $$ = make_list ($3); }
|
{ $$ = NULL; }
;
2001-05-23 15:26:42 +02:00
input_proc_parameters : input_proc_parameter
| input_proc_parameters ',' input_proc_parameter
{ $$ = make_node (nod_list, 2, $1, $3); }
;
input_proc_parameter : simple_column_def_name non_array_type
default_par_opt end_trigger
{ $$ = make_node (nod_def_field, (int) e_dfl_count,
$1, $3, $4, NULL, NULL, NULL, NULL); }
;
output_proc_parameters : proc_parameter
| output_proc_parameters ',' proc_parameter
2001-05-23 15:26:42 +02:00
{ $$ = 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); }
;
default_par_opt : DEFAULT begin_string default_value
{ $$ = $3; }
| '=' begin_string default_value
{ $$ = $3; }
| begin_string
2004-04-10 02:25:22 +02:00
{ $$ = (dsql_nod*) NULL; }
;
local_declaration_list : local_declarations
{ $$ = make_list ($1); }
|
{ $$ = NULL; }
;
2001-05-23 15:26:42 +02:00
local_declarations : local_declaration
| local_declarations local_declaration
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_list, 2, $1, $2); }
;
local_declaration : DECLARE var_decl_opt local_declaration_item ';'
{ $$ = $3; }
;
local_declaration_item : var_declaration_item
| cursor_declaration_item
;
var_declaration_item : 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,
$1, $3, 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
;
cursor_declaration_item : symbol_cursor_name CURSOR FOR '(' select ')'
{ $$ = make_flag_node (nod_cursor, NOD_CURSOR_EXPLICIT,
(int) e_cur_count, $1, $5, NULL, NULL); }
;
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, (int) e_blk_count, make_list ($1), NULL); }
| proc_statements excp_hndl_statements
{ $$ = make_node (nod_block, (int) e_blk_count, make_list ($1), make_list ($2)); }
|
{ $$ = make_node (nod_block, (int) 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 ';'
| insert ';'
| update ';'
| delete ';'
| singleton_select ';'
| exec_procedure ';'
| exec_sql ';'
| exec_into ';'
| exec_udf ';'
| excp_statement ';'
| raise_statement ';'
| post_event ';'
| cursor_statement ';'
| breakleave ';'
2001-05-23 15:26:42 +02:00
| SUSPEND ';'
{ $$ = make_node (nod_return, (int) e_rtn_count, NULL); }
2001-05-23 15:26:42 +02:00
| EXIT ';'
{ $$ = make_node (nod_exit, 0, NULL); }
| if_then_else
| while
| for_select
| for_exec_into
2001-05-23 15:26:42 +02:00
;
excp_statement : EXCEPTION symbol_exception_name
{ $$ = make_node (nod_exception_stmt, (int) e_xcp_count, $2, NULL); }
| EXCEPTION symbol_exception_name value
{ $$ = make_node (nod_exception_stmt, (int) e_xcp_count, $2, $3); }
;
raise_statement : EXCEPTION
{ $$ = make_node (nod_exception_stmt, (int) e_xcp_count, NULL, NULL); }
2003-09-04 23:26:15 +02:00
;
exec_sql : EXECUTE STATEMENT value
{ $$ = make_node (nod_exec_sql, (int) e_exec_sql_count, $3); }
2002-04-04 15:53:20 +02:00
;
for_select : label_opt FOR select INTO variable_list cursor_def DO proc_block
{ $$ = make_node (nod_for_select, (int) 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 STATEMENT value INTO variable_list DO proc_block
{ $$ = make_node (nod_exec_into, (int) e_exec_into_count, $5, $9, make_list ($7), $1); }
;
exec_into : EXECUTE STATEMENT value INTO variable_list
{ $$ = make_node (nod_exec_into, (int) 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, (int) e_if_count, $3, $6, $8); }
2001-05-23 15:26:42 +02:00
| IF '(' search_condition ')' THEN proc_block
{ $$ = make_node (nod_if, (int) e_if_count, $3, $6, NULL); }
2001-05-23 15:26:42 +02:00
;
post_event : POST_EVENT value event_argument_opt
{ $$ = make_node (nod_post, (int) 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; }
;
singleton_select : select INTO variable_list
{ $$ = make_node (nod_for_select, (int) e_flp_count, $1,
2001-05-23 15:26:42 +02:00
make_list ($3), NULL, NULL); }
;
variable : ':' symbol_variable_name
{ $$ = make_node (nod_var_name, (int) e_vrn_count,
$2); }
;
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, (int) e_while_count, $4, $7, $1); }
2001-05-23 15:26:42 +02:00
;
label_opt : symbol_label_name ':'
{ $$ = make_node (nod_label, (int) e_label_count, $1, NULL); }
|
{ $$ = NULL; }
;
breakleave : KW_BREAK
{ $$ = make_node (nod_breakleave, (int) e_breakleave_count, NULL); }
| LEAVE
{ $$ = make_node (nod_breakleave, (int) e_breakleave_count, NULL); }
| LEAVE symbol_label_name
{ $$ = make_node (nod_breakleave, (int) e_breakleave_count,
make_node (nod_label, (int) e_label_count, $2, NULL)); }
;
2001-05-23 15:26:42 +02:00
cursor_def : AS CURSOR symbol_cursor_name
{ $$ = make_flag_node (nod_cursor, NOD_CURSOR_FOR,
(int) e_cur_count, $3, NULL, NULL, NULL); }
2001-05-23 15:26:42 +02:00
|
{ $$ = 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
{ $$ = make_node (nod_on_error, (int) e_err_count,
2001-05-23 15:26:42 +02:00
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); }
;
cursor_statement : open_cursor
| fetch_cursor
| close_cursor
;
open_cursor : OPEN symbol_cursor_name
{ $$ = make_node (nod_cursor_open, (int) e_cur_stmt_count, $2, NULL, NULL); }
;
close_cursor : CLOSE symbol_cursor_name
{ $$ = make_node (nod_cursor_close, (int) e_cur_stmt_count, $2, NULL, NULL); }
;
2001-05-23 15:26:42 +02:00
fetch_cursor : FETCH fetch_opt symbol_cursor_name INTO variable_list
{ $$ = make_node (nod_cursor_fetch, (int) e_cur_stmt_count, $3, $2, make_list ($5)); }
;
fetch_opt :
{ $$ = NULL; }
;
/*
fetch_opt : fetch_seek_opt FROM
;
fetch_seek_opt :
| FIRST
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_bof_forward, 0)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 3, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG)); }
| LAST
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_eof_backward, 0)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 4, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG)); }
| PRIOR
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_backward, 1)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 2, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)); }
| NEXT
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_forward, 1)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)); }
| ABSOLUTE value
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_bof_forward, value)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 3, CONSTANT_SLONG),
$2); }
| RELATIVE value
{ $$ = make_node (nod_fetch_seek, 2,
// corresponds to (blr_forward, value)
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG),
$2); }
;
*/
2001-05-23 15:26:42 +02:00
/* EXECUTE PROCEDURE */
exec_procedure : EXECUTE PROCEDURE symbol_procedure_name proc_inputs proc_outputs_opt
{ $$ = make_node (nod_exec_procedure, (int) e_exe_count,
$3, $4, $5); }
;
proc_inputs : value_list
{ $$ = make_list ($1); }
| '(' value_list ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
proc_outputs_opt : RETURNING_VALUES variable_list
{ $$ = make_list ($2); }
| RETURNING_VALUES '(' variable_list ')'
{ $$ = make_list ($3); }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
/* EXECUTE BLOCK */
exec_block : EXECUTE BLOCK block_input_params output_parameters AS
local_declaration_list
full_proc_block
{ $$ = make_node (nod_exec_block,
(int) e_exe_blk_count,
$3, $4, $6, $7, make_node (nod_all, (int) 0, NULL)); }
;
block_input_params : '(' block_parameters ')'
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
block_parameters : block_parameter
| block_parameters ',' block_parameter
{ $$ = make_node (nod_list, 2, $1, $3); }
;
block_parameter : proc_parameter '=' parameter
{ $$ = make_node (nod_param_val, e_prm_val_count, $1, $3); }
;
2001-05-23 15:26:42 +02:00
/* CREATE VIEW */
view_clause : symbol_view_name column_parens_opt AS begin_string select_expr
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
rview_clause : symbol_view_name column_parens_opt AS begin_string select_expr
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 select_expr
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 select_expr
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
;
*/
2001-05-23 15:26:42 +02:00
/* 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 :
{ $$ = (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 :
{ $$ = (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); }
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); }
;
2001-05-23 15:26:42 +02:00
trigger_active : ACTIVE
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| INACTIVE
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
|
{ $$ = NULL; }
;
trigger_type : trigger_type_prefix trigger_type_suffix
{ $$ = MAKE_trigger_type ($1, $2); }
;
trigger_type_prefix : BEFORE
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 0, CONSTANT_SLONG); }
| AFTER
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG); }
;
trigger_type_suffix : INSERT
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (1, 0, 0), CONSTANT_SLONG); }
| UPDATE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (2, 0, 0), CONSTANT_SLONG); }
| KW_DELETE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (3, 0, 0), CONSTANT_SLONG); }
| INSERT OR UPDATE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (1, 2, 0), CONSTANT_SLONG); }
| INSERT OR KW_DELETE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (1, 3, 0), CONSTANT_SLONG); }
| UPDATE OR INSERT
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (2, 1, 0), CONSTANT_SLONG); }
| UPDATE OR KW_DELETE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (2, 3, 0), CONSTANT_SLONG); }
| KW_DELETE OR INSERT
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (3, 1, 0), CONSTANT_SLONG); }
| KW_DELETE OR UPDATE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (3, 2, 0), CONSTANT_SLONG); }
| INSERT OR UPDATE OR KW_DELETE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (1, 2, 3), CONSTANT_SLONG); }
| INSERT OR KW_DELETE OR UPDATE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (1, 3, 2), CONSTANT_SLONG); }
| UPDATE OR INSERT OR KW_DELETE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (2, 1, 3), CONSTANT_SLONG); }
| UPDATE OR KW_DELETE OR INSERT
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (2, 3, 1), CONSTANT_SLONG); }
| KW_DELETE OR INSERT OR UPDATE
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (3, 1, 2), CONSTANT_SLONG); }
| KW_DELETE OR UPDATE OR INSERT
{ $$ = MAKE_constant ((dsql_str*)(IPTR) trigger_type_suffix (3, 2, 1), CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
trigger_position : POSITION nonneg_short_integer
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $2, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
|
{ $$ = NULL; }
;
trigger_action : AS begin_trigger local_declaration_list full_proc_block
{ $$ = make_node (nod_list, (int) e_trg_act_count, $3, $4); }
2001-05-23 15:26:42 +02:00
;
/* 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 alter_exception_clause
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
| 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)); }
2004-12-31 11:41:43 +01:00
| 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)); }
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); }
| SEQUENCE alter_sequence_clause
{ $$ = $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-11-10 10:16:38 +01:00
MAKE_constant ((dsql_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,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) $4, CONSTANT_SLONG)); }
2001-05-23 15:26:42 +02:00
| 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
| ADMIN /* added in IB 5.0 */
| COLUMN /* added in IB 6.0 */
| EXTRACT
| YEAR
| MONTH
| DAY
| HOUR
| MINUTE
| SECOND
| 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
| RECREATE
| CURRENT_CONNECTION /* added in FB 1.5 */
| CURRENT_TRANSACTION
| BIGINT
| CASE
| RELEASE
| ROW_COUNT
| SAVEPOINT
| OPEN /* added in FB 2.0 */
| CLOSE
| FETCH
| ROWS
| USING
| CROSS
;
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_sequence_clause : symbol_generator_name RESTART WITH signed_long_integer
{ $$ = make_node (nod_set_generator2, e_gen_id_count, $1,
MAKE_constant ((dsql_str*) $4, CONSTANT_SLONG)); }
2004-12-03 08:07:15 +01:00
| symbol_generator_name RESTART WITH NUMBER64BIT
{ $$ = make_node (nod_set_generator2, e_gen_id_count, $1,
MAKE_constant((dsql_str*) $4, CONSTANT_SINT64)); }
2004-12-03 08:07:15 +01:00
| symbol_generator_name RESTART WITH '-' NUMBER64BIT
{ $$ = make_node (nod_set_generator2, e_gen_id_count, $1,
make_node(nod_negate, 1, MAKE_constant((dsql_str*) $5, CONSTANT_SINT64))); }
;
2001-05-23 15:26:42 +02:00
/* ALTER DATABASE */
init_alter_db :
{ 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
;
/*
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
;
*/
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); }
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); }
| SEQUENCE 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
{ if ((IPTR) $1 < 1)
2003-09-04 23:26:15 +02:00
$$ = make_node (nod_list, (int) 2,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG));
2003-09-04 23:26:15 +02:00
else
$$ = make_node (nod_list, (int) 2,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG) ); }
2001-05-23 15:26:42 +02:00
| signed_long_integer ':' signed_long_integer
2003-09-04 23:26:15 +02:00
{ $$ = make_node (nod_list, (int) 2,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG),
MAKE_constant ((dsql_str*) $3, CONSTANT_SLONG)); }
2001-05-23 15:26:42 +02:00
;
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-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "BIGINT",
2003-09-04 23:26:15 +02:00
0);
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "BIGINT",
2003-09-04 23:26:15 +02:00
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;
2001-05-23 15:26:42 +02:00
if (client_dialect <= SQL_DIALECT_V5)
2003-09-04 23:26:15 +02:00
{
/* Post warning saying that DATE is equivalent to TIMESTAMP */
2003-11-08 00:27:24 +01:00
ERRD_post_warning (isc_sqlwarn, isc_arg_number, (SLONG) 301,
2003-09-04 23:26:15 +02:00
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-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
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;
lex.g_field->fld_length = sizeof(ISC_QUAD);
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_length = sizeof(ISC_QUAD);
lex.g_field->fld_seg_length = (USHORT)(IPTR) $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_length = sizeof(ISC_QUAD);
lex.g_field->fld_seg_length = (USHORT)(IPTR) $3;
lex.g_field->fld_sub_type = (USHORT)(IPTR) $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_length = sizeof(ISC_QUAD);
2003-05-23 18:55:40 +02:00
lex.g_field->fld_seg_length = 80;
lex.g_field->fld_sub_type = (USHORT)(IPTR) $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)(IPTR) $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)(IPTR) $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)(IPTR) $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)(IPTR) $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)(IPTR) $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)(IPTR) $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
{
if ( ((IPTR) $2 < 1) || ((IPTR) $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 ((IPTR) $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) ) )
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr,
isc_arg_number, (SLONG) -817,
isc_arg_gds,
2001-05-23 15:26:42 +02:00
isc_ddl_not_allowed_by_db_sql_dial,
2003-11-08 00:27:24 +01:00
isc_arg_number, (SLONG) db_dialect,
2001-05-23 15:26:42 +02:00
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,
2003-11-08 00:27:24 +01:00
isc_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,
2003-11-08 00:27:24 +01:00
isc_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,
2003-11-08 00:27:24 +01:00
isc_arg_end );
2001-05-23 15:26:42 +02:00
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
if ((IPTR) $2 < 5)
2003-09-04 23:26:15 +02:00
{
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)(IPTR) $2;
2001-05-23 15:26:42 +02:00
}
| '(' signed_long_integer ',' signed_long_integer ')'
{
if ( ((IPTR) $2 < 1) || ((IPTR) $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 (((IPTR) $4 > (IPTR) $2) || ((IPTR) $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 ((IPTR) $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) ) )
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr,
isc_arg_number, (SLONG) -817,
isc_arg_gds,
2001-05-23 15:26:42 +02:00
isc_ddl_not_allowed_by_db_sql_dial,
2003-11-08 00:27:24 +01:00
isc_arg_number, (SLONG) db_dialect,
2001-05-23 15:26:42 +02:00
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,
2003-11-08 00:27:24 +01:00
isc_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,
2003-11-08 00:27:24 +01:00
isc_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,
2003-11-08 00:27:24 +01:00
isc_arg_end );
2001-05-23 15:26:42 +02:00
}
/* 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 ((IPTR) $2 < 5)
2003-09-04 23:26:15 +02:00
{
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)(IPTR) $2;
lex.g_field->fld_scale = - (SSHORT)(IPTR) $4;
2001-05-23 15:26:42 +02:00
}
;
decimal_keyword : DECIMAL
| KW_DEC
;
/* floating point type */
float_type : KW_FLOAT precision_opt
{
if ((IPTR) $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 ((dsql_str*) $5, CONSTANT_SLONG)); }
2004-12-03 08:07:15 +01:00
| SET GENERATOR symbol_generator_name TO NUMBER64BIT
{ $$ = make_node (nod_set_generator2, e_gen_id_count, $3,
MAKE_constant((dsql_str*) $5, CONSTANT_SINT64)); }
| SET GENERATOR symbol_generator_name TO '-' NUMBER64BIT
{ $$ = make_node (nod_set_generator2, e_gen_id_count, $3,
2004-12-03 08:07:15 +01:00
make_node(nod_negate, 1, MAKE_constant((dsql_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
{ $$ = (dsql_nod*) NOD_SHARED; }
2001-05-23 15:26:42 +02:00
| PROTECTED
{ $$ = (dsql_nod*) NOD_PROTECTED ; }
2001-05-23 15:26:42 +02:00
|
{ $$ = (dsql_nod*) 0; }
2001-05-23 15:26:42 +02:00
;
lock_mode : READ
{ $$ = (dsql_nod*) NOD_READ; }
2001-05-23 15:26:42 +02:00
| WRITE
{ $$ = (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)(IPTR) $2 | (SSHORT)(IPTR) $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 : select_expr for_update_clause lock_clause
{ $$ = make_node (nod_select, (int) e_select_count, $1, $2, $3); }
;
for_update_clause : FOR UPDATE for_update_list
{ $$ = make_node (nod_for_update, 1, $3); }
2001-05-23 15:26:42 +02:00
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
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); }
|
{ $$ = NULL; }
2002-11-01 09:53:05 +01:00
;
2001-05-23 15:26:42 +02:00
/* SELECT expression */
select_expr : select_expr_body order_clause rows_clause
{ $$ = make_node (nod_select_expr, (int) e_sel_count, $1, $2, $3); }
;
column_select : select_expr_body order_clause rows_clause
{ $$ = make_flag_node (nod_select_expr, NOD_SELECT_EXPR_VALUE,
(int) e_sel_count, $1, $2, $3); }
;
column_singleton : select_expr_body order_clause rows_clause
{ $$ = make_flag_node (nod_select_expr, NOD_SELECT_EXPR_VALUE | NOD_SELECT_EXPR_SINGLETON,
(int) e_sel_count, $1, $2, $3); }
;
select_expr_body : query_term
| select_expr_body UNION distinct_noise query_term
{ $$ = make_node (nod_list, 2, $1, $4); }
| select_expr_body UNION ALL query_term
{ $$ = make_flag_node (nod_list, NOD_UNION_ALL, 2, $1, $4); }
;
query_term : query_spec
;
query_spec : 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
{ $$ = make_node (nod_query_spec, (int) e_qry_count,
$2, $3, $4, $5, $6, $7, $8, $9); }
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
{ $$ = make_node (nod_limit, (int) e_limit_count, $2, $1); }
2003-05-23 18:55:40 +02:00
| first_clause end_limit
{ $$ = make_node (nod_limit, (int) e_limit_count, NULL, $1); }
2002-11-01 09:53:05 +01:00
| skip_clause
{ $$ = make_node (nod_limit, (int) e_limit_count, $1, NULL); }
2002-11-01 09:53:05 +01:00
|
{ $$ = 0; }
;
2003-05-23 18:55:40 +02:00
first_clause : FIRST long_integer begin_limit
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_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
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_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 : value
| value 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_primary
;
table_primary : table_proc
| derived_table
| '(' joined_table ')'
{ $$ = $2; }
;
2003-08-15 02:02:18 +02:00
/* AB: derived table support */
derived_table :
'(' select_expr ')' 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 : cross_join
| natural_join
| qualified_join
;
cross_join : table_reference CROSS JOIN table_primary
{ $$ = make_node (nod_join, (int) e_join_count, $1,
make_node (nod_join_inner, (int) 0, NULL), $4, NULL); }
;
natural_join : table_reference NATURAL join_type JOIN table_primary
{ $$ = make_node (nod_join, (int) e_join_count, $1, $3, $5,
make_node (nod_flag, 0, NULL)); }
;
qualified_join : table_reference join_type JOIN table_reference join_specification
{ $$ = make_node (nod_join, (int) e_join_count, $1, $2, $4, $5); }
;
join_specification : join_condition
| named_columns_join
;
join_condition : ON search_condition
2001-05-23 15:26:42 +02:00
{ $$ = $2; }
;
named_columns_join : USING '(' column_list ')'
{ $$ = make_list ($3); }
;
table_proc : symbol_procedure_name table_proc_inputs as_noise symbol_table_alias_name
{ $$ = make_node (nod_rel_proc_name,
(int) e_rpn_count, $1, $4, $2); }
| symbol_procedure_name table_proc_inputs
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_rel_proc_name,
(int) e_rpn_count, $1, NULL, $2); }
;
table_proc_inputs : '(' value_list ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($2); }
|
{ $$ = NULL; }
;
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 outer_noise
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_join_left, (int) 0, NULL); }
| RIGHT outer_noise
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_join_right, (int) 0, NULL); }
| FULL outer_noise
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_join_full, (int) 0, NULL); }
|
{ $$ = make_node (nod_join_inner, (int) 0, NULL); }
;
outer_noise : OUTER
|
;
2001-05-23 15:26:42 +02:00
/* other clauses in the select expression */
group_clause : GROUP BY group_by_list
2001-05-23 15:26:42 +02:00
{ $$ = make_list ($3); }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
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; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
where_clause : WHERE search_condition
{ $$ = $2; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
/* PLAN clause to specify an access plan for a query */
plan_clause : PLAN plan_expression
{ $$ = $2; }
|
{ $$ = NULL; }
2001-05-23 15:26:42 +02:00
;
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
/* ORDER BY clause */
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); }
;
order_item : value order_direction nulls_clause
{ $$ = make_node (nod_order, (int) e_order_count, $1, $2, $3); }
;
order_direction : ASC
{ $$ = 0; }
| DESC
{ $$ = make_node (nod_flag, 0, NULL); }
|
{ $$ = 0; }
;
nulls_placement : FIRST
{ $$ = MAKE_constant((dsql_str*) NOD_NULLS_FIRST, CONSTANT_SLONG); }
| LAST
{ $$ = MAKE_constant((dsql_str*) NOD_NULLS_LAST, CONSTANT_SLONG); }
;
nulls_clause : NULLS begin_first nulls_placement end_first
{ $$ = $3; }
|
{ $$ = 0; }
;
/* ROWS clause */
rows_clause : ROWS value
/* equivalent to FIRST value */
{ $$ = make_node (nod_rows, (int) e_rows_count, NULL, $2); }
| ROWS value TO value
/* equivalent to FIRST (upper_value - lower_value + 1) SKIP (lower_value - 1) */
{ $$ = make_node (nod_rows, (int) e_rows_count,
make_node (nod_subtract, 2, $2,
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)),
make_node (nod_add, 2,
make_node (nod_subtract, 2, $4, $2),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG))); }
|
{ $$ = NULL; }
;
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 '(' value_list ')'
{ $$ = make_node (nod_insert, (int) e_ins_count,
2001-05-23 15:26:42 +02:00
$3, make_list ($4), make_list ($7), NULL); }
| INSERT INTO simple_table_name ins_column_parens_opt select_expr
{ $$ = make_node (nod_insert, (int) e_ins_count, $3, $4, NULL, $5); }
2001-05-23 15:26:42 +02:00
;
/* DELETE statement */
2001-05-23 15:26:42 +02:00
delete : delete_searched
| delete_positioned
;
delete_searched : KW_DELETE FROM table_name where_clause plan_clause order_clause rows_clause
{ $$ = make_node (nod_delete, (int) e_del_count, $3, $4, $5, $6, $7, NULL); }
2001-05-23 15:26:42 +02:00
;
delete_positioned : KW_DELETE FROM table_name cursor_clause
{ $$ = make_node (nod_delete, (int) e_del_count, $3, NULL, NULL, NULL, NULL, $4); }
2001-05-23 15:26:42 +02:00
;
/* UPDATE statement */
update : update_searched
| update_positioned
;
update_searched : UPDATE table_name SET assignments where_clause plan_clause order_clause rows_clause
{ $$ = make_node (nod_update, (int) e_upd_count,
$2, make_list ($4), $5, $6, $7, $8, 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, (int) e_upd_count,
$2, make_list ($4), NULL, NULL, NULL, NULL, $5); }
2001-05-23 15:26:42 +02:00
;
cursor_clause : WHERE CURRENT OF symbol_cursor_name
{ $$ = make_node (nod_cursor, (int) e_cur_count, $4, NULL, NULL, NULL); }
;
/* Assignments */
2001-05-23 15:26:42 +02:00
assignments : assignment
| assignments ',' assignment
{ $$ = make_node (nod_list, 2, $1, $3); }
;
assignment : update_column_name '=' value
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_assign, 2, $3, $1); }
;
exec_udf : udf
{ $$ = make_node (nod_assign, 2, $1, make_node (nod_null, 0, NULL)); }
;
2001-05-23 15:26:42 +02:00
/* 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
{ $$ = make_node (nod_get_segment, (int) 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
{ $$ = make_node (nod_put_segment, (int) e_blb_count, $3, $5, $6, $7); }
2001-05-23 15:26:42 +02:00
;
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
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
segment_clause : MAX_SEGMENT segment_length
{ $$ = $2; }
|
;
segment_length : unsigned_short_integer
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| 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
| distinct_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
| singular_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
;
2001-05-23 15:26:42 +02:00
/* other predicates */
distinct_predicate : value IS DISTINCT FROM value
{ $$ = make_node (nod_not, 1, make_node (nod_equiv, 2, $1, $5)); }
| value IS NOT DISTINCT FROM value
{ $$ = make_node (nod_equiv, 2, $1, $6); }
;
2001-05-23 15:26:42 +02:00
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)); }
;
2001-05-23 15:26:42 +02:00
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)); }
;
2001-05-23 15:26:42 +02:00
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)); }
;
2001-05-23 15:26:42 +02:00
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)); }
;
2001-05-23 15:26:42 +02:00
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)); }
;
2001-05-23 15:26:42 +02:00
exists_predicate : EXISTS '(' select_expr ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_exists, 1, $3); }
;
2001-05-23 15:26:42 +02:00
singular_predicate : SINGULAR '(' select_expr ')'
2001-05-23 15:26:42 +02:00
{ $$ = make_node (nod_singular, 1, $3); }
;
2001-05-23 15:26:42 +02:00
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, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)); }
| UPDATING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 2, CONSTANT_SLONG)); }
| DELETING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 3, CONSTANT_SLONG)); }
;
special_trigger_action_predicate : KW_INSERTING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)); }
| KW_UPDATING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 2, CONSTANT_SLONG)); }
| KW_DELETING
{ $$ = make_node (nod_eql, 2,
make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_trigger_action, CONSTANT_SLONG)),
MAKE_constant ((dsql_str*) 3, CONSTANT_SLONG)); }
;
2001-05-23 15:26:42 +02:00
/* set values */
in_predicate_value : table_subquery
| '(' value_list ')'
{ $$ = make_list ($2); }
;
table_subquery : '(' column_select ')'
{ $$ = $2; }
2001-05-23 15:26:42 +02:00
;
/* value types */
value : column_name
2001-05-23 15:26:42 +02:00
| array_element
| function
| u_constant
| parameter
| variable
| cast_specification
| case_expression
| next_value_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
{ $$ = make_node (nod_collate, (int) 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
| null_value
2001-05-23 15:26:42 +02:00
;
datetime_value_expression : CURRENT_DATE
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "DATE",
2003-09-04 23:26:15 +02:00
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "DATE",
2003-09-04 23:26:15 +02:00
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-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
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, (int) e_ary_count, $1, make_list ($3)); }
2001-05-23 15:26:42 +02:00
;
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
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_STRING); }
2001-05-23 15:26:42 +02:00
| NUMBER
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SLONG); }
| FLOAT_NUMBER
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_DOUBLE); }
2001-05-23 15:26:42 +02:00
| NUMBER64BIT
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SINT64); }
2001-05-23 15:26:42 +02:00
| SCALEDINT
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $1, CONSTANT_SINT64); }
2001-05-23 15:26:42 +02:00
;
u_constant : u_numeric_constant
| sql_string
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_str_constant ((dsql_str*) $1, lex.att_charset); }
2001-05-23 15:26:42 +02:00
| DATE STRING
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "DATE",
2003-09-04 23:26:15 +02:00
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "DATE",
2003-09-04 23:26:15 +02:00
0);
2003-11-10 10:16:38 +01:00
$$ = MAKE_constant ((dsql_str*) $2, CONSTANT_DATE);
2001-05-23 15:26:42 +02:00
}
| TIME STRING
{
if (client_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_dialect_datatype_unsupport,
isc_arg_number, client_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
0);
2001-05-23 15:26:42 +02:00
if (db_dialect < SQL_DIALECT_V6_TRANSITION)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
isc_arg_gds, isc_sql_db_dialect_dtype_unsupport,
isc_arg_number, db_dialect,
isc_arg_string, "TIME",
2003-09-04 23:26:15 +02:00
0);
2003-11-10 10:16:38 +01:00
$$ = MAKE_constant ((dsql_str*) $2, CONSTANT_TIME);
2001-05-23 15:26:42 +02:00
}
| TIMESTAMP STRING
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*) $2, CONSTANT_TIMESTAMP); }
2001-05-23 15:26:42 +02:00
;
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, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_connection_id, CONSTANT_SLONG)); }
| CURRENT_TRANSACTION
{ $$ = make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_transaction_id, CONSTANT_SLONG)); }
| GDSCODE
{ $$ = make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_gdscode, CONSTANT_SLONG)); }
| SQLCODE
{ $$ = make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) internal_sqlcode, CONSTANT_SLONG)); }
| ROW_COUNT
{ $$ = make_node (nod_internal_info, (int) e_internal_info_count,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_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 */
2003-11-10 10:16:38 +01:00
{ ((dsql_str*) $2)->str_charset = (TEXT *) $1;
2001-05-23 15:26:42 +02:00
$$ = $2; }
;
signed_short_integer : nonneg_short_integer
| '-' neg_short_integer
{ $$ = (dsql_nod*) - (IPTR) $2; }
2001-05-23 15:26:42 +02:00
;
nonneg_short_integer : NUMBER
{ if ((IPTR) $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 ((IPTR) $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 ((IPTR) $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 ((IPTR) $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
{ $$ = (dsql_nod*) - (IPTR) $2; }
2001-05-23 15:26:42 +02:00
;
long_integer : NUMBER
{ $$ = $1;}
;
/* functions */
function : aggregate_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); }
;
numeric_value_function : extract_expression
;
extract_expression : EXTRACT '(' timestamp_part FROM value ')'
{ $$ = make_node (nod_extract, (int) 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, (int) e_substr_count, $3,
make_node (nod_subtract, 2, $5,
2003-11-10 10:16:38 +01:00
MAKE_constant ((dsql_str*) 1, CONSTANT_SLONG)), $6); }
;
string_length_opt : FOR value
{ $$ = $2; }
|
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_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 '(' value AS data_type_descriptor ')'
{ $$ = make_node (nod_cast, (int) 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); }
| IIF '(' search_condition ',' value ',' value ')'
{ $$ = make_node (nod_searched_case, 2,
make_node (nod_list, 2, $3, $5), $7); }
| COALESCE '(' value ',' 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 : value
;
/* next value expression */
next_value_expression : NEXT KW_VALUE FOR symbol_generator_name
{ $$ = make_node (nod_gen_id, 2, $4,
MAKE_constant((dsql_str*) 1, CONSTANT_SLONG)); }
| GEN_ID '(' symbol_generator_name ',' value ')'
{
if (client_dialect >= SQL_DIALECT_V6_TRANSITION)
$$ = make_node (nod_gen_id2, 2, $3, $5);
else
$$ = make_node (nod_gen_id, 2, $3, $5);
}
;
2001-05-23 15:26:42 +02:00
timestamp_part : YEAR
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_year, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| MONTH
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_month, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| DAY
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_day, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| HOUR
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_hour, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| MINUTE
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_minute, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| SECOND
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_second, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| WEEKDAY
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_weekday, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
| YEARDAY
2003-11-10 10:16:38 +01:00
{ $$ = MAKE_constant ((dsql_str*)blr_extract_yearday, CONSTANT_SLONG); }
2001-05-23 15:26:42 +02:00
;
all_noise : ALL
|
;
distinct_noise : DISTINCT
|
;
2001-05-23 15:26:42 +02:00
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 :
ACTION /* added in IB 5.0 */
| CASCADE
| FREE_IT
| RESTRICT
| ROLE
| TYPE /* added in IB 6.0 */
| KW_BREAK /* added in FB 1.0 */
| KW_DESCRIPTOR
| SUBSTRING
| COALESCE /* added in FB 1.5 */
| LAST
| LEAVE
| LOCK
| NULLIF
| NULLS
| STATEMENT
2003-09-04 23:26:15 +02:00
| INSERTING
| UPDATING
| DELETING
/* | FIRST | SKIP -- this is handled by the lexer. */
| BLOCK
2003-09-04 23:26:15 +02:00
| BACKUP /* added in FB 2.0 */
| KW_DIFFERENCE
| IIF
| SCALAR_ARRAY
| WEEKDAY
| YEARDAY
| SEQUENCE
| NEXT
| RESTART
;
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;
dsql_str* str = FB_NEW_RPT(*DSQL_permanent_pool, symbol->sym_length) dsql_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 (
const TEXT* string,
2003-09-04 23:26:15 +02:00
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 const TEXT* lex_position (void)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* 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
static bool long_int(dsql_nod* string,
2003-09-04 23:26:15 +02:00
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 UCHAR* p = (UCHAR*)((dsql_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(((dsql_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
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.
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
2003-09-04 23:26:15 +02:00
if (field_name == NULL)
{
2003-11-10 10:16:38 +01:00
dsql_fld* field =
FB_NEW_RPT(*tdsql->getDefaultPool(), sizeof (INTERNAL_FIELD_NAME)) dsql_fld;
2004-10-08 13:08:42 +02:00
strcpy (field->fld_name, INTERNAL_FIELD_NAME);
2003-09-04 23:26:15 +02:00
return field;
}
2003-11-10 10:16:38 +01:00
const dsql_str* string = (dsql_str*) field_name->nod_arg[1];
dsql_fld* field =
FB_NEW_RPT(*tdsql->getDefaultPool(), strlen ((SCHAR*) string->str_data)) dsql_fld;
2003-09-04 23:26:15 +02:00
strcpy (field->fld_name, (TEXT*) string->str_data);
return field;
2001-05-23 15:26:42 +02:00
}
static dsql_fil* make_file()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a k e _ f i l e
*
**************************************
*
* Functional description
* Make a file block
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
2003-09-04 23:26:15 +02:00
dsql_fil* temp_file = FB_NEW(*tdsql->getDefaultPool()) dsql_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
}
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.
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
2001-05-23 15:26:42 +02:00
2004-10-03 13:39:03 +02:00
if (node)
{
DsqlNodStack stack;
stack_nodes(node, stack);
USHORT l = stack.getCount();
dsql_nod* old = node;
node = FB_NEW_RPT(*tdsql->getDefaultPool(), l) dsql_nod;
node->nod_count = l;
node->nod_type = nod_list;
if (MemoryPool::blk_type(old) == dsql_type_nod)
{
node->nod_flags = old->nod_flags;
}
dsql_nod** ptr = node->nod_arg + node->nod_count;
2001-05-23 15:26:42 +02:00
2004-10-03 13:39:03 +02:00
while (stack.hasData())
*--ptr = stack.pop();
}
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
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
dsql_nod* node = FB_NEW_RPT(*tdsql->getDefaultPool(), 1) dsql_nod;
2003-09-04 23:26:15 +02:00
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*)(IPTR) lex.param_number++;
2003-09-04 23:26:15 +02:00
return node;
}
static dsql_nod* make_node (NOD_TYPE type,
2003-09-04 23:26:15 +02:00
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
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
2001-05-23 15:26:42 +02:00
dsql_nod* node = FB_NEW_RPT(*tdsql->getDefaultPool(), count) dsql_nod;
2003-09-04 23:26:15 +02:00
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;
2003-11-10 10:16:38 +01:00
dsql_nod** p = node->nod_arg;
va_list ptr;
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
va_end(ptr);
2003-09-04 23:26:15 +02:00
return node;
2001-05-23 15:26:42 +02:00
}
static dsql_nod* make_flag_node (NOD_TYPE type,
2003-09-04 23:26:15 +02:00
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
*
**************************************/
tsql* tdsql = DSQL_get_thread_data();
2001-05-23 15:26:42 +02:00
dsql_nod* node = FB_NEW_RPT(*tdsql->getDefaultPool(), count) dsql_nod;
2003-09-04 23:26:15 +02:00
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;
2003-11-10 10:16:38 +01:00
dsql_nod** p = node->nod_arg;
va_list ptr;
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
va_end(ptr);
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
static bool short_int(dsql_nod* string,
2003-09-04 23:26:15 +02:00
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?
*
*************************************/
2003-11-10 10:16:38 +01:00
if (((dsql_str*) string)->str_length > 5) {
2003-09-04 23:26:15 +02:00
return false;
}
2001-05-23 15:26:42 +02:00
for (UCHAR* p = (UCHAR*)((dsql_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];
2003-11-10 10:16:38 +01:00
buf[0] = ((dsql_str*) string)->str_data[0];
buf[1] = ((dsql_str*) string)->str_data[1];
buf[2] = ((dsql_str*) string)->str_data[2];
buf[3] = ((dsql_str*) string)->str_data[3];
buf[4] = ((dsql_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
static void stack_nodes (dsql_nod* node,
DsqlNodStack& 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
if (node->nod_type != nod_list)
{
stack.push(node);
2003-09-04 23:26:15 +02:00
return;
}
2003-09-04 23:26:15 +02:00
/* 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. */
2003-11-10 10:16:38 +01:00
dsql_nod* start_chain = node;
dsql_nod* end_chain = NULL;
dsql_nod* curr_node = node;
dsql_nod* next_node = node->nod_arg[0];
2003-09-04 23:26:15 +02:00
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)
2003-11-10 10:16:38 +01:00
{
2003-09-04 23:26:15 +02:00
/* pattern was found so reverse the links and go to next node */
2003-11-10 10:16:38 +01:00
dsql_nod* save_link = next_node->nod_arg[0];
2003-09-04 23:26:15 +02:00
next_node->nod_arg[0] = curr_node;
curr_node = next_node;
next_node = save_link;
end_chain = curr_node;
2003-11-10 10:16:38 +01:00
}
2003-09-04 23:26:15 +02:00
/* see if any chain was found */
2003-11-10 10:16:38 +01:00
if (end_chain)
{
2003-09-04 23:26:15 +02:00
/* first, handle the rest of the nodes */
/* note that next_node still points to the first non-pattern node */
stack_nodes (next_node, stack);
2003-09-04 23:26:15 +02:00
/* stack the non-list nodes and reverse the chain on the way back */
curr_node = end_chain;
while (true)
2003-11-10 10:16:38 +01:00
{
stack.push(curr_node->nod_arg[1]);
2003-09-04 23:26:15 +02:00
if ( curr_node == start_chain)
break;
2003-11-10 10:16:38 +01:00
dsql_nod* save_link = curr_node->nod_arg[0];
2003-09-04 23:26:15 +02:00
curr_node->nod_arg[0] = next_node;
next_node = curr_node;
curr_node = save_link;
}
2003-11-10 10:16:38 +01:00
return;
}
2003-09-04 23:26:15 +02:00
2003-11-10 10:16:38 +01:00
dsql_nod** ptr = node->nod_arg;
for (const dsql_nod* const* const end = ptr + node->nod_count; ptr < end; ptr++)
2003-09-04 23:26:15 +02:00
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,
bool* stmt_ambiguous)
2003-05-23 18:55:40 +02:00
{
2003-11-10 10:16:38 +01:00
const 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,
bool* 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];
SSHORT c;
/* 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-11-10 10:16:38 +01: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;
2003-11-10 10:16:38 +01:00
}
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)
2003-11-10 10:16:38 +01:00
{
2003-09-04 23:26:15 +02:00
/* The Introducer (_) is skipped, all other idents are copied
* to become the name of the character set
*/
char* p = string;
for (; ptr < end && classes[static_cast<UCHAR>(*ptr)] & CHR_IDENT; ptr++)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
if (ptr >= end)
return -1;
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
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;
2003-11-10 10:16:38 +01:00
}
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)
2003-11-10 10:16:38 +01:00
{
char* buffer = string;
size_t buffer_len = sizeof (string);
const char* buffer_end = buffer + buffer_len - 1;
char* p;
for (p = buffer; ; ++p)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
if (ptr >= end)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
if (buffer != string)
gds__free (buffer);
return -1;
2003-09-04 23:26:15 +02:00
}
// Care about multi-line constants and identifiers
if (*ptr == '\n') {
lines++;
2004-04-10 02:25:22 +02:00
line_start = ptr + 1;
}
2003-11-10 10:16:38 +01:00
/* *ptr is quote - if next != quote we're at the end */
if ((*ptr == c) && ((++ptr == end) || (*ptr != c)))
break;
2003-09-04 23:26:15 +02:00
if (p > buffer_end)
{
char* const new_buffer = (char*) gds__alloc (2 * buffer_len);
2003-09-04 23:26:15 +02:00
/* FREE: at outer block */
2003-11-10 10:16:38 +01:00
if (!new_buffer) /* NOMEM: */
{
if (buffer != string)
gds__free (buffer);
return -1;
}
2003-09-04 23:26:15 +02:00
memcpy (new_buffer, buffer, buffer_len);
2003-11-10 10:16:38 +01:00
if (buffer != string)
gds__free (buffer);
buffer = new_buffer;
p = buffer + buffer_len;
buffer_len = 2 * buffer_len;
buffer_end = buffer + buffer_len - 1;
2003-09-04 23:26:15 +02:00
}
2003-11-10 10:16:38 +01:00
*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
{
*stmt_ambiguous = true; /* string delimited by double quotes could be
2003-09-04 23:26:15 +02:00
** either a string constant or a SQL delimited
** identifier, therefore marks the SQL
** statement as ambiguous */
2003-11-10 10:16:38 +01:00
if (client_dialect == SQL_DIALECT_V6_TRANSITION)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
if (buffer != string)
gds__free (buffer);
yyabandon (-104, isc_invalid_string_constant);
2003-09-04 23:26:15 +02:00
}
2003-11-10 10:16:38 +01:00
else if (client_dialect >= SQL_DIALECT_V6)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
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);
2003-11-10 10:16:38 +01:00
dsql_str* delimited_id_str = (dsql_str*) yylval;
delimited_id_str->str_flags |= STR_delimited_id;
if (buffer != string)
gds__free (buffer);
return SYMBOL;
2003-09-04 23:26:15 +02:00
}
2001-05-23 15:26:42 +02:00
}
yylval = (dsql_nod*) MAKE_string(buffer, p - buffer);
2003-09-04 23:26:15 +02:00
if (buffer != string)
2001-05-23 15:26:42 +02:00
gds__free (buffer);
2003-09-04 23:26:15 +02:00
return STRING;
2003-11-10 10:16:38 +01:00
}
2003-09-04 23:26:15 +02:00
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-11-04 00:59:24 +01:00
fb_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[static_cast<UCHAR>(*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;
2003-11-10 10:16:38 +01:00
if (have_exp_digit && (! (classes[c] & CHR_DIGIT)))
2003-09-04 23:26:15 +02:00
/* First non-digit after exponent and digit terminates
the token. */
break;
2003-11-10 10:16:38 +01:00
else if (have_exp_sign && (! (classes[c] & CHR_DIGIT)))
2003-09-04 23:26:15 +02:00
{
/* 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;
2003-11-10 10:16:38 +01:00
else if ( classes[c] & CHR_DIGIT )
2003-09-04 23:26:15 +02:00
/* 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;
}
}
2003-11-10 10:16:38 +01:00
else if (classes[c] & CHR_DIGIT)
2003-09-04 23:26:15 +02:00
{
/* 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)
2003-11-10 10:16:38 +01:00
{
2003-09-04 23:26:15 +02:00
/* possibility of an overflow */
if ((number > limit_by_10) || (c > '8'))
{
have_error = true;
break;
}
2003-11-10 10:16:38 +01:00
}
2003-09-04 23:26:15 +02:00
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-11-04 00:59:24 +01:00
fb_assert(have_digit);
2001-05-23 15:26:42 +02:00
2003-09-04 23:26:15 +02:00
if (have_exp_digit)
2003-11-10 10:16:38 +01:00
{
yylval = (dsql_nod*) MAKE_string(last_token, ptr - last_token);
2003-09-04 23:26:15 +02:00
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;
2003-11-10 10:16:38 +01:00
}
2003-09-04 23:26:15 +02:00
else if (!have_exp)
2003-11-10 10:16:38 +01:00
{
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*) (IPTR) number;
2003-11-10 10:16:38 +01:00
return NUMBER;
2003-09-04 23:26:15 +02:00
}
else
{
2003-11-10 10:16:38 +01:00
/* 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)
2003-09-04 23:26:15 +02:00
{
2003-11-10 10:16:38 +01:00
/* 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,
2003-11-08 00:27:24 +01:00
isc_arg_string,
2003-11-10 10:16:38 +01:00
ERR_string( last_token, ptr - last_token ),
2003-11-08 00:27:24 +01:00
isc_arg_end );
2003-11-10 10:16:38 +01:00
ERRD_post_warning( isc_dsql_warning_number_ambiguous1,
2003-11-08 00:27:24 +01:00
isc_arg_end );
2003-09-04 23:26:15 +02:00
}
2001-05-23 15:26:42 +02:00
yylval = (dsql_nod*) MAKE_string(last_token, ptr - last_token);
2001-05-23 15:26:42 +02:00
2003-11-10 10:16:38 +01:00
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
2001-05-23 15:26:42 +02:00
2003-11-10 10:16:38 +01: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)
{
char* p = string;
2003-10-15 00:22:32 +02:00
check_copy_incr(p, UPPER (c), string);
for (; ptr < end && classes[static_cast<UCHAR>(*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;
dsql_sym* sym =
HSHD_lookup (NULL, (TEXT *) string, (SSHORT)(p - string), SYM_keyword, parser_version);
2003-09-04 23:26:15 +02:00
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) )
{
const LexerState savedState = lex;
const int nextToken = yylex(client_dialect, db_dialect, parser_version, stmt_ambiguous);
lex = savedState;
if (nextToken == OR || nextToken == AND) {
switch (sym->sym_keyword) {
case INSERTING:
yylval = (dsql_nod*) sym->sym_object;
return KW_INSERTING;
case UPDATING:
yylval = (dsql_nod*) sym->sym_object;
return KW_UPDATING;
case DELETING:
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
*/
2003-11-10 10:16:38 +01:00
if ((sym->sym_keyword == FIRST && !first_detection) ||
sym->sym_keyword == SKIP)
{
2003-09-04 23:26:15 +02:00
if (prev_keyword == SELECT || limit_clause) {
const LexerState savedState = lex;
const int nextToken = yylex(client_dialect, db_dialect, parser_version, stmt_ambiguous);
2003-09-04 23:26:15 +02:00
lex = savedState;
if (nextToken != NUMBER && nextToken != '?' && nextToken != '(') {
yylval = (dsql_nod*) MAKE_string(string, p - string);
2003-09-04 23:26:15 +02:00
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return SYMBOL;
}
else {
yylval = (dsql_nod*) sym->sym_object;
2003-09-04 23:26:15 +02:00
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 */
2003-11-10 10:16:38 +01:00
}
else {
yylval = (dsql_nod*) sym->sym_object;
2003-09-04 23:26:15 +02:00
last_token_bk = last_token;
line_start_bk = line_start;
lines_bk = lines;
return sym->sym_keyword;
}
}
yylval = (dsql_nod*) MAKE_string(string, p - string);
2003-09-04 23:26:15 +02:00
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)
{
dsql_sym* sym =
HSHD_lookup (NULL, last_token, (SSHORT) 2, SYM_keyword, (USHORT) parser_version);
2003-09-04 23:26:15 +02:00
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) )
2003-09-04 23:26:15 +02:00
{
LexerState savedState = lex;
brace_analysis = true;
int openCount = 0;
int nextToken;
do {
openCount++;
nextToken = yylex(client_dialect, db_dialect, parser_version, stmt_ambiguous);
2003-09-04 23:26:15 +02:00
} while (nextToken == '(');
dsql_nod* temp_val = yylval;
2003-09-04 23:26:15 +02:00
if (nextToken == INSERTING || nextToken == UPDATING || nextToken == DELETING)
{
/* Skip closing braces. */
while ( openCount &&
yylex(client_dialect, db_dialect,
parser_version, stmt_ambiguous) == ')')
2003-09-04 23:26:15 +02:00
{
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) )
{
2003-09-04 23:26:15 +02:00
savedState = lex;
const int token = yylex(client_dialect, db_dialect, parser_version, stmt_ambiguous);
2003-09-04 23:26:15 +02:00
lex = savedState;
if (token == OR || token == AND) {
switch (nextToken) {
2003-09-04 23:26:15 +02:00
case INSERTING:
return KW_INSERTING;
case UPDATING:
return KW_UPDATING;
case DELETING:
return KW_DELETING;
}
}
}
return nextToken;
}
2003-11-10 10:16:38 +01:00
}
else {
2003-09-04 23:26:15 +02:00
/* Restore status quo. */
lex = savedState;
}
}
/* Single character punctuation are simply passed on */
return c;
2001-05-23 15:26:42 +02:00
}
2003-11-10 10:16:38 +01:00
// The argument passed to this function is ignored. Therefore, messages like
// "syntax error" and "yacc stack overflow" are never seen.
static void yyerror(const 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)
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
2003-11-10 10:16:38 +01:00
isc_arg_gds, isc_command_end_err, /* Unexpected end of command */
0);
2003-09-04 23:26:15 +02:00
else
{
2003-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) -104,
2003-11-10 10:16:38 +01:00
isc_arg_gds, isc_dsql_token_unk_err,
isc_arg_number, (SLONG) lex.lines,
isc_arg_number, (SLONG) (lex.last_token - lex.line_start + 1), /*CVC: +1*/
2003-09-04 23:26:15 +02:00
/* Token unknown - line %d, char %d */
2003-11-10 10:16:38 +01:00
isc_arg_gds, isc_random,
isc_arg_cstring, (int) (lex.ptr - lex.last_token), lex.last_token, 0);
2003-09-04 23:26:15 +02:00
}
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-11-08 00:27:24 +01:00
ERRD_post (isc_sqlerr, isc_arg_number, (SLONG) sql_code,
isc_arg_gds, error_symbol, 0);
2001-05-23 15:26:42 +02:00
}
2003-11-10 10:16:38 +01:00