8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 00:43:03 +01:00
firebird-mirror/src/qli/parse.cpp
brodsom 2bcfbe983f - ifdef unused or conditioned code
- Correct formats
- Change NULL to 0 or FALSE
2003-04-03 09:34:54 +00:00

5696 lines
118 KiB
C++

/*
* PROGRAM: JRD Command Oriented Query Language
* MODULE: parse.c
* DESCRIPTION: Statement 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): ______________________________________.
*/
#include "firebird.h"
#include "../jrd/ib_stdio.h"
#include <stdlib.h>
#include <string.h>
#define PARSER_MAIN
#include "../qli/dtr.h"
#include "../qli/exe.h" /* This is only included to suppress a compiler warning */
#include "../qli/parse.h"
#include "../qli/compile.h"
#include "../qli/report.h"
#include "../qli/all_proto.h"
#include "../qli/err_proto.h"
#include "../qli/hsh_proto.h"
#include "../qli/lex_proto.h"
#include "../qli/mov_proto.h"
#include "../qli/parse_proto.h"
#include "../qli/proc_proto.h"
#define SYNTAX_NODE(type,count) syntax_node (type, count)
#define KEYWORD(kw) (QLI_token->tok_keyword == kw)
#define INT_CAST (SYN) (IPTR)
static void check_end(void);
static void command_end(void);
static DBB get_dbb(SYM);
static SYN make_list(LLS);
static NAM make_name(void);
static CON make_numeric_constant(TEXT *, USHORT);
static TEXT *make_string(TEXT *, USHORT);
static SYN negate(SYN);
static KWWORDS next_keyword(void);
static SYN parse_abort(void);
static SYN parse_accept(void);
static SYN parse_add(USHORT *, USHORT *);
static SYN parse_and(USHORT *);
static SYN parse_assignment(void);
static SYN parse_boolean(USHORT *);
static SYN parse_copy(void);
static DBB parse_database(void);
static SYN parse_declare(void);
static SYN parse_define(void);
static SYN parse_def_index(void);
static SYN parse_def_relation(void);
static SYN parse_delete(void);
static SYN parse_drop(void);
static int parse_dtype(USHORT *, USHORT *);
static int parse_dtype_subtype(void);
static SYN parse_edit(void);
static TEXT *parse_edit_string(void);
static SYN parse_erase(void);
static SYN parse_extract(void);
static QLI_FLD parse_field(int);
static SYN parse_field_name(SYN *);
static SYN parse_for(void);
#ifdef PYXIS
static SYN parse_form(void);
#endif
static SYN parse_from(USHORT *, USHORT *);
static SYN parse_function(void);
static TEXT *parse_header(void);
static SYN parse_help(void);
static SYN parse_if(void);
static SYN parse_in(SYN, NOD_T, USHORT);
static SYN parse_insert(void);
static NOD_T parse_join_type(void);
static SYN parse_list_fields(void);
static CON parse_literal(void);
static SYN parse_matches(void);
static void parse_matching_paren(void);
static SYN parse_menu(void);
static SYN parse_modify(void);
static SYN parse_modify_index(void);
static SYN parse_modify_relation(void);
static SYN parse_multiply(USHORT *, USHORT *);
static NAM parse_name(void);
static SYN parse_not(USHORT *);
static int parse_ordinal(void);
static SYN parse_output(void);
static SYN parse_primitive_value(USHORT *, USHORT *);
static SYN parse_print_list(void);
static SYN parse_print(void);
static SYN parse_prompt(void);
static QFL parse_qualified_filter(void);
static QFN parse_qualified_function(void);
static QPR parse_qualified_procedure(void);
static QLI_REL parse_qualified_relation(void);
static SYN parse_ready(NOD_T);
static SYN parse_relational(USHORT *);
static SYN parse_relation(void);
static SYN parse_rename(void);
static SYN parse_repeat(void);
static SYN parse_report(void);
static SYN parse_rse(void);
static SYN parse_select(void);
static SYN parse_set(void);
static SYN parse_shell(void);
static SYN parse_show(void);
static SYN parse_sort(void);
static SYN parse_sql_alter(void);
static SYN parse_sql_create(void);
static int parse_sql_dtype(USHORT *, USHORT *);
static QLI_FLD parse_sql_field(void);
static SYN parse_sql_grant_revoke(USHORT);
static SYN parse_sql_index_create(USHORT, USHORT);
static SYN parse_sql_joined_relation(SYN);
static SYN parse_sql_join_clause(SYN);
static SYN parse_sql_table_create(void);
#ifdef NOT_USED_OR_REPLACED
static SYN parse_sql_view_create(void);
#endif
static SYN parse_sql_relation(void);
static SYN parse_sql_rse(void);
static SYN parse_sql_singleton_select(void);
static SYN parse_sql_subquery(void);
static SYN parse_statement(void);
static SYN parse_statistical(void);
static SYN parse_store(void);
static TEXT *parse_string(void);
static SYM parse_symbol(void);
static void parse_terminating_parens(USHORT *, USHORT *);
static SYN parse_transaction(NOD_T);
static SYN parse_udf_or_field(void);
static SYN parse_update(void);
static SYN parse_value(USHORT *, USHORT *);
static int potential_rse(void);
static QLI_REL resolve_relation(SYM, SYM);
static SYN syntax_node(NOD_T, USHORT);
static int test_end(void);
typedef struct {
SLONG gds_quad_high;
ULONG gds_quad_low;
} gds_quad;
/*
The following flags are:
sql_flag indicates whether we are parsing in a SQL environment.
The flag is used to turn off automatic end-of-command
recognition.
else_count indicates the depth of if/then/else's
sw_report indicates whether we're in a report statement
sw_statement indicates that we're actively parsing a statement/command
sw_sql_view indicates parsing a SQL VIEW, so restrict the select.
*/
static USHORT sql_flag, else_count, sw_report, sw_statement, sw_sql_view;
static SSHORT function_count; /* indicates the depth of UDF calls */
static struct nod_types {
KWWORDS nod_t_keyword;
NOD_T nod_t_node;
NOD_T nod_t_rpt_node;
NOD_T nod_t_sql_node;
} statisticals[] = {
{ KW_MAX, nod_max, nod_rpt_max, nod_agg_max },
{ KW_MIN, nod_min, nod_rpt_min, nod_agg_min },
{ KW_COUNT, nod_count, nod_rpt_count, nod_agg_count },
{ KW_AVERAGE, nod_average, nod_rpt_average, nod_agg_average },
{ KW_TOTAL, nod_total, nod_rpt_total, nod_agg_total },
{ KW_none, nod_any, nod_any, nod_any }
};
static NOD_T relationals[] = {
nod_eql, nod_neq, nod_leq, nod_lss, nod_gtr, nod_geq, nod_containing,
nod_like, nod_starts, nod_missing, nod_between, nod_sleuth,
nod_matches, nod_and, nod_or, nod_not, nod_any, nod_unique, (NOD_T) 0
};
SYN PARQ_parse(void)
{
/**************************************
*
* P A R Q _ p a r s e
*
**************************************
*
* Functional description
* Parse a single statement or command.
*
**************************************/
LLS stack;
SYN node;
sql_flag = else_count = sw_report = 0;
sw_statement = TRUE;
switch (next_keyword()) {
case KW_COMMIT:
return parse_transaction(nod_commit);
case KW_COPY:
return parse_copy();
case KW_EXIT:
return SYNTAX_NODE(nod_exit, 0);
case KW_EXTRACT:
return parse_extract();
case KW_QUIT:
return SYNTAX_NODE(nod_quit, 0);
case KW_DELETE:
case KW_DROP:
if (node = parse_drop())
return node;
node = parse_delete();
check_end();
if (!MATCH(KW_THEN))
return node;
stack = NULL;
LLS_PUSH(node, &stack);
LLS_PUSH(parse_statement(), &stack);
return make_list(stack);
case KW_DEFINE:
return parse_define();
case KW_CREATE:
return parse_sql_create();
case KW_ALTER:
return parse_sql_alter();
case KW_EDIT:
return parse_edit();
case KW_FINISH:
return parse_transaction(nod_finish);
case KW_GRANT:
return parse_sql_grant_revoke(nod_sql_grant);
case KW_HELP:
return parse_help();
case KW_PREPARE:
return parse_transaction(nod_prepare);
case KW_READY:
return parse_ready(nod_ready);
case KW_RENAME:
return parse_rename();
case KW_REVOKE:
return parse_sql_grant_revoke(nod_sql_revoke);
case KW_ROLLBACK:
return parse_transaction(nod_rollback);
case KW_SET:
return parse_set();
case KW_SHELL:
return parse_shell();
case KW_SHOW:
return parse_show();
}
return parse_statement();
}
int PAR_match( KWWORDS keyword)
{
/**************************************
*
* P A R _ m a t c h
*
**************************************
*
* Functional description
* Test the current token for a particular keyword.
* If so, advance the token stream.
*
**************************************/
SYM symbol;
if (KEYWORD(keyword)) {
PAR_token();
return TRUE;
}
for (symbol = QLI_token->tok_symbol; symbol; symbol = symbol->sym_homonym)
if (symbol->sym_type == SYM_keyword &&
symbol->sym_keyword == (int) keyword) {
PAR_token();
return TRUE;
}
return FALSE;
}
void PAR_real(void)
{
/**************************************
*
* P A R _ r e a l
*
**************************************
*
* Functional description
* Get a real (not EOL) token.
*
* If the token is an end of line, get the next token.
* If the next token is a colon, start reading the
* procedure.
*
**************************************/
DBB database;
while ((QLI_token->tok_type == tok_eol) || KEYWORD(KW_continuation))
LEX_token();
if (MATCH(KW_COLON)) {
database = parse_database();
PRO_invoke(database, QLI_token->tok_string);
}
}
void PAR_real_token(void)
{
/**************************************
*
* P A R _ r e a l _ t o k e n
*
**************************************
*
* Functional description
* Composition of PAR_token followed by PAR_real.
*
**************************************/
PAR_token();
PAR_real();
}
void PAR_token(void)
{
/**************************************
*
* P A R _ t o k e n
*
**************************************
*
* Functional description
* Get the next token.
* If it's a procedure invocation, handle it
* or complain and get rid of the evidence.
*
**************************************/
DBB database;
for (;;) {
LEX_token();
if (!(KEYWORD(KW_continuation)) &&
!(sw_statement && QLI_semi && QLI_token->tok_type == tok_eol))
break;
}
if (MATCH(KW_COLON)) {
if (!QLI_databases) {
ERRQ_error_format(159, NULL, NULL, NULL, NULL, NULL); /* Msg159 no databases are ready */
ERRQ_pending();
LEX_token();
}
else {
database = parse_database();
PRO_invoke(database, QLI_token->tok_string);
}
}
}
static void check_end(void)
{
/**************************************
*
* c h e c k _ e n d
*
**************************************
*
* Functional description
* Check for end of statement. If it isn't complain bitterly.
*
**************************************/
if (QLI_token->tok_type == tok_eol ||
KEYWORD(KW_SEMI) ||
KEYWORD(KW_THEN) || (else_count && KEYWORD(KW_ELSE))) {
sw_statement = FALSE;
return;
}
SYNTAX_ERROR(161); /* Msg161 end of statement */
}
static void command_end(void)
{
/**************************************
*
* c o m m a n d _ e n d
*
**************************************
*
* Functional description
* Check for end of command. If it isn't complain bitterly.
*
**************************************/
if (QLI_token->tok_type == tok_eol || KEYWORD(KW_SEMI)) {
sw_statement = FALSE;
return;
}
SYNTAX_ERROR(162); /* Msg162 end of command */
}
static DBB get_dbb( SYM symbol)
{
/**************************************
*
* g e t _ d b b
*
**************************************
*
* Functional description
* Find a database block from a symbol
* or its homonyms.
*
**************************************/
for (; symbol; symbol = symbol->sym_homonym)
if (symbol->sym_type == SYM_database)
return (DBB) symbol->sym_object;
return NULL;
}
static SYN make_list( LLS stack)
{
/**************************************
*
* m a k e _ l i s t
*
**************************************
*
* Functional description
* Dump a stack of junk into a list node. Best count
* them first.
*
**************************************/
SYN node, *ptr;
LLS temp;
USHORT count;
temp = stack;
count = 0;
while (temp) {
count++;
temp = temp->lls_next;
}
node = SYNTAX_NODE(nod_list, count);
ptr = &node->syn_arg[count];
while (stack)
*--ptr = (SYN) LLS_POP(&stack);
return node;
}
static NAM make_name(void)
{
/**************************************
*
* m a k e _ n a m e
*
**************************************
*
* Functional description
* Generate a unique name for something
* (like a database) that needs one.
*
**************************************/
NAM name;
SSHORT l, i;
TEXT c, *p, *q, string[32];
for (i = 0; i < 1000; i++) {
sprintf(string, "QLI_%d", i);
if (i < 10)
l = 5;
else
l = (i < 100) ? 6 : 7;
if (!(HSH_lookup(string, l)))
break;
}
name = (NAM) ALLOCDV(type_nam, l);
name->nam_length = l;
p = name->nam_string;
q = string;
if (l)
do {
c = *q++;
*p++ = UPPER(c);
} while (--l);
return name;
}
static CON make_numeric_constant( TEXT * string, USHORT length)
{
/**************************************
*
* m a k e _ n u m e r i c _ c o n s t a n t
*
**************************************
*
* Functional description
* Build a constant block for a numeric
* constant. Numeric constants are normally
* stored as long words, but especially large
* ones become text. They ought to become
* double precision, one would think, but they'd
* have to be VAX style double precision which is
* more pain than gain.
*
**************************************/
CON constant;
TEXT *p;
USHORT l;
p = string;
l = length;
/* If there are a reasonable number of digits, convert to binary */
if (length < 9) {
constant = (CON) ALLOCDV(type_con, sizeof(SLONG));
constant->con_desc.dsc_dtype = dtype_long;
constant->con_desc.dsc_length = sizeof(SLONG);
constant->con_desc.dsc_address = constant->con_data;
constant->con_desc.dsc_scale =
MOVQ_decompose(string, length, (SLONG*) constant->con_data);
}
else {
++length;
constant = (CON) ALLOCDV(type_con, length);
constant->con_desc.dsc_dtype = dtype_text;
constant->con_desc.dsc_length = length;
constant->con_desc.dsc_address = constant->con_data;
p = (TEXT *) constant->con_desc.dsc_address;
*p++ = '0';
while (--length)
*p++ = *string++;
}
return constant;
}
static TEXT *make_string( TEXT * address, USHORT length)
{
/**************************************
*
* m a k e _ s t r i n g
*
**************************************
*
* Functional description
* Copy a string into a temporary string block.
*
**************************************/
STR string;
TEXT *p;
string = (STR) ALLOCDV(type_str, length);
p = string->str_data;
if (length)
do
*p++ = *address++;
while (--length);
return string->str_data;
}
static SYN negate( SYN expr)
{
/**************************************
*
* n e g a t e
*
**************************************
*
* Functional description
* Build negation of expression.
*
**************************************/
SYN node;
node = SYNTAX_NODE(nod_not, 1);
node->syn_arg[0] = expr;
return node;
}
static KWWORDS next_keyword(void)
{
/**************************************
*
* n e x t _ k e y w o r d
*
**************************************
*
* Functional description
* Get a real token and return the keyword number.
*
**************************************/
SYM symbol;
PAR_real();
for (symbol = QLI_token->tok_symbol; symbol; symbol = symbol->sym_homonym)
if (symbol->sym_type == SYM_keyword)
return (KWWORDS) symbol->sym_keyword;
return KW_none;
}
static SYN parse_abort(void)
{
/**************************************
*
* p a r s e _ a b o r t
*
**************************************
*
* Functional description
* Parse an ABORT statement.
*
**************************************/
SYN node;
PAR_token();
node = SYNTAX_NODE(nod_abort, 1);
if (KEYWORD(KW_SEMI))
node->syn_count = 0;
else
node->syn_arg[0] = parse_value(0, 0);
return node;
}
static SYN parse_accept(void)
{
/**************************************
*
* p a r s e _ a c c e p t
*
**************************************
*
* Functional description
* Parse form update statement.
*
**************************************/
IBERROR(484); /* FORMs not supported */
return 0;
}
static SYN parse_add( USHORT * paren_count, USHORT * bool_flag)
{
/**************************************
*
* p a r s e _ a d d
*
**************************************
*
* Functional description
* Parse the lowest precedence operatrs, ADD and SUBTRACT.
*
**************************************/
SYN node, arg;
NOD_T operatr;
node = parse_multiply(paren_count, bool_flag);
while (TRUE) {
if (MATCH(KW_PLUS))
operatr = nod_add;
else if (MATCH(KW_MINUS))
operatr = nod_subtract;
else
return node;
arg = node;
node = SYNTAX_NODE(operatr, 2);
node->syn_arg[0] = arg;
node->syn_arg[1] = parse_multiply(paren_count, bool_flag);
}
}
static SYN parse_and( USHORT * paren_count)
{
/**************************************
*
* p a r s e _ a n d
*
**************************************
*
* Functional description
* Parse an AND expression.
*
**************************************/
SYN expr, node;
expr = parse_not(paren_count);
/*
while (*paren_count && KEYWORD (KW_RIGHT_PAREN))
{
parse_matching_paren();
(*paren_count)--;
}
*/
if (!MATCH(KW_AND))
return expr;
node = SYNTAX_NODE(nod_and, 2);
node->syn_arg[0] = expr;
node->syn_arg[1] = parse_and(paren_count);
return node;
}
static SYN parse_assignment(void)
{
/**************************************
*
* p a r s e _ a s s i g n m e n t
*
**************************************
*
* Functional description
* Parse an assignment statement (or give an error). The
* assignment statement can be either a simple assignment
* (field = value) or a restructure (relation = rse).
* If the assignment operatr is missing,
* generate an "expected statement" error.
*
**************************************/
SYN node, field;
QLI_REL relation;
NAM name, name2;
node = SYNTAX_NODE(nod_assign, s_asn_count);
node->syn_arg[s_asn_to] = parse_field_name(&field);
name = (NAM) field->syn_arg[0];
/* If the next token is an equals sign, the statement really is an
assignment, and we're off the hook. */
if (!MATCH(KW_EQUALS))
ERRQ_print_error(156, name->nam_string, NULL, NULL, NULL, NULL); /* Msg156 expected statement, encountered %s */
/* See if the "field name" is really a relation reference. If so,
turn the assignment into a restructure. */
if (field->syn_count == 1)
relation = resolve_relation(0, name->nam_symbol);
else if (field->syn_count == 2 && name->nam_symbol) {
name2 = (NAM) field->syn_arg[1];
relation = resolve_relation(name->nam_symbol, name2->nam_symbol);
}
else
relation = NULL;
if (relation) {
ALL_release((FRB) field);
node->syn_type = nod_restructure;
node->syn_arg[s_asn_to] = field =
SYNTAX_NODE(nod_relation, s_rel_count);
field->syn_arg[s_rel_relation] = (SYN) relation;
node->syn_arg[s_asn_from] = parse_rse();
}
else
node->syn_arg[s_asn_from] = parse_value(0, 0);
return node;
}
static SYN parse_boolean( USHORT * paren_count)
{
/**************************************
*
* p a r s e _ b o o l e a n
*
**************************************
*
* Functional description
* Parse a general boolean expression. By precedence, handle an OR
* here.
*
**************************************/
SYN expr, node;
USHORT local_count;
if (!paren_count) {
local_count = 0;
paren_count = &local_count;
}
expr = parse_and(paren_count);
/*
while (*paren_count && KEYWORD (KW_RIGHT_PAREN))
{
parse_matching_paren();
(*paren_count)--;
}
*/
if (!MATCH(KW_OR)) {
parse_terminating_parens(paren_count, &local_count);
return expr;
}
node = SYNTAX_NODE(nod_or, 2);
node->syn_arg[0] = expr;
node->syn_arg[1] = parse_boolean(paren_count);
parse_terminating_parens(paren_count, &local_count);
return node;
}
static SYN parse_copy(void)
{
/**************************************
*
* p a r s e _ c o p y
*
**************************************
*
* Functional description
* Parse a copy command, which copies
* one procedure to another.
*
**************************************/
SYN node;
PAR_real_token();
if (MATCH(KW_PROCEDURE)) {
node = SYNTAX_NODE(nod_copy_proc, 2);
node->syn_arg[0] = (SYN) parse_qualified_procedure();
MATCH(KW_TO);
node->syn_arg[1] = (SYN) parse_qualified_procedure();
return node;
}
ERRQ_print_error(157, QLI_token->tok_string, NULL, NULL, NULL, NULL); /* Msg157 Expected PROCEDURE encountered %s */
return NULL;
}
static DBB parse_database(void)
{
/**************************************
*
* p a r s e _ d a t a b a s e
*
**************************************
*
* Functional description
* Pick up a database for a meta-data or
* procedure update. Return NULL if the
* token is not a database name.
*
**************************************/
DBB database;
SYM db_symbol;
if ((db_symbol = QLI_token->tok_symbol)
&& db_symbol->sym_type == SYM_database) {
database = (DBB) db_symbol->sym_object;
PAR_real_token();
if (!MATCH(KW_DOT))
SYNTAX_ERROR(158); /* Msg158 period in qualified name */
PAR_real();
return database;
}
if (!QLI_databases)
IBERROR(159); /* Msg159 no databases are ready */
return NULL;
}
static SYN parse_declare(void)
{
/**************************************
*
* p a r s e _ d e c l a r e
*
**************************************
*
* Functional description
* Parse a variable declaration.
*
**************************************/
SYN node, field_node;
SYM name, query_name;
QLI_FLD global_variable;
USHORT dtype, length, scale;
SSHORT sub_type, sub_type_missing;
TEXT *edit_string, *query_header;
QLI_REL relation;
NAM db_name, rel_name;
PAR_token();
PAR_real();
dtype = length = scale = 0;
sub_type = 0;
sub_type_missing = 1;
field_node = NULL;
relation = NULL;
query_name = NULL;
edit_string = query_header = NULL;
name = parse_symbol();
/*if (global_flag) PAR_real();*/
while (!KEYWORD(KW_SEMI) && !KEYWORD(KW_COMMA)) {
PAR_real();
switch (QLI_token->tok_keyword) {
case KW_SHORT:
case KW_LONG:
case KW_FLOAT:
case KW_DOUBLE:
case KW_DATE:
case KW_CHAR:
case KW_VARYING:
if (dtype)
SYNTAX_ERROR(164); /* Msg164 variable definition clause */
dtype = parse_dtype(&length, &scale);
break;
case KW_BLOB:
IBERROR(160); /* Msg160 blob variables are not supported */
break;
case KW_SUB_TYPE:
sub_type = parse_dtype_subtype();
sub_type_missing = 0;
break;
case KW_EDIT_STRING:
PAR_token();
if (QLI_token->tok_type != tok_quoted)
SYNTAX_ERROR(163); /* Msg163 quoted edit string */
edit_string =
make_string(QLI_token->tok_string + 1,
QLI_token->tok_length - 2);
PAR_token();
break;
case KW_QUERY_NAME:
PAR_token();
MATCH(KW_IS);
if (QLI_token->tok_type != tok_ident)
SYNTAX_ERROR(199); /* Msg199 identifier */
query_name = parse_symbol();
break;
case KW_QUERY_HEADER:
PAR_token();
query_header = parse_header();
break;
case KW_BASED:
PAR_token();
MATCH(KW_ON);
field_node = parse_field_name(0);
break;
default:
SYNTAX_ERROR(164); /* Msg164 variable definition clause */
break;
}
}
if (field_node && field_node->syn_count == 3) {
db_name = (NAM) field_node->syn_arg[0];
rel_name = (NAM) field_node->syn_arg[1];
if (!db_name->nam_symbol)
ERRQ_print_error(165, db_name->nam_string, NULL, NULL, NULL, NULL); /* Msg165 %s is not a database */
if (!
(relation =
resolve_relation(db_name->nam_symbol,
rel_name->nam_symbol))) ERRQ_print_error(166,
rel_name->
nam_string,
db_name->
nam_string, NULL, NULL, NULL); /* Msg166 %s is not a relation in database %s */
}
if (!dtype && !field_node)
SYNTAX_ERROR(167); /* Msg167 variable data type */
if (field_node && (dtype || length || scale))
IBERROR(168); /* Msg168 no datatype may be specified for a variable based on a field */
node = SYNTAX_NODE(nod_declare, 2);
global_variable = (QLI_FLD) ALLOCDV(type_fld, length);
node->syn_arg[0] = (SYN) global_variable;
global_variable->fld_name = name;
global_variable->fld_dtype = dtype;
global_variable->fld_scale = scale;
global_variable->fld_sub_type = sub_type;
global_variable->fld_sub_type_missing = sub_type_missing;
global_variable->fld_length = length;
global_variable->fld_edit_string = edit_string;
global_variable->fld_query_name = query_name;
global_variable->fld_query_header = query_header;
global_variable->fld_relation = relation;
node->syn_arg[1] = field_node;
check_end();
return node;
}
static SYN parse_define(void)
{
/**************************************
*
* p a r s e _ d e f i n e
*
**************************************
*
* Functional description
* Parse a DEFINE command.
* There are, of course, a whole class of define commands.
*
**************************************/
SYN node;
PAR_real_token();
if (MATCH(KW_PROCEDURE)) {
PAR_real();
node = SYNTAX_NODE(nod_define, 1);
node->syn_arg[0] = (SYN) parse_qualified_procedure();
return node;
}
if (MATCH(KW_FIELD)) {
PAR_real();
node = SYNTAX_NODE(nod_def_field, 2);
node->syn_arg[0] = (SYN) parse_database();
node->syn_arg[1] = (SYN) parse_field(TRUE);
return node;
}
if (MATCH(KW_RELATION))
return parse_def_relation();
if (KEYWORD(KW_DATABASE))
return parse_ready(nod_def_database);
if (MATCH(KW_INDEX))
return parse_def_index();
SYNTAX_ERROR(169); /* Msg169 object type for DEFINE */
return NULL;
}
static SYN parse_def_index(void)
{
/**************************************
*
* p a r s e _ d e f _ i n d e x
*
**************************************
*
* Functional description
* Parse a DEFINE INDEX command.
*
**************************************/
LLS stack;
SYN node;
PAR_real();
node = SYNTAX_NODE(nod_def_index, s_dfi_count);
node->syn_arg[s_dfi_name] = (SYN) parse_symbol();
PAR_real();
MATCH(KW_FOR);
if (!(node->syn_arg[s_dfi_relation] = (SYN) parse_qualified_relation()))
SYNTAX_ERROR(170); /* Msg170 relation name */
PAR_real();
while (TRUE) {
PAR_real();
if (MATCH(KW_UNIQUE))
node->syn_flags |= s_dfi_flag_unique;
else if (MATCH(KW_DUPLICATE))
node->syn_flags &= ~s_dfi_flag_unique;
else if (MATCH(KW_ACTIVE))
node->syn_flags &= ~s_dfi_flag_inactive;
else if (MATCH(KW_INACTIVE))
node->syn_flags |= s_dfi_flag_inactive;
else if (MATCH(KW_DESCENDING))
node->syn_flags |= s_dfi_flag_descending;
else if (MATCH(KW_ASCENDING))
node->syn_flags &= ~s_dfi_flag_descending;
else
break;
}
stack = NULL;
for (;;) {
LLS_PUSH(parse_name(), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_dfi_fields] = make_list(stack);
command_end();
return node;
}
static SYN parse_def_relation(void)
{
/**************************************
*
* p a r s e _ d e f _ r e l a t i o n
*
**************************************
*
* Functional description
* Parse a DEFINE RELATION command,
* which include the field definitions
* for a primitive relation definition
* or it may just reference another relatio
* whose field definitions we will copy.
*
**************************************/
QLI_REL relation;
QLI_FLD field, *ptr;
SYN node;
PAR_real();
node = SYNTAX_NODE(nod_def_relation, 2);
relation = (QLI_REL) ALLOCD(type_rel);
node->syn_arg[0] = (SYN) relation;
relation->rel_database = parse_database();
relation->rel_symbol = parse_symbol();
PAR_real();
if (MATCH(KW_BASED)) {
PAR_real();
MATCH(KW_ON);
PAR_real();
MATCH(KW_RELATION);
PAR_real();
relation = (QLI_REL) ALLOCD(type_rel);
node->syn_arg[1] = (SYN) relation;
relation->rel_database = parse_database();
relation->rel_symbol = parse_symbol();
}
else {
node->syn_arg[1] = NULL;
ptr = &relation->rel_fields;
for (;;) {
MATCH(KW_ADD);
PAR_real();
MATCH(KW_FIELD);
*ptr = field = parse_field(FALSE);
ptr = &field->fld_next;
if (KEYWORD(KW_SEMI))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(171); /* Msg171 comma between field definitions */
}
}
command_end();
return node;
}
static SYN parse_delete(void)
{
/**************************************
*
* p a r s e _ d e l e t e
*
**************************************
*
* Functional description
* Parse a SQL DELETE statement.
* (DELETE PROCEDURE is parsed in parse_drop)
*
**************************************/
SYN node, rse;
++sql_flag;
if (!MATCH(KW_FROM))
SYNTAX_ERROR(172); /* Msg172 FROM */
node = SYNTAX_NODE(nod_erase, s_era_count);
node->syn_arg[s_era_rse] = rse =
SYNTAX_NODE(nod_rse, (int) s_rse_count + 2);
rse->syn_count = 1;
rse->syn_arg[s_rse_count] = parse_sql_relation();
/* Pick up boolean, if present */
if (MATCH(KW_WITH))
rse->syn_arg[s_rse_boolean] = parse_boolean(0);
--sql_flag;
return node;
}
static SYN parse_drop(void)
{
/**************************************
*
* p a r s e _ d r o p
*
**************************************
*
* Functional description
* Parse a DDL DROP/DELETE command. It it isn't one,
* just return NULL.
*
**************************************/
SYN node;
NOD_T type;
DBB database;
SSHORT l;
TEXT *p, *q;
PAR_real_token();
if (MATCH(KW_RELATION) || MATCH(KW_VIEW) || MATCH(KW_TABLE)) {
node = SYNTAX_NODE(nod_del_relation, 1);
if (!(node->syn_arg[0] = (SYN) parse_qualified_relation()))
SYNTAX_ERROR(173); /* Msg173 relation or view name */
return node;
}
switch (QLI_token->tok_keyword) {
case KW_PROCEDURE:
type = nod_delete_proc;
break;
case KW_INDEX:
type = nod_del_index;
break;
case KW_FIELD:
type = nod_del_field;
break;
case KW_DATABASE:
LEX_filename();
if (!(l = QLI_token->tok_length))
ERRQ_error(429, NULL, NULL, NULL, NULL, NULL); /* Msg429 database file name required on DROP DATABASE */
q = QLI_token->tok_string;
if (QLI_token->tok_type == tok_quoted) {
l -= 2;
q++;
}
database = (DBB) ALLOCDV(type_dbb, l);
database->dbb_filename_length = l;
p = database->dbb_filename;
do
*p++ = *q++;
while (--l);
PAR_token();
/* parse an optional user name and password if given */
for (;;) {
if (MATCH(KW_USER))
database->dbb_user = parse_literal();
else if (MATCH(KW_PASSWORD))
database->dbb_password = parse_literal();
else
break;
}
command_end();
node = SYNTAX_NODE(nod_del_database, 1);
node->syn_arg[0] = (SYN) database;
return node;
default:
return NULL;
}
PAR_real_token();
node = SYNTAX_NODE(type, 2);
if (type == nod_delete_proc)
node->syn_arg[0] = (SYN) parse_qualified_procedure();
else {
node->syn_arg[0] = (SYN) parse_database();
node->syn_arg[1] = (SYN) parse_name();
}
return node;
}
static int parse_dtype( USHORT * length, USHORT * scale)
{
/**************************************
*
* p a r s e _ d t y p e
*
**************************************
*
* Functional description
* Parse a datatype clause.
*
**************************************/
KWWORDS keyword;
USHORT dtype, l;
keyword = QLI_token->tok_keyword;
PAR_token();
*scale = 0;
switch (keyword) {
case KW_SHORT:
*length = sizeof(SSHORT);
dtype = dtype_short;
break;
case KW_LONG:
*length = sizeof(SLONG);
dtype = dtype_long;
break;
case KW_FLOAT:
*length = sizeof(float);
return dtype_real;
case KW_DOUBLE:
*length = sizeof(double);
return dtype_double;
case KW_DATE:
*length = sizeof(gds_quad);
return dtype_timestamp;
case KW_CHAR:
dtype = dtype_text;
break;
case KW_VARYING:
dtype = dtype_varying;
break;
case KW_BLOB:
*length = sizeof(gds_quad);
return dtype_blob;
}
if (dtype == dtype_short || dtype == dtype_long) {
if (MATCH(KW_SCALE)) {
l = (MATCH(KW_MINUS)) ? TRUE : FALSE;
*scale = parse_ordinal();
if (l)
*scale = -(*scale);
}
}
else if (dtype == dtype_text || dtype == dtype_varying) {
if (!MATCH(KW_L_BRCKET) && !MATCH(KW_LT))
SYNTAX_ERROR(174); /* Msg174 "[" */
l = parse_ordinal();
if (dtype == dtype_varying)
l += sizeof(SSHORT);
*length = l;
if (!MATCH(KW_R_BRCKET) && !MATCH(KW_GT))
SYNTAX_ERROR(175); /* Msg175 "]" */
}
return dtype;
}
static int parse_dtype_subtype(void)
{
/**************************************
*
* p a r s e _ d t y p e _ s u b t y p e
*
**************************************
*
* Functional description
* Parse a sub-type definition, which can be any of
* SUB_TYPE {IS} [TEXT | FIXED | <n>]
*
* Returns the numeric subtype value,
*
**************************************/
int sign;
/* grab KW_SUB_TYPE */
PAR_token();
MATCH(KW_IS);
if (MATCH(KW_TEXT) || MATCH(KW_FIXED))
return 1;
sign = (MATCH(KW_MINUS)) ? -1 : 1;
return (sign * parse_ordinal());
}
static SYN parse_edit(void)
{
/**************************************
*
* p a r s e _ e d i t
*
**************************************
*
* Functional description
* Parse an edit statement which can
* be any of EDIT <procedure_name>
* EDIT <n>
* EDIT <*>
* EDIT
*
*
**************************************/
SYN node;
NOD_T type;
int l;
LLS start, stop, statement_list;
LEX_token();
/*
* edit previous statements. The top of the statment list
* is this edit command, which we conveniently ignore.
*/
if (KEYWORD(KW_SEMI) ||
(QLI_token->tok_type == tok_number) || (KEYWORD(KW_ASTERISK))) {
if (!(statement_list = LEX_statement_list()))
IBERROR(176); /* Msg176 No statements issued yet */
if (MATCH(KW_ASTERISK))
LEX_edit((SLONG) 0, (SLONG) statement_list->lls_object);
else {
if (KEYWORD(KW_SEMI))
l = 1;
else if (QLI_token->tok_type == tok_number)
l = parse_ordinal();
for (start = stop = statement_list;
l && start->lls_next; l--, start = start->lls_next);
command_end();
LEX_edit((SLONG) start->lls_object, (SLONG) stop->lls_object);
}
}
#ifdef PYXIS
else if (MATCH(KW_FORM))
IBERROR(484); /* FORMs not supported */
#endif
else {
type = nod_edit_proc;
node = SYNTAX_NODE(type, 2);
node->syn_arg[0] = (SYN) parse_qualified_procedure();
command_end();
return node;
}
return NULL;
}
static TEXT *parse_edit_string(void)
{
/**************************************
*
* p a r s e _ e d i t _ s t r i n g
*
**************************************
*
* Functional description
* Look for and parse a clause of the form:
* USING <edit_string>
*
**************************************/
if (!KEYWORD(KW_USING))
return NULL;
LEX_edit_string();
return parse_string();
}
static SYN parse_erase(void)
{
/**************************************
*
* p a r s e _ e r a s e
*
**************************************
*
* Functional description
* Parse an ERASE statement. Erase can be any of the
* following:
*
* ERASE [ALL] [OF <rse>]
*
**************************************/
SYN node;
PAR_token();
MATCH(KW_ALL);
MATCH(KW_OF);
node = SYNTAX_NODE(nod_erase, s_era_count);
if (MATCH(KW_ALL) || potential_rse()) {
MATCH(KW_OF);
node->syn_arg[s_era_rse] = parse_rse();
}
return node;
}
static SYN parse_extract(void)
{
/**************************************
*
* p a r s e _ e x t r a c t
*
**************************************
*
* Functional description
* Parse a procedure extract statement. Syntax is:
*
* EXTRACT [ON <file>] proc [, ...] [ON <file> ]
*
**************************************/
LLS stack;
SYN node;
PAR_real_token();
node = SYNTAX_NODE(nod_extract, 2);
node->syn_arg[1] = parse_output();
if (!MATCH(KW_ALL)) {
stack = NULL;
for (;;) {
LLS_PUSH(parse_qualified_procedure(), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[0] = make_list(stack);
}
if (!node->syn_arg[1] && !(node->syn_arg[1] = parse_output()))
SYNTAX_ERROR(177); /* Msg177 "ON or TO" */
return node;
}
static QLI_FLD parse_field( int global_flag)
{
/**************************************
*
* p a r s e _ f i e l d
*
**************************************
*
* Functional description
* Parse a field description.
*
**************************************/
QLI_FLD field;
SYM name, query_name, based_on;
USHORT dtype, length, scale;
SSHORT sub_type, sub_type_missing;
TEXT *edit_string, *query_header;
PAR_real();
dtype = length = scale = 0;
sub_type = 0;
sub_type_missing = 1;
query_name = based_on = NULL;
edit_string = query_header = NULL;
name = parse_symbol();
if (global_flag)
PAR_real();
while (!KEYWORD(KW_SEMI) && !KEYWORD(KW_COMMA)) {
PAR_real();
switch (QLI_token->tok_keyword) {
case KW_SHORT:
case KW_LONG:
case KW_FLOAT:
case KW_DOUBLE:
case KW_DATE:
case KW_CHAR:
case KW_VARYING:
case KW_BLOB:
if (dtype)
SYNTAX_ERROR(179); /* Msg179 field definition clause */
dtype = parse_dtype(&length, &scale);
break;
case KW_SUB_TYPE:
sub_type = parse_dtype_subtype();
sub_type_missing = 0;
break;
case KW_EDIT_STRING:
PAR_token();
if (QLI_token->tok_type != tok_quoted)
SYNTAX_ERROR(178); /* Msg178 quoted edit string */
edit_string =
make_string(QLI_token->tok_string + 1,
QLI_token->tok_length - 2);
PAR_token();
break;
case KW_QUERY_NAME:
PAR_token();
MATCH(KW_IS);
if (QLI_token->tok_type != tok_ident)
SYNTAX_ERROR(199); /* Msg199 identifier */
query_name = parse_symbol();
break;
case KW_BASED:
PAR_token();
MATCH(KW_ON);
based_on = parse_symbol();
break;
default:
SYNTAX_ERROR(179); /* Msg179 field definition clause */
break;
}
}
field = (QLI_FLD) ALLOCDV(type_fld, length);
field->fld_name = name;
field->fld_dtype = dtype;
field->fld_scale = scale;
field->fld_sub_type = sub_type;
field->fld_sub_type_missing = sub_type_missing;
field->fld_length = length;
field->fld_edit_string = edit_string;
field->fld_query_name = query_name;
field->fld_query_header = query_header;
if (!global_flag)
field->fld_based = based_on;
else if (based_on)
IBERROR(180); /* Msg180 global fields may not be based on other fields */
return field;
}
static SYN parse_field_name( SYN * fld_ptr)
{
/**************************************
*
* p a r s e _ f i e l d _ n a m e
*
**************************************
*
* Functional description
* Parse a qualified field name, or
* qualified * expression.
*
**************************************/
SYN field, node;
LLS stack;
stack = NULL;
while (TRUE) {
if (MATCH(KW_ASTERISK)) {
if (!stack)
SYNTAX_ERROR(181); /* Msg181 field name or asterisk expression */
field = make_list(stack);
field->syn_type = nod_star;
return field;
}
LLS_PUSH(parse_name(), &stack);
if (!MATCH(KW_DOT))
break;
}
field = make_list(stack);
field->syn_type = nod_field;
if (fld_ptr)
*fld_ptr = field;
if (!(MATCH(KW_L_BRCKET)))
return field;
/* Parse an array reference */
stack = NULL;
for (;;) {
LLS_PUSH(parse_value(0, 0), &stack);
if (MATCH(KW_R_BRCKET))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(183); /* Msg183 comma */
}
node = SYNTAX_NODE(nod_index, s_idx_count);
node->syn_arg[s_idx_field] = field;
node->syn_arg[s_idx_subs] = make_list(stack);
return node;
}
static SYN parse_for(void)
{
/**************************************
*
* p a r s e _ f o r
*
**************************************
*
* Functional description
* Parse a FOR statement.
*
**************************************/
SYN node;
PAR_token();
#ifdef PYXIS
if (MATCH(KW_FORM))
IBERROR(484); /* FORMs not supported */
#endif
node = SYNTAX_NODE(nod_for, s_for_count);
node->syn_arg[s_for_rse] = parse_rse();
node->syn_arg[s_for_statement] = parse_statement();
return node;
}
#ifdef PYXIS
static SYN parse_form(void)
{
/**************************************
*
* p a r s e _ f o r m
*
**************************************
*
* Functional description
* Parse qualified form name, returning a form block.
*
**************************************/
IBERROR(484); /* FORMs not supported */
}
#endif
static SYN parse_from( USHORT * paren_count, USHORT * bool_flag)
{
/**************************************
*
* p a r s e _ f r o m
*
**************************************
*
* Functional description
* Parse either an explicit or implicit FIRST ... FROM statement.
*
**************************************/
SYN node, value;
PAR_real();
if (MATCH(KW_FIRST)) {
value = parse_primitive_value(0, 0);
PAR_real();
if (!MATCH(KW_FROM))
SYNTAX_ERROR(182); /* Msg182 FROM rse clause */
}
else {
value = parse_primitive_value(paren_count, bool_flag);
if (sql_flag || !MATCH(KW_FROM))
return value;
}
node = SYNTAX_NODE(nod_from, s_stt_count);
node->syn_arg[s_stt_value] = value;
node->syn_arg[s_stt_rse] = parse_rse();
if (MATCH(KW_ELSE))
node->syn_arg[s_stt_default] = parse_value(0, 0);
return node;
}
static SYN parse_function(void)
{
/**************************************
*
* p a r s e _ f u n c t i o n
*
**************************************
*
* Functional description
* Parse a function reference.
*
**************************************/
SYN node;
LLS stack;
function_count++;
node = SYNTAX_NODE(nod_function, s_fun_count);
node->syn_arg[s_fun_function] = (SYN) QLI_token->tok_symbol;
node->syn_count = 1;
PAR_token();
stack = NULL;
if (MATCH(KW_LEFT_PAREN))
for (;;) {
LLS_PUSH(parse_value(0, 0), &stack);
if (MATCH(KW_RIGHT_PAREN))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(183); /* Msg183 comma */
}
node->syn_arg[s_fun_args] = make_list(stack);
function_count--;
return node;
}
static TEXT *parse_header(void)
{
/**************************************
*
* p a r s e _ h e a d e r
*
**************************************
*
* Functional description
* Parse and store headers of the form:
* "quoted_string" [/ "more_string"]...
* or even the non-header -
*
**************************************/
TEXT *p, *q, header[1024];
p = header;
while (TRUE) {
PAR_real();
if ((QLI_token->tok_keyword != KW_MINUS) &&
(QLI_token->tok_type != tok_quoted)) SYNTAX_ERROR(184); /* Msg184 quoted header segment */
q = QLI_token->tok_string;
while (*q)
*p++ = *q++;
PAR_real_token();
if (!MATCH(KW_SLASH))
break;
}
return make_string(header, p - header);
}
static SYN parse_help(void)
{
/**************************************
*
* p a r s e _ h e l p
*
**************************************
*
* Functional description
* Parse HELP statement. Unreasonable, but the masses
* must be appeased. Bread, circuses, help.
*
**************************************/
SYN node;
LLS stack;
stack = NULL;
PAR_token();
while (!KEYWORD(KW_SEMI)) {
LLS_PUSH(parse_name(), &stack);
MATCH(KW_COMMA);
}
node = make_list(stack);
node->syn_type = nod_help;
command_end();
return node;
}
static SYN parse_if(void)
{
/**************************************
*
* p a r s e _ i f
*
**************************************
*
* Functional description
* Parse an IF THEN ELSE statement.
*
**************************************/
SYN node;
PAR_token();
node = SYNTAX_NODE(nod_if, s_if_count);
node->syn_arg[s_if_boolean] = parse_boolean(0);
PAR_real();
MATCH(KW_THEN);
++else_count;
node->syn_arg[s_if_true] = parse_statement();
--else_count;
if (MATCH(KW_ELSE))
node->syn_arg[s_if_false] = parse_statement();
return node;
}
static SYN parse_in( SYN value, NOD_T operatr, USHORT all_flag)
{
/**************************************
*
* p a r s e _ i n
*
**************************************
*
* Functional description
* Parse a SQL "IN" clause. This can have two forms:
*
* value IN (exp1, exp2...)
*
* value IN (column <select expression>)
*
**************************************/
SYN node, node2, value2, rse;
PAR_real();
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(185); /* Msg185 left parenthesis */
/* Time to chose between two forms of the expression */
if (!MATCH(KW_SELECT)) {
node = SYNTAX_NODE(operatr, 2);
node->syn_arg[0] = value;
node->syn_arg[1] = parse_primitive_value(0, 0);
while (MATCH(KW_COMMA)) {
node2 = node;
node = SYNTAX_NODE(nod_or, 2);
node->syn_arg[0] = node2;
node->syn_arg[1] = node2 = SYNTAX_NODE(nod_eql, 2);
node2->syn_arg[0] = value;
node2->syn_arg[1] = parse_value(0, 0);
}
parse_matching_paren();
return node;
}
value2 = parse_value(0, 0);
/* We have the "hard" -- an implicit ANY */
rse = parse_sql_rse();
parse_matching_paren();
rse->syn_arg[s_rse_outer] = value;
rse->syn_arg[s_rse_inner] = value2;
rse->syn_arg[s_rse_op] = INT_CAST operatr;
rse->syn_arg[s_rse_all_flag] = INT_CAST all_flag;
/* Finally, construct an ANY node */
node = SYNTAX_NODE(nod_any, 1);
node->syn_arg[0] = rse;
return (all_flag) ? negate(node) : node;
}
static SYN parse_insert(void)
{
/**************************************
*
* p a r s e _ i n s e r t
*
**************************************
*
* Functional description
* Parse a STORE statement.
*
**************************************/
SYN node;
USHORT select_flag;
LLS fields, values, distinct;
++sql_flag;
PAR_real_token();
MATCH(KW_INTO);
node = SYNTAX_NODE(nod_store, s_sto_count);
/* Pick up relation name for insert */
node->syn_arg[s_sto_relation] = parse_sql_relation();
/* Pick up field list, provided one is present */
distinct = fields = values = NULL;
PAR_real();
if (MATCH(KW_LEFT_PAREN))
while (TRUE) {
LLS_PUSH(parse_field_name(0), &fields);
if (MATCH(KW_RIGHT_PAREN))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(186); /* Msg186 comma or terminating right parenthesis */
}
/* Pick up value list or SELECT statement */
PAR_real();
if (MATCH(KW_VALUES)) {
select_flag = FALSE;
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(187); /* Msg187 left parenthesis */
}
else if (MATCH(KW_SELECT))
select_flag = TRUE;
else
SYNTAX_ERROR(188); /* Msg188 VALUES list or SELECT clause */
while (TRUE) {
if (distinct || MATCH(KW_DISTINCT)) {
LLS_PUSH(parse_value(0, 0), &distinct);
LLS_PUSH(distinct->lls_object, &values);
LLS_PUSH(0, &distinct);
}
else
LLS_PUSH(parse_value(0, 0), &values);
if (!MATCH(KW_COMMA))
break;
}
if (select_flag)
node->syn_arg[s_sto_rse] = parse_sql_rse();
else
parse_matching_paren();
if (distinct)
node->syn_arg[s_sto_rse]->syn_arg[s_rse_reduced] =
make_list(distinct);
node->syn_arg[s_sto_fields] = (SYN) fields;
node->syn_arg[s_sto_values] = (SYN) values;
--sql_flag;
return node;
}
static NOD_T parse_join_type(void)
{
/**************************************
*
* p a r s e _ j o i n _ t y p e
*
**************************************
*
* Functional description
* Parse a join type.
*
**************************************/
NOD_T operatr;
if (MATCH(KW_INNER))
operatr = nod_join_inner;
else if (MATCH(KW_LEFT))
operatr = nod_join_left;
else if (MATCH(KW_RIGHT))
operatr = nod_join_right;
else if (MATCH(KW_FULL))
operatr = nod_join_full;
else if (MATCH(KW_JOIN))
return nod_join_inner;
else
return (NOD_T) 0;
if (operatr != nod_join_inner)
MATCH(KW_OUTER);
if (!MATCH(KW_JOIN))
SYNTAX_ERROR(489); /* Msg489 JOIN */
return operatr;
}
static SYN parse_list_fields(void)
{
/**************************************
*
* p a r s e _ l i s t _ f i e l d s
*
**************************************
*
* Functional description
* Parse a LIST statement. LIST is like PRINT, but does vertical
* formatting.
*
**************************************/
SYN node, item;
LLS stack;
node = SYNTAX_NODE(nod_list_fields, s_prt_count);
PAR_token();
if (!MATCH(KW_ALL) && MATCH(KW_DISTINCT))
node->syn_arg[s_prt_distinct] = INT_CAST TRUE;
if (node->syn_arg[s_prt_output] = parse_output())
return node;
if (test_end())
return node;
/* If there is a potential record selection expression, there obviously
can't be a print list. Get the rse. Otherwise, pick up the print
list. */
if (potential_rse())
node->syn_arg[s_prt_rse] = parse_rse();
else {
if (!test_end()) {
stack = NULL;
for (;;) {
item = SYNTAX_NODE(nod_print_item, s_itm_count);
item->syn_arg[s_itm_value] = parse_value(0, 0);
item->syn_arg[s_itm_edit_string] = (SYN) parse_edit_string();
LLS_PUSH(item, &stack);
if (!MATCH(KW_COMMA) && !MATCH(KW_AND))
break;
PAR_real();
if (MATCH(KW_AND))
PAR_real();
}
node->syn_arg[s_prt_list] = make_list(stack);
}
if (MATCH(KW_OF))
node->syn_arg[s_prt_rse] = parse_rse();
}
node->syn_arg[s_prt_output] = parse_output();
return node;
}
static CON parse_literal(void)
{
/**************************************
*
* p a r s e _ l i t e r a l
*
**************************************
*
* Functional description
* Parse a literal expression.
*
**************************************/
CON constant;
USHORT l;
UCHAR *p, *q;
PAR_real();
q = (UCHAR *) QLI_token->tok_string;
l = QLI_token->tok_length;
if (QLI_token->tok_type == tok_quoted) {
q++;
l -= 2;
constant = (CON) ALLOCDV(type_con, l);
constant->con_desc.dsc_dtype = dtype_text;
p = constant->con_desc.dsc_address = constant->con_data;
if (constant->con_desc.dsc_length = l)
do
*p++ = *q++;
while (--l);
}
else if (QLI_token->tok_type == tok_number)
constant = make_numeric_constant(QLI_token->tok_string,
QLI_token->tok_length);
else
SYNTAX_ERROR(190); /* Msg190 value expression */
PAR_token();
return constant;
}
static SYN parse_matches(void)
{
/**************************************
*
* p a r s e _ m a t c h e s
*
**************************************
*
* Functional description
* Parse a matching expressing, including
* the preset matching language template.
*
**************************************/
SYN node, language;
PAR_token();
node = SYNTAX_NODE(nod_sleuth, 3);
node->syn_arg[1] = parse_value(0, 0);
if (MATCH(KW_USING))
node->syn_arg[2] = parse_value(0, 0);
else if (QLI_matching_language) {
node->syn_arg[2] = language = SYNTAX_NODE(nod_constant, 1);
language->syn_arg[0] = (SYN) QLI_matching_language;
}
else {
node->syn_type = nod_matches;
node->syn_count = 2;
}
return node;
}
static void parse_matching_paren(void)
{
/**************************************
*
* p a r s e _ m a t c h i n g _ p a r e n
*
**************************************
*
* Functional description
* Check for a trailing (right) parenthesis. Complain if
* its not there.
*
**************************************/
PAR_real();
if (MATCH(KW_RIGHT_PAREN))
return;
SYNTAX_ERROR(191); /* Msg191 right parenthesis */
}
static SYN parse_menu(void)
{
/**************************************
*
* p a r s e _ m e n u
*
**************************************
*
* Functional description
* Parse a MENU statement.
*
**************************************/
LLS labels, statements;
SYN node;
PAR_real_token();
if (QLI_token->tok_type != tok_quoted)
SYNTAX_ERROR(192); /* Msg192 quoted string */
node = SYNTAX_NODE(nod_menu, s_men_count);
node->syn_arg[s_men_string] = parse_primitive_value(0, 0);
labels = statements = NULL;
PAR_real();
while (!MATCH(KW_END)) {
if (!MATCH(KW_ENTREE))
SYNTAX_ERROR(193); /* Msg193 ENTREE or END */
if (QLI_token->tok_type != tok_quoted)
SYNTAX_ERROR(194); /* Msg194 quoted string */
LLS_PUSH(parse_primitive_value(0, 0), &labels);
MATCH(KW_COLON);
LLS_PUSH(parse_statement(), &statements);
PAR_real();
}
node->syn_arg[s_men_statements] = make_list(statements);
node->syn_arg[s_men_labels] = make_list(labels);
return node;
}
static SYN parse_modify(void)
{
/**************************************
*
* p a r s e _ m o d i f y
*
**************************************
*
* Functional description
* Parse a MODIFY statement. Modify can be any of the
* following:
*
* MODIFY [ALL] [<field> [, <field>]...] [OF <rse> ]
* MODIFY [ALL] USING <statement> [OF <rse>]
*
**************************************/
SYN node;
LLS stack;
PAR_token();
/* If this is a meta-data change, handle it elsewhere */
if (MATCH(KW_INDEX))
return parse_modify_index();
if (MATCH(KW_FIELD)) {
node = SYNTAX_NODE(nod_mod_field, 1);
node->syn_arg[0] = (SYN) parse_database();
node->syn_arg[1] = (SYN) parse_field(TRUE);
return node;
}
if (MATCH(KW_RELATION))
return parse_modify_relation();
/* Not a meta-data modification, just a simple data modify */
MATCH(KW_ALL);
node = SYNTAX_NODE(nod_modify, s_mod_count);
if (MATCH(KW_USING))
#if PYXIS
if (MATCH(KW_FORM))
IBERROR(484); /* FORMs not supported */
else
#endif
node->syn_arg[s_mod_statement] = parse_statement();
else if (!KEYWORD(KW_SEMI)) {
stack = NULL;
while (TRUE) {
LLS_PUSH(parse_field_name(0), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_mod_list] = make_list(stack);
}
if (MATCH(KW_OF))
node->syn_arg[s_mod_rse] = parse_rse();
return node;
}
static SYN parse_modify_index(void)
{
/**************************************
*
* p a r s e _ m o d i f y _ i n d e x
*
**************************************
*
* Functional description
* Parse a MODIFY INDEX statement.
*
**************************************/
SYN node;
node = SYNTAX_NODE(nod_mod_index, s_mfi_count);
node->syn_arg[s_mfi_database] = (SYN) parse_database();
node->syn_arg[s_mfi_name] = (SYN) parse_name();
PAR_real();
while (TRUE) {
if (MATCH(KW_UNIQUE))
node->syn_flags |= (s_dfi_flag_selectivity | s_dfi_flag_unique);
else if (MATCH(KW_DUPLICATE)) {
node->syn_flags |= s_dfi_flag_selectivity;
node->syn_flags &= ~s_dfi_flag_unique;
}
else if (MATCH(KW_INACTIVE))
node->syn_flags |= (s_dfi_flag_activity | s_dfi_flag_inactive);
else if (MATCH(KW_ACTIVE)) {
node->syn_flags |= s_dfi_flag_activity;
node->syn_flags &= ~s_dfi_flag_inactive;
}
else if (MATCH(KW_DESCENDING))
node->syn_flags |= (s_dfi_flag_order | s_dfi_flag_descending);
else if (MATCH(KW_ASCENDING)) {
node->syn_flags |= s_dfi_flag_order;
node->syn_flags &= ~s_dfi_flag_inactive;
}
else if (MATCH(KW_STATISTICS))
node->syn_flags |= s_dfi_flag_statistics;
else
break;
}
if (!node->syn_flags)
SYNTAX_ERROR(195); /* Msg195 index state option */
command_end();
return node;
}
static SYN parse_modify_relation(void)
{
/**************************************
*
* p a r s e _ m o d i f y _ r e l a t i o n
*
**************************************
*
* Functional description
* Parse a MODIFY RELATION statement.
*
**************************************/
SYN node;
QLI_REL relation;
QLI_FLD field;
node = SYNTAX_NODE(nod_mod_relation, 2);
relation = parse_qualified_relation();
node->syn_arg[0] = (SYN) relation;
if (!relation)
SYNTAX_ERROR(196); /* Msg196 relation name */
for (;;) {
PAR_real();
if (MATCH(KW_ADD)) {
PAR_real();
MATCH(KW_FIELD);
field = parse_field(FALSE);
}
else if (MATCH(KW_MODIFY)) {
PAR_real();
MATCH(KW_FIELD);
field = parse_field(FALSE);
field->fld_flags |= FLD_modify;
}
else if (MATCH(KW_DROP)) {
PAR_real();
MATCH(KW_FIELD);
field = parse_field(FALSE);
field->fld_flags |= FLD_drop;
}
else
SYNTAX_ERROR(197); /* Msg197 ADD, MODIFY, or DROP */
field->fld_next = (QLI_FLD) node->syn_arg[1];
node->syn_arg[1] = (SYN) field;
if (KEYWORD(KW_SEMI))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(198); /* Msg198 comma between field definitions */
}
command_end();
return node;
}
static SYN parse_multiply( USHORT * paren_count, USHORT * bool_flag)
{
/**************************************
*
* p a r s e _ m u l t i p l y
*
**************************************
*
* Functional description
* Parse the operatrs * and /.
*
**************************************/
SYN node, arg;
NOD_T operatr;
node = parse_from(paren_count, bool_flag);
while (TRUE) {
if (MATCH(KW_ASTERISK))
operatr = nod_multiply;
else if (MATCH(KW_SLASH))
operatr = nod_divide;
else
return node;
arg = node;
node = SYNTAX_NODE(operatr, 2);
node->syn_arg[0] = arg;
node->syn_arg[1] = parse_from(paren_count, bool_flag);
}
}
static NAM parse_name(void)
{
/**************************************
*
* p a r s e _ n a m e
*
**************************************
*
* Functional description
* Turn current token into a name and get the next token.
*
**************************************/
NAM name;
SSHORT l;
TEXT c, *p, *q;
PAR_real();
if (QLI_token->tok_type != tok_ident)
SYNTAX_ERROR(199); /* Msg199 identifier */
l = QLI_token->tok_length;
name = (NAM) ALLOCDV(type_nam, l);
name->nam_length = l;
name->nam_symbol = QLI_token->tok_symbol;
q = QLI_token->tok_string;
p = name->nam_string;
if (l)
do {
c = *q++;
*p++ = UPPER(c);
} while (--l);
PAR_token();
return name;
}
static SYN parse_not( USHORT * paren_count)
{
/**************************************
*
* p a r s e _ n o t
*
**************************************
*
* Functional description
* Parse a prefix NOT expression.
*
**************************************/
PAR_real();
if (!MATCH(KW_NOT))
return parse_relational(paren_count);
return negate(parse_not(paren_count));
}
static int parse_ordinal(void)
{
/**************************************
*
* p a r s e _ o r d i n a l
*
**************************************
*
* Functional description
* Pick up a simple number as a number. This is
* used for SKIP [n], SPACE [n], COL n, and SQL
* positions.
*
**************************************/
int n;
PAR_real();
if (QLI_token->tok_type != tok_number)
SYNTAX_ERROR(200); /* Msg200 positive number */
n = atoi(QLI_token->tok_string);
if (n < 0)
SYNTAX_ERROR(200); /* Msg200 positive number */
PAR_token();
return n;
}
static SYN parse_output(void)
{
/**************************************
*
* p a r s e _ o u t p u t
*
**************************************
*
* Functional description
* Parse an output clause the the absence thereof.
*
**************************************/
SYN node;
USHORT flag;
if (MATCH(KW_ON))
flag = FALSE;
else if (MATCH(KW_TO))
flag = TRUE;
else
return NULL;
node = SYNTAX_NODE(nod_output, s_out_count);
node->syn_arg[s_out_file] = parse_value(0, 0);
node->syn_arg[s_out_pipe] = INT_CAST flag;
return node;
}
static SYN parse_primitive_value( USHORT * paren_count, USHORT * bool_flag)
{
/**************************************
*
* p a r s e _ p r i m i t i v e _ v a l u e
*
**************************************
*
* Functional description
* Pick up a value expression. This may be either a field reference
* or a constant value.
*
**************************************/
SYM symbol;
SYN node, sub;
CON constant;
UCHAR *p;
USHORT local_count;
if (!paren_count) {
local_count = 0;
paren_count = &local_count;
}
PAR_real();
switch (next_keyword()) {
case KW_LEFT_PAREN:
PAR_token();
(*paren_count)++;
if (bool_flag && *bool_flag)
node = parse_boolean(paren_count);
else
node = parse_value(paren_count, bool_flag);
/*
if (*paren_count && KEYWORD (KW_RIGHT_PAREN))
*/
{
parse_matching_paren();
(*paren_count)--;
}
break;
case KW_PLUS:
PAR_token();
return parse_primitive_value(paren_count, 0);
case KW_MINUS:
PAR_token();
sub = parse_primitive_value(paren_count, 0);
if (sub->syn_type == nod_constant) {
constant = (CON) sub->syn_arg[0];
p = constant->con_desc.dsc_address;
switch (constant->con_desc.dsc_dtype) {
case dtype_long:
*(SLONG *) p = -*(SLONG *) p;
return sub;
case dtype_text:
*p = '-';
return sub;
}
}
node = SYNTAX_NODE(nod_negate, 1);
node->syn_arg[0] = sub;
break;
case KW_ASTERISK:
node = parse_prompt();
break;
case KW_EDIT:
PAR_token();
node = SYNTAX_NODE(nod_edit_blob, 1);
if (!KEYWORD(KW_SEMI))
node->syn_arg[0] = parse_value(0, 0);
break;
case KW_FORMAT:
PAR_token();
node = SYNTAX_NODE(nod_format, s_fmt_count);
node->syn_arg[s_fmt_value] = parse_value(0, 0);
node->syn_arg[s_fmt_edit] = (SYN) parse_edit_string();
break;
case KW_NULL:
PAR_token();
node = SYNTAX_NODE(nod_null, 0);
break;
case KW_USER_NAME:
PAR_token();
node = SYNTAX_NODE(nod_user_name, 0);
break;
case KW_COUNT:
case KW_MAX:
case KW_MIN:
case KW_AVERAGE:
case KW_TOTAL:
node = parse_statistical();
break;
case KW_RUNNING:
if (function_count > 0)
IBERROR(487); /* Msg487 Invalid argument for UDF */
PAR_real_token();
node = SYNTAX_NODE(nod_running_total, s_stt_count);
if (MATCH(KW_COUNT))
node->syn_type = nod_running_count;
else {
MATCH(KW_TOTAL);
node->syn_arg[s_stt_value] = parse_value(0, 0);
}
break;
case KW_SELECT:
node = parse_sql_subquery();
break;
default:
if ((symbol = QLI_token->tok_symbol)
&& symbol->sym_type == SYM_function) {
node = parse_function();
break;
}
if (QLI_token->tok_type == tok_ident) {
node = parse_field_name(0);
break;
}
node = SYNTAX_NODE(nod_constant, 1);
node->syn_arg[0] = (SYN) parse_literal();
}
return node;
}
static SYN parse_print_list(void)
{
/**************************************
*
* p a r s e _ p r i n t _ l i s t
*
**************************************
*
* Functional description
* Pick up a print item. The syntax of a print item is:
*
* <value> [ '[ <query_header> '] ] [USING <edit_string> ]
*
**************************************/
NOD_T op;
SYN node;
LLS stack;
stack = NULL;
while (TRUE) {
if (MATCH(KW_SKIP))
op = nod_skip;
else if (MATCH(KW_SPACE))
op = nod_space;
else if (MATCH(KW_TAB))
op = nod_tab;
else if (MATCH(KW_COLUMN))
op = nod_column;
else if (MATCH(KW_NEW_PAGE))
op = nod_new_page;
else if (MATCH(KW_REPORT_HEADER))
op = nod_report_header;
else if (MATCH(KW_COLUMN_HEADER))
op = nod_column_header;
else {
op = nod_print_item;
node = SYNTAX_NODE(nod_print_item, s_itm_count);
node->syn_arg[s_itm_value] = parse_value(0, 0);
if (MATCH(KW_LEFT_PAREN)) {
if (MATCH(KW_MINUS))
node->syn_arg[s_itm_header] = (SYN) "-";
else
node->syn_arg[s_itm_header] = (SYN) parse_header();
parse_matching_paren();
}
node->syn_arg[s_itm_edit_string] = (SYN) parse_edit_string();
}
if (op != nod_print_item) {
node = SYNTAX_NODE(op, 1);
node->syn_count = 0;
node->syn_arg[0] = INT_CAST 1;
if (op == nod_column || QLI_token->tok_type == tok_number)
node->syn_arg[0] = INT_CAST parse_ordinal();
if ((op == nod_skip) && ((int) node->syn_arg[0] < 1))
SYNTAX_ERROR(478); /* Msg478 number > 0 */
}
LLS_PUSH(node, &stack);
if (!MATCH(KW_COMMA) && !MATCH(KW_AND))
break;
PAR_real();
if (MATCH(KW_AND))
PAR_real();
}
node = make_list(stack);
return node;
}
static SYN parse_print(void)
{
/**************************************
*
* p a r s e _ p r i n t
*
**************************************
*
* Functional description
* Parse the PRINT statement. This is the richest and most general
* Datatrieve statement. Hence this may get a bit long. The syntax is:
*
* [<item> [, <item>]] OF <rse>
* PRINT [ ] [ON <file>]
* <rse>
*
**************************************/
SYN node;
node = SYNTAX_NODE(nod_print, s_prt_count);
PAR_token();
if (!MATCH(KW_ALL) && MATCH(KW_DISTINCT))
node->syn_arg[s_prt_distinct] = INT_CAST TRUE;
if (node->syn_arg[s_prt_output] = parse_output())
return node;
if (test_end())
return node;
/* If there is a potential record selection expression, there obviously
can't be a print list. Get the rse. Otherwise, pick up the print
list. */
if (potential_rse())
node->syn_arg[s_prt_rse] = parse_rse();
else if (!KEYWORD(KW_USING)) {
if (!KEYWORD(KW_THEN) && !KEYWORD(KW_OF) && !KEYWORD(KW_ON))
node->syn_arg[s_prt_list] = parse_print_list();
if (MATCH(KW_OF))
node->syn_arg[s_prt_rse] = parse_rse();
}
if (!node->syn_arg[s_prt_list] && MATCH(KW_USING)) {
IBERROR(484); /* FORMs not supported */
}
else
node->syn_arg[s_prt_output] = parse_output();
return node;
}
static SYN parse_prompt(void)
{
/**************************************
*
* p a r s e _ p r o m p t
*
**************************************
*
* Functional description
* Parse a prompt expression.
*
**************************************/
SYN node;
USHORT l;
TEXT *q;
PAR_token();
node = SYNTAX_NODE(nod_prompt, 1);
/* If there is a period, get the prompt string and make a string node */
if (MATCH(KW_DOT)) {
PAR_real();
l = QLI_token->tok_length;
q = QLI_token->tok_string;
if (QLI_token->tok_type == tok_quoted) {
q++;
l -= 2;
}
node->syn_arg[0] = (SYN) make_string(q, l);
PAR_token();
}
return node;
}
static QFL parse_qualified_filter(void)
{
/**************************************
*
* p a r s e _ q u a l i f i e d _ f i l t e r
*
**************************************
*
* Functional description
* This token ought to be a filter, possibly qualified.
* Return a qualified filter block, containing the
* filter name in a NAM block and the database in a
* DBB block if a database was specified. Somebody
* else will decide what to do if the database was not
* specified.
*
**************************************/
QFL filter;
filter = (QFL) ALLOCD(type_qfl);
filter->qfl_database = parse_database();
filter->qfl_name = parse_name();
return filter;
}
static QFN parse_qualified_function(void)
{
/**************************************
*
* p a r s e _ q u a l i f i e d _ f u n c t i o n
*
**************************************
*
* Functional description
* This token ought to be a function, possibly qualified.
* Return a qualified function block, containing the
* function name in a NAM block and the database in a
* DBB block if a database was specified. Somebody
* else will decide what to do if the database was not
* specified.
*
**************************************/
QFN func;
func = (QFN) ALLOCD(type_qfn);
func->qfn_database = parse_database();
func->qfn_name = parse_name();
return func;
}
static QPR parse_qualified_procedure(void)
{
/**************************************
*
* p a r s e _ q u a l i f i e d _ p r o c e d u r e
*
**************************************
*
* Functional description
* This token ought to be a procedure, possibly qualified.
* Return a qualified procedure block, containing the
* procedure name in a NAM block and the database in a
* DBB block if a database was specified. Somebody
* else will decide what to do if the database was not
* specified.
*
**************************************/
QPR proc;
proc = (QPR) ALLOCD(type_qpr);
proc->qpr_database = parse_database();
proc->qpr_name = parse_name();
return proc;
}
static QLI_REL parse_qualified_relation(void)
{
/**************************************
*
* p a r s e _ q u a l i f i e d _ r e l a t i o n
*
**************************************
*
* Functional description
* Check for a relation name, possible qualified. If there
* is a relation name, parse it and return the relation. If
* not, return NULL. Produce a syntax error only if there is
* a partially qualified name.
*
**************************************/
SYM db_symbol;
QLI_REL relation;
PAR_real();
/* If the next token is a database symbol, take it as a qualifier */
if ((db_symbol = QLI_token->tok_symbol)
&& db_symbol->sym_type == SYM_database) {
PAR_real_token();
if (!MATCH(KW_DOT))
SYNTAX_ERROR(202); /* Msg202 period in qualified relation name */
PAR_real();
if (relation = resolve_relation(db_symbol, QLI_token->tok_symbol)) {
PAR_token();
return relation;
}
ERRQ_print_error(203, QLI_token->tok_string, db_symbol->sym_string,
NULL, NULL, NULL); /* Msg203 %s is not a relation in database %s */
}
if (relation = resolve_relation(0, QLI_token->tok_symbol))
PAR_token();
return relation;
}
static SYN parse_ready( NOD_T node_type)
{
/**************************************
*
* p a r s e _ r e a d y
*
**************************************
*
* Functional description
* Parse a READY statement:
*
* READY <filename> [AS <symbol>] [,...];
*
**************************************/
SYN node;
DBB database;
NAM name;
LLS stack;
SSHORT l;
TEXT *p, *q;
stack = NULL;
while (TRUE) {
LEX_filename();
if (!(l = QLI_token->tok_length))
ERRQ_error(204, NULL, NULL, NULL, NULL, NULL); /* Msg204 database file name required on READY */
q = QLI_token->tok_string;
if (QLI_token->tok_type == tok_quoted) {
l -= 2;
q++;
}
database = (DBB) ALLOCDV(type_dbb, l);
database->dbb_filename_length = l;
p = database->dbb_filename;
do
*p++ = *q++;
while (--l);
PAR_token();
if (node_type == nod_def_database || node_type == nod_ready) {
if (MATCH(KW_AS)) {
name = parse_name();
database->dbb_symbol = (SYM) name;
if (HSH_lookup(name->nam_string, name->nam_length))
ERRQ_error(408, name->nam_string, NULL, NULL, NULL, NULL); /* Database handle is not unique */
}
else
database->dbb_symbol = (SYM) make_name();
}
else if (node_type == nod_sql_database) {
if (MATCH(KW_PAGESIZE)) {
if (database->dbb_pagesize)
SYNTAX_ERROR(390); /* Msg390 Multiple page size specifications */
if (!MATCH(KW_EQUALS))
SYNTAX_ERROR(396); /* Msg396 = (equals) */
database->dbb_pagesize = parse_ordinal();
}
database->dbb_symbol = (SYM) make_name();
}
for (;;) {
if (MATCH(KW_USER))
database->dbb_user = parse_literal();
else if (MATCH(KW_PASSWORD))
database->dbb_password = parse_literal();
else
break;
}
LLS_PUSH(database, &stack);
if (!KEYWORD(KW_COMMA) || (node_type == nod_sql_database))
break;
}
command_end();
node = make_list(stack);
node->syn_type = node_type;
return node;
}
static SYN parse_relational( USHORT * paren_count)
{
/**************************************
*
* p a r s e _ r e l a t i o n a l
*
**************************************
*
* Functional description
* Parse a relational expression.
*
**************************************/
SYN node, expr1, or_node;
USHORT negation;
NOD_T operatr, *rel_ops;
USHORT local_flag;
local_flag = TRUE;
if (MATCH(KW_ANY)) {
node = SYNTAX_NODE(nod_any, 1);
node->syn_arg[0] = parse_rse();
return node;
}
operatr = (NOD_T) 0;
if (MATCH(KW_EXISTS))
operatr = nod_any;
else if (MATCH(KW_SINGULAR))
operatr = nod_unique;
if (operatr != (NOD_T) 0) {
PAR_real();
if (MATCH(KW_LEFT_PAREN)) {
PAR_real();
if (MATCH(KW_SELECT)) {
PAR_real();
node = SYNTAX_NODE(operatr, 2);
if (!MATCH(KW_ASTERISK))
node->syn_arg[1] = parse_value(0, 0);
node->syn_arg[0] = parse_sql_rse();
parse_matching_paren();
return node;
}
}
if (operatr == nod_any)
SYNTAX_ERROR(205); /* Msg205 EXISTS (SELECT * <sql rse>) */
else
SYNTAX_ERROR(488); /* Msg488 SINGULAR (SELECT * <sql rse>) */
}
if (MATCH(KW_UNIQUE)) {
node = SYNTAX_NODE(nod_unique, 1);
node->syn_arg[0] = parse_rse();
return node;
}
expr1 = parse_value(paren_count, &local_flag);
if (KEYWORD(KW_RIGHT_PAREN))
return expr1;
if (KEYWORD(KW_SEMI))
for (rel_ops = relationals; *rel_ops != (NOD_T) 0; rel_ops++)
if (expr1->syn_type == *rel_ops)
return expr1;
negation = FALSE;
node = NULL;
MATCH(KW_IS);
PAR_real();
if (MATCH(KW_NOT)) {
negation = TRUE;
PAR_real();
}
switch (next_keyword()) {
case KW_IN:
PAR_token();
node = parse_in(expr1, nod_eql, FALSE);
break;
case KW_EQUALS:
case KW_EQ:
operatr = (negation) ? nod_neq : nod_eql;
negation = FALSE;
break;
case KW_NE:
operatr = (negation) ? nod_eql : nod_neq;
negation = FALSE;
break;
case KW_GT:
operatr = (negation) ? nod_leq : nod_gtr;
negation = FALSE;
break;
case KW_GE:
operatr = (negation) ? nod_lss : nod_geq;
negation = FALSE;
break;
case KW_LE:
operatr = (negation) ? nod_gtr : nod_leq;
negation = FALSE;
break;
case KW_LT:
operatr = (negation) ? nod_geq : nod_lss;
negation = FALSE;
break;
case KW_CONTAINING:
operatr = nod_containing;
break;
case KW_MATCHES:
node = parse_matches();
node->syn_arg[0] = expr1;
operatr = node->syn_type;
break;
case KW_LIKE:
PAR_token();
node = SYNTAX_NODE(nod_like, 3);
node->syn_arg[0] = expr1;
node->syn_arg[1] = parse_value(0, 0);
if (MATCH(KW_ESCAPE))
node->syn_arg[2] = parse_value(0, 0);
else
node->syn_count = 2;
break;
case KW_STARTS:
PAR_token();
MATCH(KW_WITH);
node = SYNTAX_NODE(nod_starts, 2);
node->syn_arg[0] = expr1;
node->syn_arg[1] = parse_value(0, 0);
break;
case KW_NULL:
case KW_MISSING:
PAR_token();
node = SYNTAX_NODE(nod_missing, 1);
node->syn_arg[0] = expr1;
break;
case KW_BETWEEN:
PAR_token();
node = SYNTAX_NODE(nod_between, 3);
node->syn_arg[0] = expr1;
node->syn_arg[1] = parse_value(0, 0);
MATCH(KW_AND);
node->syn_arg[2] = parse_value(0, 0);
break;
default:
for (rel_ops = relationals; *rel_ops != (NOD_T) 0; rel_ops++)
if (expr1->syn_type == *rel_ops)
return expr1;
SYNTAX_ERROR(206); /* Msg206 relational operatr */
}
/* If we haven't already built a node, it must be an ordinary binary operatr.
Build it. */
if (!node) {
PAR_token();
if (MATCH(KW_ANY))
return parse_in(expr1, operatr, FALSE);
if (MATCH(KW_ALL))
return parse_in(expr1, operatr, TRUE);
node = SYNTAX_NODE(operatr, 2);
node->syn_arg[0] = expr1;
node->syn_arg[1] = parse_value(paren_count, &local_flag);
}
/* If a negation remains to be handled, zap the node under a NOT. */
if (negation)
node = negate(node);
/* If the node isn't an equality, we've done. Since equalities can be
structured as implicit ORs, build them here. */
if (operatr != nod_eql)
return node;
/* We have an equality operation, which can take a number of values. Generate
implicit ORs */
while (MATCH(KW_COMMA)) {
PAR_real();
MATCH(KW_OR);
or_node = SYNTAX_NODE(nod_or, 2);
or_node->syn_arg[0] = node;
or_node->syn_arg[1] = node = SYNTAX_NODE(nod_eql, 2);
node->syn_arg[0] = expr1;
node->syn_arg[1] = parse_value(paren_count, &local_flag);
node = or_node;
}
return node;
}
static SYN parse_relation(void)
{
/**************************************
*
* p a r s e _ r e l a t i o n
*
**************************************
*
* Functional description
* Parse a relation expression. Syntax is:
*
* [ <context_variable> IN ] <relation>
*
**************************************/
SYN node;
SYM context;
node = SYNTAX_NODE(nod_relation, s_rel_count);
/* Token wasn't a relation name, maybe it's a context variable. */
if (!(node->syn_arg[s_rel_relation] = (SYN) parse_qualified_relation())) {
context = parse_symbol();
node->syn_arg[s_rel_context] = (SYN) context;
if (sql_flag || !MATCH(KW_IN)) {
if (!QLI_databases)
IBERROR(207); /* Msg207 a database has not been readied */
ERRQ_print_error(208, context->sym_string, NULL, NULL, NULL, NULL); /* Msg208 expected \"relation_name\", encountered \"%s\ */
}
if (!
(node->syn_arg[s_rel_relation] =
(SYN) parse_qualified_relation()))SYNTAX_ERROR(209); /* Msg209 relation name */
}
return node;
}
static SYN parse_rename(void)
{
/**************************************
*
* p a r s e _ r e n a m e
*
**************************************
*
* Functional description
* Parse a PROCEDURE rename statement.
*
**************************************/
SYN node;
PAR_real_token();
if (!MATCH(KW_PROCEDURE))
SYNTAX_ERROR(210); /* Msg210 PROCEDURE */
node = SYNTAX_NODE(nod_rename_proc, 2);
node->syn_arg[0] = (SYN) parse_qualified_procedure();
MATCH(KW_TO);
node->syn_arg[1] = (SYN) parse_qualified_procedure();
return node;
}
static SYN parse_repeat(void)
{
/**************************************
*
* p a r s e _ r e p e a t
*
**************************************
*
* Functional description
* Parse REPEAT statement.
*
**************************************/
SYN node;
PAR_token();
node = SYNTAX_NODE(nod_repeat, s_rpt_count);
node->syn_arg[s_rpt_value] = parse_value(0, 0);
node->syn_arg[s_rpt_statement] = parse_statement();
return node;
}
static SYN parse_report(void)
{
/**************************************
*
* p a r s e _ r e p o r t
*
**************************************
*
* Functional description
* Parse a report specification.
*
**************************************/
SYN node, rse, flds, qli_fld, rse_fld;
RPT report;
BRK control, *ptr, tmpptr, tmpptr1;
USHORT top, i, srt_syn, ctl_syn, syn_count;
NAM name1, name2;
++sw_report;
PAR_token();
report = (RPT) ALLOCD(type_rpt);
node = SYNTAX_NODE(nod_report, s_prt_count);
node->syn_arg[s_prt_list] = (SYN) report;
/* Pick up record select expression */
rse = node->syn_arg[s_prt_rse] = parse_rse();
node->syn_arg[s_prt_output] = parse_output();
/* Pick up report clauses */
for (;;) {
PAR_real();
if (MATCH(KW_END_REPORT))
break;
switch (next_keyword()) {
case KW_PRINT:
PAR_token();
report->rpt_detail_line = (QLI_NOD) parse_print_list();
break;
case KW_AT:
PAR_token();
if (MATCH(KW_TOP))
top = TRUE;
else if (MATCH(KW_BOTTOM))
top = FALSE;
else
SYNTAX_ERROR(382); /* Msg382 TOP or BOTTOM */
MATCH(KW_OF);
if (MATCH(KW_REPORT)) {
control = (BRK) ALLOCD(type_brk);
ptr = (top) ? &report->rpt_top_rpt : &report->rpt_bottom_rpt;
control->brk_next = *ptr;
*ptr = control;
MATCH(KW_PRINT);
control->brk_line = parse_print_list();
}
else if (MATCH(KW_PAGE)) {
control = (BRK) ALLOCD(type_brk);
ptr =
(top) ? &report->rpt_top_page : &report->rpt_bottom_page;
control->brk_next = *ptr;
*ptr = control;
MATCH(KW_PRINT);
control->brk_line = parse_print_list();
}
else {
ptr = (top) ? &report->rpt_top_breaks :
&report->rpt_bottom_breaks;
if (!*ptr) {
/* control breaks should only be on sorted fields, set up list
of control breaks based on sorted fields and then add action (print)
items to that list. */
if (!(flds = rse->syn_arg[s_rse_sort]))
SYNTAX_ERROR(383); /* Msg383 sort field */
tmpptr = *ptr;
for (i = 0; i < flds->syn_count; i += 2) {
control = (BRK) ALLOCD(type_brk);
control->brk_field = flds->syn_arg[i];
control->brk_line = NULL;
control->brk_statisticals = NULL;
control->brk_next = tmpptr;
tmpptr = control;
}
if (!top) {
/* reverse the 'at bottom' control break list as the
lower control breaks should be performed prior to the higher ones. */
for (tmpptr1 = tmpptr->brk_next, control = 0; tmpptr;) {
tmpptr->brk_next = control;
control = tmpptr;
if (tmpptr = tmpptr1)
tmpptr1 = tmpptr->brk_next;
}
tmpptr = control;
}
*ptr = tmpptr;
}
qli_fld = parse_field_name(0);
for (control = *ptr; control; control = control->brk_next) {
rse_fld = (SYN) control->brk_field;
if (rse_fld->syn_type != qli_fld->syn_type)
continue;
/* if number of field qualifiers on sort field and control field
are not equal test match of rightmost set */
syn_count = MIN(rse_fld->syn_count, qli_fld->syn_count);
srt_syn = ctl_syn = 0;
if (syn_count != rse_fld->syn_count)
srt_syn = rse_fld->syn_count - syn_count;
if (syn_count != qli_fld->syn_count)
ctl_syn = qli_fld->syn_count - syn_count;
for (i = 0; i < syn_count; i++) {
name1 = (NAM) rse_fld->syn_arg[i + srt_syn];
name2 = (NAM) qli_fld->syn_arg[i + ctl_syn];
if (strcmp(name1->nam_string, name2->nam_string))
break;
}
if (i == qli_fld->syn_count)
break;
}
if (!control)
SYNTAX_ERROR(383); /* Msg383 sort field */
MATCH(KW_PRINT);
control->brk_field = qli_fld;
control->brk_line = parse_print_list();
}
break;
case KW_SET:
PAR_token();
if (MATCH(KW_COLUMNS)) {
MATCH(KW_EQUALS);
report->rpt_columns = parse_ordinal();
}
else if (MATCH(KW_LINES)) {
MATCH(KW_EQUALS);
report->rpt_lines = parse_ordinal();
}
else if (MATCH(KW_REPORT_NAME)) {
MATCH(KW_EQUALS);
report->rpt_name = (TEXT *) parse_header();
}
else
SYNTAX_ERROR(212); /* Msg212 report writer SET option */
break;
default:
SYNTAX_ERROR(213); /* Msg213 report item */
}
MATCH(KW_SEMI);
}
if (!node->syn_arg[s_prt_output])
node->syn_arg[s_prt_output] = parse_output();
check_end();
--sw_report;
return node;
}
static SYN parse_rse(void)
{
/**************************************
*
* p a r s e _ r s e
*
**************************************
*
* Functional description
* Parse a record selection expression.
*
**************************************/
SYN node, first, over, *ptr;
LLS stack, field_stack;
USHORT count, sw_with;
count = sw_with = 0;
first = NULL;
stack = NULL;
PAR_real();
if (MATCH(KW_ALL))
PAR_real();
if (MATCH(KW_FIRST))
first = parse_value(0, 0);
while (TRUE) {
count++;
LLS_PUSH(parse_relation(), &stack);
over = NULL;
if (MATCH(KW_OVER)) {
field_stack = NULL;
while (TRUE) {
LLS_PUSH(parse_field_name(0), &field_stack);
if (!MATCH(KW_COMMA))
break;
}
over = make_list(field_stack);
}
LLS_PUSH(over, &stack);
if (!MATCH(KW_CROSS))
break;
}
node = SYNTAX_NODE(nod_rse, (int) s_rse_count + 2 * count);
node->syn_count = count;
node->syn_arg[s_rse_first] = first;
ptr = &node->syn_arg[(int) s_rse_count + 2 * count];
while (stack)
*--ptr = (SYN) LLS_POP(&stack);
/* Pick up various other clauses */
while (TRUE) {
if (MATCH(KW_WITH)) {
if (!sw_with) {
sw_with++;
node->syn_arg[s_rse_boolean] = parse_boolean(0);
}
else
IBERROR(384); /* Msg384 Too many WITHs */
}
else if (MATCH(KW_SORTED)) {
PAR_real();
MATCH(KW_BY);
node->syn_arg[s_rse_sort] = parse_sort();
}
#ifdef PC_ENGINE
else if (MATCH(KW_USING)) {
PAR_real();
MATCH(KW_INDEX);
node->syn_arg[s_rse_index] = (SYN) parse_name();
}
#endif
else if (MATCH(KW_REDUCED)) {
PAR_real();
MATCH(KW_TO);
node->syn_arg[s_rse_reduced] = parse_sort();
}
else if (MATCH(KW_GROUP)) {
PAR_real();
MATCH(KW_BY);
stack = NULL;
while (TRUE) {
LLS_PUSH(parse_udf_or_field(), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_rse_group_by] = make_list(stack);
}
else
break;
}
return node;
}
static SYN parse_select(void)
{
/**************************************
*
* p a r s e _ s e l e c t
*
**************************************
*
* Functional description
* Parse a SQL select statement.
*
**************************************/
SYN node, item, rse;
LLS stack;
++sql_flag;
PAR_token();
node = SYNTAX_NODE(nod_print, s_prt_count);
if (!MATCH(KW_ALL) && MATCH(KW_DISTINCT))
node->syn_arg[s_prt_distinct] = INT_CAST TRUE;
/* Get list of items */
if (!MATCH(KW_ASTERISK)) {
stack = NULL;
while (TRUE) {
item = SYNTAX_NODE(nod_print_item, s_itm_count);
item->syn_arg[s_itm_value] = parse_value(0, 0);
LLS_PUSH(item, &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_prt_list] = make_list(stack);
}
node->syn_arg[s_prt_rse] = rse = parse_sql_rse();
rse->syn_arg[s_rse_list] = node->syn_arg[s_prt_list];
if (MATCH(KW_ORDER)) {
PAR_real();
MATCH(KW_BY);
node->syn_arg[s_prt_order] = parse_sort();
}
--sql_flag;
return node;
}
static SYN parse_set(void)
{
/**************************************
*
* p a r s e _ s e t
*
**************************************
*
* Functional description
* Parse a SET statement.
*
**************************************/
SYN node;
LLS stack;
ENUM set_t sw;
USHORT count;
U_IPTR value;
PAR_token();
stack = NULL;
count = 0;
while (TRUE) {
PAR_real();
value = TRUE;
if (MATCH(KW_NO)) {
value = FALSE;
PAR_real();
}
switch (QLI_token->tok_keyword) {
case KW_BLR:
sw = set_blr;
PAR_token();
break;
case KW_STATISTICS:
sw = set_statistics;
PAR_token();
break;
case KW_COLUMNS:
sw = set_columns;
PAR_token();
MATCH(KW_EQUALS);
value = parse_ordinal();
break;
case KW_LINES:
sw = set_lines;
PAR_token();
MATCH(KW_EQUALS);
value = parse_ordinal();
break;
case KW_SEMICOLON:
sw = set_semi;
PAR_token();
break;
case KW_ECHO:
sw = set_echo;
PAR_token();
break;
#ifdef PYXIS
case KW_FORM:
case KW_FORMS:
IBERROR(484); /* FORMs not supported */
break;
#endif
case KW_MATCHING_LANGUAGE:
sw = set_matching_language;
PAR_token();
MATCH(KW_EQUALS);
if (value)
value = (U_IPTR) parse_literal();
break;
case KW_PASSWORD:
sw = set_password;
PAR_token();
value = (U_IPTR) parse_literal();
break;
case KW_PROMPT:
sw = set_prompt;
PAR_token();
value = (U_IPTR) parse_literal();
break;
case KW_CONT_PROMPT:
sw = set_continuation;
PAR_token();
value = (U_IPTR) parse_literal();
break;
case KW_USER:
sw = set_user;
PAR_token();
value = (U_IPTR) parse_literal();
break;
case KW_COUNT:
sw = set_count;
PAR_token();
break;
case KW_CHAR:
sw = set_charset;
PAR_token();
MATCH(KW_SET);
if (value) { /* allow for NO */
MATCH(KW_EQUALS);
value = (U_IPTR) parse_name();
}
break;
case KW_NAMES:
sw = set_charset;
PAR_token();
if (value) {
value = (U_IPTR) parse_name();
}
break;
#ifdef DEV_BUILD
case KW_EXPLAIN:
sw = set_explain;
PAR_token();
break;
case KW_HEXOUT:
sw = set_hex_output;
PAR_token();
break;
#endif
default:
SYNTAX_ERROR(214); /* Msg214 set option */
}
LLS_PUSH(sw, &stack);
LLS_PUSH(value, &stack);
count++;
if (!MATCH(KW_COMMA))
break;
}
command_end();
node = make_list(stack);
node->syn_count = count;
node->syn_type = nod_set;
return node;
}
static SYN parse_shell(void)
{
/**************************************
*
* p a r s e _ s h e l l
*
**************************************
*
* Functional description
* Parse SHELL command.
*
**************************************/
SYN node;
PAR_token();
node = SYNTAX_NODE(nod_shell, 1);
if (!KEYWORD(KW_SEMI))
node->syn_arg[0] = (SYN) parse_literal();
command_end();
return node;
}
static SYN parse_show(void)
{
/**************************************
*
* p a r s e _ s h o w
*
**************************************
*
* Functional description
* Parse a SHOW statement. The first
* group are individual things (one
* named relation, field, form, ...)
*
* the second group are sets of things
* and can be qualified with a FOR
* [DATABASE] <handle>
*
**************************************/
SYN node;
QLI_REL relation;
LLS stack;
SYM symbol;
ENUM show_t sw;
BLK value;
USHORT count;
QPR proc;
PAR_token();
stack = NULL;
count = 0;
while (TRUE) {
PAR_real();
value = NULL;
symbol = QLI_token->tok_symbol;
if (MATCH(KW_ALL))
sw = show_all;
else if (MATCH(KW_MATCHING_LANGUAGE))
sw = show_matching_language;
else if (MATCH(KW_VERSION))
sw = show_version;
else if (MATCH(KW_RELATION)) {
if (!(value = (BLK) parse_qualified_relation()))
SYNTAX_ERROR(216); /* Msg216 relation name */
else
sw = show_relation;
}
else if (MATCH(KW_FILTER)) {
sw = show_filter;
value = (BLK) parse_qualified_filter();
}
else if (MATCH(KW_FUNCTION)) {
sw = show_function;
value = (BLK) parse_qualified_function();
}
else if ((MATCH(KW_DATABASES)) || (MATCH(KW_READY)))
sw = show_databases;
else if (MATCH(KW_DATABASE)) {
sw = show_database;
if (value = (BLK) get_dbb(QLI_token->tok_symbol))
PAR_token();
}
else if (MATCH(KW_FIELD)) {
sw = show_field;
value = (BLK) parse_field_name(0);
}
else if (MATCH(KW_PROCEDURE)) {
sw = show_procedure;
value = (BLK) parse_qualified_procedure();
}
else if (MATCH(KW_VARIABLE)) {
sw = show_variable;
value = (BLK) parse_name();
}
else if (MATCH(KW_VARIABLES))
sw = show_variables;
else if (MATCH(KW_FIELDS)) {
if (MATCH(KW_FOR)) {
if (MATCH(KW_DATABASE)) {
if (value = (BLK) get_dbb(QLI_token->tok_symbol))
PAR_token();
else
SYNTAX_ERROR(221); /* Msg221 database name */
sw = show_db_fields;
}
else {
MATCH(KW_RELATION);
if (!(value = (BLK) parse_qualified_relation()))
SYNTAX_ERROR(218); /* Msg218 relation name */
else
sw = show_relation;
}
}
else
sw = show_all;
}
else if (MATCH(KW_INDICES)) {
sw = show_indices;
if (MATCH(KW_FOR))
if (MATCH(KW_DATABASE)) {
if (value = (BLK) get_dbb(QLI_token->tok_symbol))
PAR_token();
else
SYNTAX_ERROR(221); /* Msg221 database name */
sw = show_db_indices;
}
else if (!(value = (BLK) parse_qualified_relation()))
SYNTAX_ERROR(220); /* Msg220 relation name */
}
else if (MATCH(KW_SECURITY_CLASS)) {
sw = show_security_class;
value = (BLK) parse_name();
}
else if (MATCH(KW_TRIGGERS)) {
sw = show_triggers;
if (MATCH(KW_FOR)) {
if (MATCH(KW_DATABASE)) {
if (value = (BLK) get_dbb(QLI_token->tok_symbol))
PAR_token();
else
SYNTAX_ERROR(221); /* Msg221 database name */
}
else {
MATCH(KW_RELATION);
if (!(value = (BLK) parse_qualified_relation()))
SYNTAX_ERROR(222); /* Msg222 relation_name */
sw = show_trigger;
}
}
}
else if (MATCH(KW_RELATIONS))
sw = show_relations;
else if (MATCH(KW_VIEWS))
sw = show_views;
else if (MATCH(KW_SECURITY_CLASSES))
sw = show_security_classes;
else if (MATCH(KW_SYSTEM)) {
if (MATCH(KW_TRIGGERS))
sw = show_system_triggers;
else if (MATCH(KW_RELATIONS) ||
QLI_token->tok_type == tok_eol ||
KEYWORD(KW_SEMI) || KEYWORD(KW_FOR))
sw = show_system_relations;
else
SYNTAX_ERROR(215); /* Msg215 RELATIONS or TRIGGERS */
}
else if (MATCH(KW_PROCEDURES))
sw = show_procedures;
else if (MATCH(KW_FILTERS))
sw = show_filters;
else if (MATCH(KW_FUNCTIONS))
sw = show_functions;
#ifdef PYXIS
else if (MATCH(KW_FORMS)) {
sw = show_forms;
}
#endif
else if (MATCH(KW_GLOBAL)) {
PAR_real();
if (MATCH(KW_FIELD)) {
sw = show_global_field;
value = (BLK) parse_field_name(0);
}
else if (MATCH(KW_FIELDS))
sw = show_global_fields;
}
else if (symbol && symbol->sym_type == SYM_relation) {
sw = show_relation;
value = symbol->sym_object;
PAR_token();
}
else if (value = (BLK) get_dbb(symbol)) {
sw = show_database;
PAR_token();
if (MATCH(KW_DOT))
if (MATCH(KW_RELATIONS))
sw = show_relations;
else if (MATCH(KW_FIELDS))
sw = show_db_fields;
else if (MATCH(KW_INDICES))
sw = show_db_indices;
else if (MATCH(KW_SECURITY_CLASS))
sw = show_security_class;
else if (MATCH(KW_TRIGGERS))
sw = show_triggers;
else if (MATCH(KW_VIEWS))
sw = show_views;
else if (MATCH(KW_SECURITY_CLASSES))
sw = show_security_classes;
else if (MATCH(KW_SYSTEM)) {
if (MATCH(KW_TRIGGERS))
sw = show_system_triggers;
else if (MATCH(KW_RELATIONS) ||
QLI_token->tok_type == tok_eol ||
KEYWORD(KW_SEMI) || KEYWORD(KW_FOR))
sw = show_system_relations;
else
SYNTAX_ERROR(215); /* Msg215 RELATIONS or TRIGGERS */
}
else if (MATCH(KW_PROCEDURES))
sw = show_procedures;
else if (MATCH(KW_FILTERS))
sw = show_filters;
else if (MATCH(KW_FUNCTIONS))
sw = show_functions;
#ifdef PYXIS
else if (MATCH(KW_FORMS))
sw = show_forms;
#endif
else if (MATCH(KW_GLOBAL)) {
PAR_real();
if (MATCH(KW_FIELD)) {
sw = show_global_field;
value = (BLK) parse_field_name(0);
}
else if (MATCH(KW_FIELDS))
sw = show_global_fields;
}
else if (relation =
resolve_relation(symbol, QLI_token->tok_symbol)) {
sw = show_relation;
value = relation->rel_symbol->sym_object;
PAR_token();
}
else {
sw = show_procedure;
proc = (QPR) ALLOCD(type_qpr);
proc->qpr_database = (DBB) value;
proc->qpr_name = parse_name();
value = (BLK) proc;
}
}
else {
sw = show_procedure;
value = (BLK) parse_qualified_procedure();
}
LLS_PUSH(sw, &stack);
if (!value && (sw == show_relations || sw == show_views ||
sw == show_security_classes
|| sw == show_system_triggers
|| sw == show_system_relations || sw == show_procedures
#ifdef PYXIS
|| sw == show_forms
#endif
|| sw == show_filters
|| sw == show_functions || sw == show_global_fields))
if (MATCH(KW_FOR)) {
MATCH(KW_DATABASE);
if (value = (BLK) get_dbb(QLI_token->tok_symbol))
PAR_token();
else
SYNTAX_ERROR(221); /* Msg221 database name */
}
LLS_PUSH(value, &stack);
count++;
if (!MATCH(KW_COMMA))
break;
}
command_end();
node = make_list(stack);
node->syn_count = count;
node->syn_type = nod_show;
return node;
}
static SYN parse_sort(void)
{
/**************************************
*
* p a r s e _ s o r t
*
**************************************
*
* Functional description
* Parse a sort list.
*
**************************************/
SYN node, upcase;
LLS stack;
USHORT direction, sensitive;
direction = sensitive = 0;
stack = NULL;
while (TRUE) {
PAR_real();
if (!sql_flag) {
if (MATCH(KW_ASCENDING)) {
direction = 0;
continue;
}
else if (MATCH(KW_DESCENDING)) {
direction = 1;
continue;
}
else if (MATCH(KW_EXACTCASE)) {
sensitive = 0;
continue;
}
else if (MATCH(KW_ANYCASE)) {
sensitive = 1;
continue;
}
}
if (sql_flag && QLI_token->tok_type == tok_number) {
node = SYNTAX_NODE(nod_position, 1);
node->syn_arg[0] = INT_CAST parse_ordinal();
}
else
node = parse_value(0, 0);
if (sensitive) {
upcase = SYNTAX_NODE(nod_upcase, 1);
upcase->syn_arg[0] = node;
LLS_PUSH(upcase, &stack);
}
else
LLS_PUSH(node, &stack);
if (sql_flag)
if (MATCH(KW_ASCENDING))
direction = 0;
else if (MATCH(KW_DESCENDING))
direction = 1;
LLS_PUSH(direction, &stack);
if (!MATCH(KW_COMMA))
break;
}
return make_list(stack);
}
static SYN parse_sql_alter(void)
{
/**************************************
*
* p a r s e _ s q l _ a l t e r
*
**************************************
*
* Functional description
* Parse the leading clauses of a SQL ALTER statement.
*
**************************************/
SYN node;
QLI_REL relation;
QLI_FLD field;
PAR_real_token();
if (!MATCH(KW_TABLE))
SYNTAX_ERROR(407); /* Msg407 TABLE */
node = SYNTAX_NODE(nod_sql_al_table, 2);
relation = parse_qualified_relation();
node->syn_arg[0] = (SYN) relation;
for (;;) {
if (MATCH(KW_ADD)) {
field = parse_sql_field();
field->fld_flags |= FLD_add;
}
else if (MATCH(KW_DROP)) {
field = parse_field(FALSE);
field->fld_flags |= FLD_drop;
}
else
SYNTAX_ERROR(405); /* Msg405 ADD or DROP */
field->fld_next = (QLI_FLD) node->syn_arg[1];
node->syn_arg[1] = (SYN) field;
if (!MATCH(KW_COMMA))
break;
}
command_end();
return node;
}
static SYN parse_sql_create(void)
{
/**************************************
*
* p a r s e _ s q l _ c r e a t e
*
**************************************
*
* Functional description
* Parse the leading clauses of a SQL CREATE statement.
*
**************************************/
USHORT unique, descending;
PAR_real_token();
if (KEYWORD(KW_DATABASE))
return parse_ready(nod_sql_database);
if (KEYWORD(KW_UNIQUE) || KEYWORD(KW_ASCENDING)
|| KEYWORD(KW_DESCENDING) || KEYWORD(KW_INDEX)) {
unique = descending = FALSE;
while (TRUE) {
if (MATCH(KW_UNIQUE))
unique = TRUE;
else if (MATCH(KW_ASCENDING))
descending = FALSE;
else if (MATCH(KW_DESCENDING))
descending = TRUE;
else if (MATCH(KW_INDEX))
return parse_sql_index_create(unique, descending);
else
SYNTAX_ERROR(389); /* Msg389 INDEX */
}
}
if (MATCH(KW_TABLE))
return parse_sql_table_create();
#ifdef NOT_USED_OR_REPLACED
/***
if (MATCH (KW_VIEW))
return parse_sql_view_create();
***/
#endif
SYNTAX_ERROR(386); /* Msg386 object type for CREATE */
return NULL;
}
static int parse_sql_dtype( USHORT * length, USHORT * scale)
{
/**************************************
*
* p a r s e _ s q l _ d t y p e
*
**************************************
*
* Functional description
* Parse a SQL datatype clause.
*
**************************************/
KWWORDS keyword;
USHORT dtype, l;
keyword = QLI_token->tok_keyword;
PAR_token();
*scale = 0;
*length = 1;
switch (keyword) {
case KW_DATE:
*length = 8;
return dtype_timestamp;
case KW_CHAR:
dtype = dtype_text;
break;
case KW_VARCHAR:
dtype = dtype_varying;
break;
case KW_SMALLINT:
*length = sizeof(SSHORT);
return dtype_short;
case KW_INTEGER:
*length = sizeof(SLONG);
return dtype_long;
case KW_REAL:
case KW_FLOAT:
*length = sizeof(float);
dtype = dtype_real;
break;
case KW_LONG:
if (!MATCH(KW_FLOAT))
SYNTAX_ERROR(388); /* Msg388 "FLOAT" */
case KW_DOUBLE:
case KW_PRECISION:
*length = sizeof(double);
dtype = dtype_double;
break;
case KW_DECIMAL:
*length = sizeof(SLONG);
dtype = dtype_long;
break;
}
if (dtype == dtype_long || dtype == dtype_real || dtype == dtype_double) {
if (MATCH(KW_LEFT_PAREN)) {
l = (MATCH(KW_MINUS)) ? TRUE : FALSE;
*scale = parse_ordinal();
if (l)
*scale = -(*scale);
parse_matching_paren();
}
}
else if (dtype == dtype_text || dtype == dtype_varying) {
if (MATCH(KW_LEFT_PAREN)) {
l = parse_ordinal();
if (dtype == dtype_varying)
l += sizeof(SSHORT);
*length = l;
parse_matching_paren();
}
}
return dtype;
}
static QLI_FLD parse_sql_field(void)
{
/**************************************
*
* p a r s e _ s q l _ f i e l d
*
**************************************
*
* Functional description
* Parse a field description.
*
**************************************/
QLI_FLD field;
SYM name;
USHORT dtype, length, scale;
PAR_real();
dtype = length = scale = 0;
name = parse_symbol();
PAR_real();
switch (QLI_token->tok_keyword) {
case KW_DOUBLE:
MATCH(KW_PRECISION);
case KW_NUMERIC:
case KW_REAL:
case KW_DATE:
case KW_CHAR:
case KW_VARCHAR:
case KW_SMALLINT:
case KW_INTEGER:
case KW_FLOAT:
case KW_LONG:
case KW_DECIMAL:
dtype = parse_sql_dtype(&length, &scale);
break;
default:
SYNTAX_ERROR(179); /* Msg179 field definition clause */
break;
}
field = (QLI_FLD) ALLOCDV(type_fld, length);
field->fld_name = name;
field->fld_dtype = dtype;
field->fld_scale = scale;
field->fld_length = length;
if (MATCH(KW_NOT))
if (MATCH(KW_NULL)) {
field->fld_flags |= FLD_not_null;
}
else {
SYNTAX_ERROR(393); /* Msg393 NULL */
}
return field;
}
static SYN parse_sql_grant_revoke( USHORT type)
{
/**************************************
*
* p a r s e _ s q l _ g r a n t _ r e v o k e
*
**************************************
*
* Functional description
* Parse a SQL GRANT/REVOKE statement.
*
**************************************/
SYN node;
LLS stack;
USHORT privileges;
PAR_real_token();
node = SYNTAX_NODE((NOD_T) type, s_grant_count);
stack = NULL;
privileges = 0;
if (MATCH(KW_ALL)) {
MATCH(KW_PRIVILEGES);
privileges |= PRV_all;
}
else
while (TRUE) {
PAR_real();
if (MATCH(KW_SELECT)) {
privileges |= PRV_select;
continue;
}
else if (MATCH(KW_INSERT)) {
privileges |= PRV_insert;
continue;
}
else if (MATCH(KW_DELETE)) {
privileges |= PRV_delete;
continue;
}
else if (MATCH(KW_UPDATE)) {
privileges |= PRV_update;
if (MATCH(KW_COMMA))
continue;
if (KEYWORD(KW_ON))
break;
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(187); /* Msg187 left parenthesis */
do {
if (KEYWORD(KW_SELECT) || KEYWORD(KW_INSERT)
|| KEYWORD(KW_DELETE) || KEYWORD(KW_UPDATE))
break;
PAR_real();
LLS_PUSH(parse_name(), &stack);
} while (MATCH(KW_COMMA));
if (!MATCH(KW_RIGHT_PAREN))
SYNTAX_ERROR(191); /* Msg191 right parenthesis */
continue;
}
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_grant_fields] = make_list(stack);
PAR_real();
if (!MATCH(KW_ON))
SYNTAX_ERROR(397); /* Msg397 ON */
PAR_real();
if (!(node->syn_arg[s_grant_relation] = (SYN) parse_qualified_relation()))
SYNTAX_ERROR(170); /* Msg170 relation name */
if (type == (USHORT) nod_sql_grant) {
if (!MATCH(KW_TO))
SYNTAX_ERROR(404); /* Msg404 TO */
}
else {
if (!MATCH(KW_FROM))
SYNTAX_ERROR(403); /* Msg403 FROM */
}
stack = NULL;
while (TRUE) {
PAR_real();
LLS_PUSH(parse_name(), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_grant_users] = make_list(stack);
if (type == (USHORT) nod_sql_grant)
if (MATCH(KW_WITH)) {
PAR_real();
if (!MATCH(KW_GRANT))
SYNTAX_ERROR(401); /* Msg401 GRANT */
MATCH(KW_OPTION);
privileges |= PRV_grant_option;
}
node->syn_arg[s_grant_privileges] = INT_CAST privileges;
return node;
}
static SYN parse_sql_index_create( USHORT unique, USHORT descending)
{
/**************************************
*
* p a r s e _ s q l _ i n d e x _ c r e a t e
*
**************************************
*
* Functional description
* Parse the SQL CREATE INDEX statement.
*
**************************************/
LLS stack;
SYN node;
PAR_real();
node = SYNTAX_NODE(nod_def_index, s_dfi_count);
if (unique)
node->syn_flags |= s_dfi_flag_unique;
if (descending)
node->syn_flags |= s_dfi_flag_descending;
node->syn_arg[s_dfi_name] = (SYN) parse_symbol();
PAR_real();
if (!MATCH(KW_ON))
SYNTAX_ERROR(397); /* Msg397 ON */
if (!(node->syn_arg[s_dfi_relation] = (SYN) parse_qualified_relation()))
SYNTAX_ERROR(170); /* Msg170 relation name */
PAR_real();
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(185); /* Msg185 left parenthesis */
stack = NULL;
for (;;) {
LLS_PUSH(parse_name(), &stack);
if (MATCH(KW_RIGHT_PAREN))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(171); /* Msg171 comma between field definitions */
}
node->syn_arg[s_dfi_fields] = make_list(stack);
command_end();
return node;
}
static SYN parse_sql_joined_relation( SYN prior_context)
{
/**************************************
*
* p a r s e _ s q l _ j o i n e d _ r e l a t i o n
*
**************************************
*
* Functional description
* Parse a join relation clause.
*
**************************************/
SYN left;
if (MATCH(KW_LEFT_PAREN)) {
left = parse_sql_joined_relation(0);
parse_matching_paren();
}
else if (!(left = parse_sql_relation()))
return NULL;
return parse_sql_join_clause(left);
}
static SYN parse_sql_join_clause( SYN left)
{
/**************************************
*
* p a r s e _ s q l _ j o i n _ c l a u s e
*
**************************************
*
* Functional description
* Parse a join relation clause.
*
**************************************/
SYN right, node;
NOD_T join_type;
join_type = parse_join_type();
if (join_type == (NOD_T) 0)
return left;
if (!(right = parse_sql_joined_relation(left)))
SYNTAX_ERROR(490); /* Msg490 joined relation clause */
if (!MATCH(KW_ON))
SYNTAX_ERROR(492); /* Msg492 ON */
node = SYNTAX_NODE(nod_rse, (int) s_rse_count + 2 * 2);
node->syn_count = 2;
node->syn_arg[s_rse_count] = left;
node->syn_arg[s_rse_count + 2] = right;
node->syn_arg[s_rse_join_type] = (SYN) join_type;
node->syn_arg[s_rse_boolean] = parse_boolean(0);
return parse_sql_join_clause(node);
}
static SYN parse_sql_table_create(void)
{
/**************************************
*
* p a r s e _ s q l _ t a b l e _ c r e a t e
*
**************************************
*
* Functional description
* Parse the SQL CREATE TABLE statement.
*
**************************************/
QLI_REL relation;
QLI_FLD field, *ptr;
SYN node;
PAR_real();
node = SYNTAX_NODE(nod_sql_cr_table, 1);
relation = (QLI_REL) ALLOCD(type_rel);
node->syn_arg[0] = (SYN) relation;
relation->rel_database = parse_database();
relation->rel_symbol = parse_symbol();
ptr = &relation->rel_fields;
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(185); /* Msg185 left parenthesis */
PAR_real();
for (;;) {
*ptr = field = parse_sql_field();
ptr = &field->fld_next;
if (MATCH(KW_RIGHT_PAREN))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(171); /* Msg171 comma between field definitions */
}
command_end();
return node;
}
#ifdef NOT_USED_OR_REPLACED
static SYN parse_sql_view_create(void)
{
/**************************************
*
* p a r s e _ s q l _ v i e w _ c r e a t e
*
**************************************
*
* Functional description
* Parse the SQL CREATE VIEW statement.
*
**************************************/
SYN node;
LLS stack;
QLI_REL relation;
PAR_real();
sw_sql_view = TRUE;
node = SYNTAX_NODE(nod_sql_cr_view, s_crv_count);
stack = NULL;
relation = (QLI_REL) ALLOCD(type_rel);
node->syn_arg[s_crv_name] = (SYN) relation;
relation->rel_database = parse_database();
relation->rel_symbol = parse_symbol();
/* if field list is present parse it and create corresponding field blocks */
if (MATCH(KW_LEFT_PAREN)) {
for (;;) {
LLS_PUSH(parse_name(), &stack);
if (MATCH(KW_RIGHT_PAREN))
break;
if (!MATCH(KW_COMMA))
SYNTAX_ERROR(171); /* Msg171 comma between field definitions */
}
}
/* node->syn_arg [s_crv_fields] = make_list (stack); */
if (!MATCH(KW_AS))
SYNTAX_ERROR(394); /* Msg394 As */
if (!KEYWORD(KW_SELECT))
SYNTAX_ERROR(395); /* Msg395 Select */
node->syn_arg[s_crv_rse] = parse_select();
sw_sql_view = FALSE;
return node;
}
#endif
static SYN parse_sql_relation(void)
{
/**************************************
*
* p a r s e _ s q l _ r e l a t i o n
*
**************************************
*
* Functional description
* Parse a SQL relation clause.
*
**************************************/
SYN node;
node = SYNTAX_NODE(nod_relation, s_rel_count);
if (!(node->syn_arg[s_rel_relation] = (SYN) parse_qualified_relation()))
SYNTAX_ERROR(223); /* Msg223 relation name */
if (!QLI_token->tok_symbol)
node->syn_arg[s_rel_context] = (SYN) parse_symbol();
return node;
}
static SYN parse_sql_rse(void)
{
/**************************************
*
* p a r s e _ s q l _ r s e
*
**************************************
*
* Functional description
* Parse the trailing clauses of a SQL SELECT statement.
*
**************************************/
SYN node, *ptr;
LLS stack;
USHORT count;
stack = NULL;
count = 0;
PAR_real();
if (!MATCH(KW_FROM))
SYNTAX_ERROR(224); /* Msg224 FROM clause */
/* Parse FROM list of relations */
while (TRUE) {
count++;
LLS_PUSH(parse_sql_joined_relation(0), &stack);
if (!MATCH(KW_COMMA))
break;
}
/* Build a syntax node. Since SQL doesn't support OVER, only every
other slot will be used in the RSE. */
node = SYNTAX_NODE(nod_rse, (int) s_rse_count + 2 * count);
node->syn_count = count;
ptr = &node->syn_arg[(int) s_rse_count + 2 * count];
while (stack) {
--ptr;
*--ptr = (SYN) LLS_POP(&stack);
}
if (MATCH(KW_WITH))
node->syn_arg[s_rse_boolean] = parse_boolean(0);
if (MATCH(KW_GROUP)) {
if (sw_sql_view)
SYNTAX_ERROR(391); /* Msg391 No group by in view def */
PAR_real();
MATCH(KW_BY);
stack = NULL;
while (TRUE) {
LLS_PUSH(parse_udf_or_field(), &stack);
if (!MATCH(KW_COMMA))
break;
}
node->syn_arg[s_rse_group_by] = make_list(stack);
if (MATCH(KW_HAVING))
node->syn_arg[s_rse_having] = parse_boolean(0);
}
return node;
}
static SYN parse_sql_singleton_select(void)
{
/**************************************
*
* p a r s e _ s q l _ s i n g l e t o n _ s e l e c t
*
**************************************
*
* Functional description
* Finish parsing an SQL singleton select and
* turn it into a FIRST ... FROM --- not exactly
* kosher, but a start.
*
**************************************/
SYN node, value;
value = parse_primitive_value(0, 0);
PAR_real();
node = SYNTAX_NODE(nod_from, s_stt_count);
node->syn_arg[s_stt_value] = value;
node->syn_arg[s_stt_rse] = parse_sql_rse();
--sql_flag;
return node;
}
static SYN parse_sql_subquery(void)
{
/**************************************
*
* p a r s e _ s q l _ s u b q u e r y
*
**************************************
*
* Functional description
* Parse an sql subquery that should
* return a single value.
*
**************************************/
struct nod_types *types;
SYN node;
KWWORDS keyword;
if (sw_sql_view)
SYNTAX_ERROR(392); /* Msg392 No aggregates in view def */
PAR_token();
keyword = next_keyword();
++sql_flag;
for (types = statisticals;; types++)
if (types->nod_t_keyword == KW_none)
return parse_sql_singleton_select();
else if (types->nod_t_keyword == keyword)
break;
PAR_token();
node = SYNTAX_NODE(types->nod_t_node, s_stt_count);
MATCH(KW_LEFT_PAREN);
if (node->syn_type != nod_count || !MATCH(KW_ASTERISK)) {
if (MATCH(KW_DISTINCT))
node->syn_arg[s_prt_distinct] = INT_CAST TRUE;
node->syn_arg[s_stt_value] = parse_value(0, 0);
}
parse_matching_paren();
node->syn_arg[s_stt_rse] = parse_sql_rse();
--sql_flag;
return node;
}
static SYN parse_statement(void)
{
/**************************************
*
* p a r s e _ s t a t e m e n t
*
**************************************
*
* Functional description
* Parse a statement. (Set statement switch
* to true here as well as in PARQ_parse to
* avoid confusion with linked statements
* e.g. THEN conjuncts )
*
**************************************/
SYN node;
LLS stack;
PAR_real();
sw_statement = TRUE;
function_count = 0;
switch (next_keyword()) {
case KW_ABORT:
node = parse_abort();
break;
case KW_ACCEPT:
node = parse_accept();
break;
case KW_COMMIT:
node = parse_transaction(nod_commit_retaining);
break;
case KW_DECLARE:
node = parse_declare();
break;
case KW_DELETE:
MATCH(KW_DELETE);
node = parse_delete();
break;
case KW_ERASE:
node = parse_erase();
break;
case KW_FOR:
node = parse_for();
break;
case KW_IF:
node = parse_if();
break;
case KW_INSERT:
node = parse_insert();
break;
case KW_LIST:
node = parse_list_fields();
break;
case KW_MENU:
node = parse_menu();
break;
case KW_MODIFY:
node = parse_modify();
break;
case KW_PRINT:
node = parse_print();
break;
case KW_REPEAT:
node = parse_repeat();
break;
case KW_REPORT:
node = parse_report();
break;
case KW_SELECT:
node = parse_select();
break;
case KW_STORE:
node = parse_store();
break;
case KW_UPDATE:
node = parse_update();
break;
case KW_BEGIN:
stack = NULL;
PAR_token();
while (TRUE) {
PAR_real();
if (MATCH(KW_END))
break;
LLS_PUSH(parse_statement(), &stack);
MATCH(KW_SEMI);
}
node = make_list(stack);
break;
default:
node = parse_assignment();
}
check_end();
/* Check for the "THEN" connective. If found, make a list of statements. */
if (QLI_token->tok_type != tok_eol || (QLI_semi && !KEYWORD(KW_SEMI)))
MATCH(KW_SEMI);
if (!MATCH(KW_THEN))
return node;
stack = NULL;
LLS_PUSH(node, &stack);
LLS_PUSH(parse_statement(), &stack);
return make_list(stack);
}
static SYN parse_statistical(void)
{
/**************************************
*
* p a r s e _ s t a t i s t i c a l
*
**************************************
*
* Functional description
* Parse statistical expression.
*
**************************************/
struct nod_types *types;
SYN node;
KWWORDS keyword;
keyword = next_keyword();
PAR_token();
for (types = statisticals;; types++)
if (types->nod_t_keyword == keyword)
break;
/* Handle SQL statisticals a little differently */
if (sql_flag) {
node = SYNTAX_NODE(types->nod_t_sql_node, s_stt_count);
if (!MATCH(KW_LEFT_PAREN))
SYNTAX_ERROR(227); /* Msg227 left parenthesis */
if (node->syn_type != nod_agg_count || !MATCH(KW_ASTERISK)) {
if (MATCH(KW_DISTINCT))
node->syn_arg[s_prt_distinct] = INT_CAST TRUE;
node->syn_arg[s_stt_value] = parse_value(0, 0);
}
parse_matching_paren();
return node;
}
/* Handle GDML statisticals */
node = SYNTAX_NODE(types->nod_t_node, s_stt_count);
if (node->syn_type != nod_count)
node->syn_arg[s_stt_value] = parse_value(0, 0);
if (!MATCH(KW_OF)) {
if (sw_report) {
if (function_count > 0)
IBERROR(487); /* Msg487 Invalid argument for UDF */
node->syn_type = types->nod_t_rpt_node;
return node;
}
PAR_real();
if (!MATCH(KW_OF))
SYNTAX_ERROR(228); /* Msg 228 OF */
}
node->syn_arg[s_stt_rse] = parse_rse();
return node;
}
static SYN parse_store(void)
{
/**************************************
*
* p a r s e _ s t o r e
*
**************************************
*
* Functional description
* Parse a STORE statement.
*
**************************************/
SYN node;
PAR_token();
node = SYNTAX_NODE(nod_store, s_sto_count);
node->syn_arg[s_sto_relation] = parse_relation();
if (test_end())
return node;
MATCH(KW_USING);
#ifdef PYXIS
if (MATCH(KW_FORM))
IBERROR(484); /* FORMs not supported */
else
#endif
node->syn_arg[s_sto_statement] = parse_statement();
return node;
}
static TEXT *parse_string(void)
{
/**************************************
*
* p a r s e _ s t r i n g
*
**************************************
*
* Functional description
* Save the current token as a string, advance to the next
* token, and return a pointer to the string.
*
**************************************/
TEXT *string;
string = make_string(QLI_token->tok_string, QLI_token->tok_length);
PAR_token();
return string;
}
static SYM parse_symbol(void)
{
/**************************************
*
* p a r s e _ s y m b o l
*
**************************************
*
* Functional description
* Parse the next token as a context symbol.
*
**************************************/
SYM context;
USHORT l;
TEXT c, *p, *q;
l = QLI_token->tok_length;
context = (SYM) ALLOCDV(type_sym, l);
context->sym_type = SYM_context;
context->sym_length = l;
q = QLI_token->tok_string;
p = context->sym_string = context->sym_name;
if (l)
do {
c = *q++;
*p++ = UPPER(c);
} while (--l);
PAR_token();
return context;
}
static void parse_terminating_parens( USHORT * paren_count, USHORT * local_count)
{
/**************************************
*
* p a r s e _ t e r m i n a t i n g _ p a r e n s
*
**************************************
*
* Functional description
* Check for balancing parenthesis. If missing, barf.
*
**************************************/
if (*paren_count && paren_count == local_count)
do
parse_matching_paren();
while (--(*paren_count));
}
static SYN parse_transaction( NOD_T node_type)
{
/**************************************
*
* p a r s e _ t r a n s a c t i o n
*
**************************************
*
* Functional description
* Parse the FINISH, COMMIT, ROLLBACK,
* and PREPARE commands and the COMMIT statement.
*
**************************************/
SYN node;
LLS stack;
SYM symbol;
stack = NULL;
PAR_token();
if (!KEYWORD(KW_SEMI))
while (TRUE) {
for (symbol = QLI_token->tok_symbol; symbol;
symbol = symbol->sym_homonym)
if (symbol->sym_type == SYM_database) break;
if (!symbol)
SYNTAX_ERROR(229); /* Msg229 database name */
LLS_PUSH(symbol->sym_object, &stack);
PAR_token();
if (!MATCH(KW_COMMA))
break;
}
command_end();
node = make_list(stack);
node->syn_type = node_type;
return node;
}
static SYN parse_udf_or_field(void)
{
/**************************************
*
* p a r s e _ u d f _ o r _ f i e l d
*
**************************************
*
* Functional description
* Parse a function or field reference.
*
**************************************/
SYM symbol;
if ((symbol = QLI_token->tok_symbol) && symbol->sym_type == SYM_function)
return parse_function();
return parse_field_name(0);
}
static SYN parse_update(void)
{
/**************************************
*
* p a r s e _ u p d a t e
*
**************************************
*
* Functional description
* Parse a SQL UPDATE statement.
*
**************************************/
SYN node, rse;
LLS stack;
++sql_flag;
PAR_token();
node = SYNTAX_NODE(nod_modify, s_mod_count);
node->syn_arg[s_mod_rse] = rse =
SYNTAX_NODE(nod_rse, (int) s_rse_count + 2);
rse->syn_count = 1;
rse->syn_arg[s_rse_count] = parse_sql_relation();
if (!MATCH(KW_SET))
SYNTAX_ERROR(230); /* Msg230 SET */
/* Pick up assignments */
stack = NULL;
while (TRUE) {
LLS_PUSH(parse_assignment(), &stack);
if (!MATCH(KW_COMMA))
break;
}
/* Pick up boolean, if present */
if (MATCH(KW_WITH))
rse->syn_arg[s_rse_boolean] = parse_boolean(0);
node->syn_arg[s_mod_statement] = make_list(stack);
--sql_flag;
return node;
}
static SYN parse_value( USHORT * paren_count, USHORT * bool_flag)
{
/**************************************
*
* p a r s e _ v a l u e
*
**************************************
*
* Functional description
* Parse a general value expression. In practice, this means parse the
* lowest precedence operatr CONCATENATE.
*
**************************************/
SYN node, arg;
USHORT local_count, local_flag;
if (!paren_count) {
local_count = 0;
paren_count = &local_count;
}
if (!bool_flag) {
local_flag = FALSE;
bool_flag = &local_flag;
}
node = parse_add(paren_count, bool_flag);
while (TRUE) {
if (!MATCH(KW_BAR)) {
parse_terminating_parens(paren_count, &local_count);
return node;
}
arg = node;
node = SYNTAX_NODE(nod_concatenate, 2);
node->syn_arg[0] = arg;
node->syn_arg[1] = parse_add(paren_count, bool_flag);
}
}
static int potential_rse(void)
{
/**************************************
*
* p o t e n t i a l _ r s e
*
**************************************
*
* Functional description
* Test to see if the current token is likely (sic!) to be part of
* a record selection expression.
*
**************************************/
SYM symbol;
for (symbol = QLI_token->tok_symbol; symbol; symbol = symbol->sym_homonym)
if ((symbol->sym_type == SYM_keyword &&
symbol->sym_keyword == (int) KW_FIRST) ||
symbol->sym_type == SYM_relation ||
symbol->sym_type == SYM_database) return TRUE;
return FALSE;
}
static QLI_REL resolve_relation( SYM db_symbol, SYM relation_symbol)
{
/**************************************
*
* r e s o l v e _ r e l a t i o n
*
**************************************
*
* Functional description
* Given symbols for a database and a relation (either may be null),
* resolve the relation. If the relation can't be resolved, return
* NULL (don't error!).
*
**************************************/
QLI_REL relation;
SYM temp;
DBB dbb;
/* If we don't recognize the relation, punt. */
if (!relation_symbol)
return NULL;
/* If a database symbol is present, resolve the relation against the
the given database. */
if (db_symbol) { /* && db_symbol->sym_type == SYM_database ? */
for (; db_symbol; db_symbol = db_symbol->sym_homonym)
for (temp = relation_symbol; temp; temp = temp->sym_homonym)
if (temp->sym_type == SYM_relation) {
relation = (QLI_REL) temp->sym_object;
if (relation->rel_database == (DBB) db_symbol->sym_object)
return relation;
}
return NULL;
}
/* No database qualifier, so search all databases. */
for (dbb = QLI_databases; dbb; dbb = dbb->dbb_next)
for (temp = relation_symbol; temp; temp = temp->sym_homonym)
if (temp->sym_type == SYM_relation) {
relation = (QLI_REL) temp->sym_object;
if (relation->rel_database == dbb)
return relation;
}
return NULL;
}
static SYN syntax_node( NOD_T type, USHORT count)
{
/**************************************
*
* s y n t a x _ n o d e
*
**************************************
*
* Functional description
* Allocate and initialize a syntax node of given type.
*
**************************************/
SYN node;
node = (SYN) ALLOCDV(type_syn, count);
node->syn_type = type;
node->syn_count = count;
return node;
}
static int test_end(void)
{
/**************************************
*
* t e s t _ e n d
*
**************************************
*
* Functional description
* Test for end of a statement. In specific, test for one of
* THEN, ELSE, ON, or a semi-colon.
*
**************************************/
if (KEYWORD(KW_THEN) ||
KEYWORD(KW_ON) || KEYWORD(KW_ELSE) || KEYWORD(KW_SEMI)) return TRUE;
return FALSE;
}