2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-10-16 10:51:06 +02:00
|
|
|
* MODULE: par.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: BLR Parser
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Interbase Public
|
|
|
|
* License Version 1.0 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy
|
|
|
|
* of the License at http://www.Inprise.com/IPL.html
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an
|
|
|
|
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
|
|
|
|
* or implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Inprise Corporation
|
|
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
|
|
* Copyright (C) Inprise Corporation.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
2002-07-01 18:59:09 +02:00
|
|
|
*
|
|
|
|
* 27-May-2001 Claudio Valderrama: par_plan() no longer uppercases
|
|
|
|
* an index's name before doing a lookup of such index.
|
|
|
|
* 2001.07.28: Added parse code for blr_skip to support LIMIT.
|
2002-09-28 16:04:35 +02:00
|
|
|
* 2002.09.28 Dmitry Yemanov: Reworked internal_info stuff, enhanced
|
|
|
|
* exception handling in SPs/triggers,
|
|
|
|
* implemented ROWS_AFFECTED system variable
|
2002-10-29 21:20:44 +01:00
|
|
|
* 2002.10.21 Nickolay Samofatov: Added support for explicit pessimistic locks
|
2002-10-29 03:59:57 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
|
2002-10-29 17:27:47 +01:00
|
|
|
* 2002.10.29 Mike Nordell - Fixed breakage.
|
2002-10-29 21:20:44 +01:00
|
|
|
* 2002.10.29 Nickolay Samofatov: Added support for savepoints
|
2002-10-30 07:40:58 +01:00
|
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
2003-11-02 12:55:17 +01:00
|
|
|
* 2003.10.05 Dmitry Yemanov: Added support for explicit cursors in PSQL
|
2004-01-16 13:59:16 +01:00
|
|
|
* 2004.01.16 Vlad Horsun: Added support for default parameters
|
2007-04-13 03:37:44 +02:00
|
|
|
* Adriano dos Santos Fernandes
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2004-04-29 00:43:34 +02:00
|
|
|
#include <stdio.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include "../jrd/jrd.h"
|
2003-11-08 17:40:17 +01:00
|
|
|
#include "../jrd/ibase.h"
|
2006-07-19 16:50:33 +02:00
|
|
|
#include "../jrd/ini.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/val.h"
|
|
|
|
#include "../jrd/align.h"
|
|
|
|
#include "../jrd/exe.h"
|
|
|
|
#include "../jrd/lls.h"
|
2003-02-27 17:28:53 +01:00
|
|
|
#include "../jrd/rse.h" // for MAX_STREAMS
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#include "../jrd/scl.h"
|
|
|
|
#include "../jrd/all.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/blb.h"
|
|
|
|
#include "../jrd/intl.h"
|
|
|
|
#include "../jrd/met.h"
|
|
|
|
#include "../jrd/cmp_proto.h"
|
|
|
|
#include "../jrd/cvt_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/fun_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#include "../jrd/par_proto.h"
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../jrd/thd.h"
|
2003-12-31 06:36:12 +01:00
|
|
|
#include "../common/utils_proto.h"
|
2007-04-12 17:56:34 +02:00
|
|
|
#include "../jrd/SysFunction.h"
|
2003-12-31 06:36:12 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* blr type classes */
|
|
|
|
|
2004-05-03 23:43:56 +02:00
|
|
|
const int OTHER = 0;
|
|
|
|
const int STATEMENT = 1;
|
|
|
|
const int TYPE_BOOL = 2;
|
|
|
|
const int VALUE = 3;
|
|
|
|
const int TYPE_RSE = 4;
|
|
|
|
const int RELATION = 5;
|
|
|
|
const int ACCESS_TYPE = 6;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
using namespace Jrd;
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "gen/blrtable.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static const TEXT elements[][14] =
|
|
|
|
{ "", "statement", "boolean", "value", "RecordSelExpr", "TABLE" };
|
2003-07-12 12:24:47 +02:00
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "gen/codetext.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static void error(CompilerScratch*, ...);
|
2005-05-12 20:28:04 +02:00
|
|
|
static SSHORT find_proc_field(const jrd_prc*, const Firebird::MetaName&);
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_args(thread_db*, CompilerScratch*, USHORT);
|
|
|
|
static jrd_nod* par_cast(thread_db*, CompilerScratch*);
|
|
|
|
static PsqlException* par_condition(thread_db*, CompilerScratch*);
|
|
|
|
static PsqlException* par_conditions(thread_db*, CompilerScratch*);
|
|
|
|
static SSHORT par_context(CompilerScratch*, SSHORT *);
|
2005-05-12 20:28:04 +02:00
|
|
|
static void par_dependency(thread_db*, CompilerScratch*, SSHORT, SSHORT, const Firebird::MetaName&);
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_exec_proc(thread_db*, CompilerScratch*, SSHORT);
|
|
|
|
static jrd_nod* par_fetch(thread_db*, CompilerScratch*, jrd_nod*);
|
|
|
|
static jrd_nod* par_field(thread_db*, CompilerScratch*, SSHORT);
|
|
|
|
static jrd_nod* par_function(thread_db*, CompilerScratch*);
|
|
|
|
static jrd_nod* par_literal(thread_db*, CompilerScratch*);
|
|
|
|
static jrd_nod* par_map(thread_db*, CompilerScratch*, USHORT);
|
|
|
|
static jrd_nod* par_message(thread_db*, CompilerScratch*);
|
2006-09-03 03:09:23 +02:00
|
|
|
static jrd_nod* par_modify(thread_db*, CompilerScratch*, SSHORT);
|
2005-05-12 20:28:04 +02:00
|
|
|
static USHORT par_name(CompilerScratch*, Firebird::MetaName&);
|
2007-06-13 03:53:00 +02:00
|
|
|
static size_t par_name(CompilerScratch* csb, Firebird::string& name);
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_plan(thread_db*, CompilerScratch*);
|
|
|
|
static jrd_nod* par_procedure(thread_db*, CompilerScratch*, SSHORT);
|
2005-01-21 09:36:06 +01:00
|
|
|
static void par_procedure_parms(thread_db*, CompilerScratch*, jrd_prc*, jrd_nod**,
|
|
|
|
jrd_nod**, bool);
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_relation(thread_db*, CompilerScratch*, SSHORT, bool);
|
|
|
|
static jrd_nod* par_rse(thread_db*, CompilerScratch*, SSHORT);
|
|
|
|
static jrd_nod* par_sort(thread_db*, CompilerScratch*, bool);
|
2007-04-09 18:15:29 +02:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_stream(thread_db*, CompilerScratch*);
|
2007-04-09 18:15:29 +02:00
|
|
|
#endif
|
2007-04-12 17:56:34 +02:00
|
|
|
static jrd_nod* par_sys_function(thread_db*, CompilerScratch*);
|
2007-07-21 23:28:56 +02:00
|
|
|
static jrd_nod* par_union(thread_db*, CompilerScratch*, bool);
|
2004-03-28 11:10:30 +02:00
|
|
|
static USHORT par_word(CompilerScratch*);
|
|
|
|
static jrd_nod* parse(thread_db*, CompilerScratch*, USHORT, USHORT expected_optional = 0);
|
2004-04-10 02:25:22 +02:00
|
|
|
static void syntax_error(CompilerScratch*, const TEXT*);
|
2004-03-28 11:10:30 +02:00
|
|
|
static void warning(CompilerScratch*, ...);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
#define BLR_PEEK *(csb->csb_running)
|
|
|
|
#define BLR_BYTE *(csb->csb_running)++
|
|
|
|
#define BLR_PUSH (csb->csb_running)--
|
2001-05-23 15:26:42 +02:00
|
|
|
#define BLR_WORD par_word (csb)
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
jrd_nod* PAR_blr(thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation,
|
2003-11-07 09:06:35 +01:00
|
|
|
const UCHAR* blr,
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch* view_csb,
|
|
|
|
CompilerScratch** csb_ptr,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req** request_ptr,
|
2004-03-18 06:56:06 +01:00
|
|
|
const bool trigger,
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT flags)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ b l r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse blr, returning a compiler scratch block with the results.
|
|
|
|
* Caller must do pool handling.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
#ifdef CMP_DEBUG
|
|
|
|
cmp_trace("BLR code given for JRD parsing:");
|
2003-12-22 11:00:59 +01:00
|
|
|
// CVC: Couldn't find isc_trace_printer, so changed it to gds__trace_printer.
|
|
|
|
gds__print_blr(blr, gds__trace_printer, 0, 0);
|
2003-09-28 23:36:05 +02:00
|
|
|
#endif
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch* csb;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(csb_ptr && (csb = *csb_ptr))) {
|
2004-03-18 06:56:06 +01:00
|
|
|
size_t count = 5;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (view_csb)
|
2003-09-28 23:36:05 +02:00
|
|
|
count += view_csb->csb_rpt.getCapacity();
|
2004-08-16 14:28:43 +02:00
|
|
|
csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), count);
|
2001-05-23 15:26:42 +02:00
|
|
|
csb->csb_g_flags |= flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there is a request ptr, this is a trigger. Set up contexts 0 and 1 for
|
|
|
|
the target relation */
|
|
|
|
|
|
|
|
if (trigger) {
|
2004-03-18 06:56:06 +01:00
|
|
|
SSHORT stream = csb->nextStream();
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* t1 = CMP_csb_element(csb, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
t1->csb_flags |= csb_used | csb_active | csb_trigger;
|
|
|
|
t1->csb_relation = relation;
|
|
|
|
t1->csb_stream = (UCHAR) stream;
|
|
|
|
|
2004-03-07 10:48:56 +01:00
|
|
|
stream = csb->nextStream();
|
2003-09-28 23:36:05 +02:00
|
|
|
t1 = CMP_csb_element(csb, 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
t1->csb_flags |= csb_used | csb_active | csb_trigger;
|
|
|
|
t1->csb_relation = relation;
|
|
|
|
t1->csb_stream = (UCHAR) stream;
|
|
|
|
}
|
2003-09-28 23:36:05 +02:00
|
|
|
else if (relation) {
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* t1 = CMP_csb_element(csb, 0);
|
2004-03-07 10:48:56 +01:00
|
|
|
t1->csb_stream = csb->nextStream();
|
2003-09-28 23:36:05 +02:00
|
|
|
t1->csb_relation = relation;
|
|
|
|
t1->csb_flags = csb_used | csb_active;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
csb->csb_running = csb->csb_blr = blr;
|
|
|
|
|
|
|
|
if (view_csb) {
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::rpt_itr ptr = view_csb->csb_rpt.begin();
|
2003-09-28 23:36:05 +02:00
|
|
|
// AB: csb_n_stream replaced by view_csb->csb_rpt.getCount(), because there could
|
2003-07-24 00:42:09 +02:00
|
|
|
// be more then just csb_n_stream-numbers that hold data.
|
|
|
|
// Certainly csb_stream (see par_context where the context is retrieved)
|
2004-03-28 11:10:30 +02:00
|
|
|
const CompilerScratch::rpt_const_itr end = view_csb->csb_rpt.end();
|
2004-03-18 06:56:06 +01:00
|
|
|
for (SSHORT stream = 0; ptr != end; ++ptr, ++stream) {
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* t2 = CMP_csb_element(csb, stream);
|
2001-12-24 03:51:06 +01:00
|
|
|
t2->csb_relation = ptr->csb_relation;
|
|
|
|
t2->csb_stream = ptr->csb_stream;
|
2003-09-30 19:21:04 +02:00
|
|
|
t2->csb_flags = ptr->csb_flags & csb_used;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
csb->csb_n_stream = view_csb->csb_n_stream;
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT version = *csb->csb_running++;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
if (version != blr_version4 && version != blr_version5) {
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_metadata_corrupt,
|
|
|
|
isc_arg_gds, isc_wroblrver,
|
|
|
|
isc_arg_number, (SLONG) blr_version4,
|
|
|
|
isc_arg_number, (SLONG) version, 0);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (version == blr_version4)
|
|
|
|
csb->csb_g_flags |= csb_blr_version4;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = parse(tdbb, csb, OTHER);
|
2001-05-23 15:26:42 +02:00
|
|
|
csb->csb_node = node;
|
|
|
|
|
|
|
|
if (*csb->csb_running++ != (UCHAR) blr_eoc)
|
|
|
|
syntax_error(csb, "end_of_command");
|
|
|
|
|
|
|
|
if (request_ptr)
|
2003-09-28 23:36:05 +02:00
|
|
|
*request_ptr = CMP_make_request(tdbb, csb);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (csb_ptr)
|
|
|
|
*csb_ptr = csb;
|
|
|
|
else
|
2001-12-24 03:51:06 +01:00
|
|
|
delete csb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
USHORT PAR_desc(thread_db* tdbb, CompilerScratch* csb, DSC* desc, ItemInfo* itemInfo)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ d e s c
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a BLR descriptor. Return the alignment requirements
|
|
|
|
* of the datatype.
|
|
|
|
*
|
|
|
|
**************************************/
|
2007-01-17 02:19:01 +01:00
|
|
|
if (itemInfo)
|
|
|
|
{
|
|
|
|
itemInfo->nullable = true;
|
|
|
|
itemInfo->explicitCollation = false;
|
|
|
|
itemInfo->fullDomain = false;
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
desc->dsc_scale = 0;
|
|
|
|
desc->dsc_sub_type = 0;
|
2003-09-01 09:58:04 +02:00
|
|
|
desc->dsc_address = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
desc->dsc_flags = 0;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT dtype = BLR_BYTE;
|
2006-08-19 04:52:39 +02:00
|
|
|
|
|
|
|
switch (dtype)
|
|
|
|
{
|
2007-01-17 02:19:01 +01:00
|
|
|
case blr_not_nullable:
|
|
|
|
PAR_desc(tdbb, csb, desc, itemInfo);
|
|
|
|
if (itemInfo)
|
|
|
|
itemInfo->nullable = false;
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_text:
|
|
|
|
desc->dsc_dtype = dtype_text;
|
|
|
|
desc->dsc_flags |= DSC_no_subtype;
|
|
|
|
desc->dsc_length = BLR_WORD;
|
|
|
|
INTL_ASSIGN_TTYPE(desc, ttype_dynamic);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_cstring:
|
|
|
|
desc->dsc_dtype = dtype_cstring;
|
|
|
|
desc->dsc_flags |= DSC_no_subtype;
|
|
|
|
desc->dsc_length = BLR_WORD;
|
|
|
|
INTL_ASSIGN_TTYPE(desc, ttype_dynamic);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_varying:
|
|
|
|
desc->dsc_dtype = dtype_varying;
|
|
|
|
desc->dsc_flags |= DSC_no_subtype;
|
|
|
|
desc->dsc_length = BLR_WORD + sizeof(USHORT);
|
|
|
|
INTL_ASSIGN_TTYPE(desc, ttype_dynamic);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_text2:
|
|
|
|
desc->dsc_dtype = dtype_text;
|
|
|
|
INTL_ASSIGN_TTYPE(desc, BLR_WORD);
|
|
|
|
desc->dsc_length = BLR_WORD;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_cstring2:
|
|
|
|
desc->dsc_dtype = dtype_cstring;
|
|
|
|
INTL_ASSIGN_TTYPE(desc, BLR_WORD);
|
|
|
|
desc->dsc_length = BLR_WORD;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_varying2:
|
|
|
|
desc->dsc_dtype = dtype_varying;
|
|
|
|
INTL_ASSIGN_TTYPE(desc, BLR_WORD);
|
|
|
|
desc->dsc_length = BLR_WORD + sizeof(USHORT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_short:
|
|
|
|
desc->dsc_dtype = dtype_short;
|
|
|
|
desc->dsc_length = sizeof(SSHORT);
|
|
|
|
desc->dsc_scale = (int) BLR_BYTE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_long:
|
|
|
|
desc->dsc_dtype = dtype_long;
|
|
|
|
desc->dsc_length = sizeof(SLONG);
|
|
|
|
desc->dsc_scale = (int) BLR_BYTE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_int64:
|
|
|
|
desc->dsc_dtype = dtype_int64;
|
|
|
|
desc->dsc_length = sizeof(SINT64);
|
|
|
|
desc->dsc_scale = (int) BLR_BYTE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_quad:
|
|
|
|
desc->dsc_dtype = dtype_quad;
|
2003-11-11 13:19:20 +01:00
|
|
|
desc->dsc_length = sizeof(ISC_QUAD);
|
2001-05-23 15:26:42 +02:00
|
|
|
desc->dsc_scale = (int) BLR_BYTE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_float:
|
|
|
|
desc->dsc_dtype = dtype_real;
|
|
|
|
desc->dsc_length = sizeof(float);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_timestamp:
|
|
|
|
desc->dsc_dtype = dtype_timestamp;
|
2003-11-11 13:19:20 +01:00
|
|
|
desc->dsc_length = sizeof(ISC_QUAD);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_sql_date:
|
|
|
|
desc->dsc_dtype = dtype_sql_date;
|
|
|
|
desc->dsc_length = type_lengths[dtype_sql_date];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_sql_time:
|
|
|
|
desc->dsc_dtype = dtype_sql_time;
|
|
|
|
desc->dsc_length = type_lengths[dtype_sql_time];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_double:
|
|
|
|
#ifndef VMS
|
|
|
|
case blr_d_float:
|
|
|
|
#endif
|
|
|
|
desc->dsc_dtype = dtype_double;
|
|
|
|
desc->dsc_length = sizeof(double);
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
case blr_d_float:
|
|
|
|
desc->dsc_dtype = dtype_d_float;
|
|
|
|
desc->dsc_length = sizeof(double);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2005-11-24 09:43:23 +01:00
|
|
|
case blr_blob2:
|
2005-05-28 00:45:31 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
desc->dsc_dtype = dtype_blob;
|
2003-11-11 13:19:20 +01:00
|
|
|
desc->dsc_length = sizeof(ISC_QUAD);
|
2005-05-28 00:45:31 +02:00
|
|
|
desc->dsc_sub_type = BLR_WORD;
|
|
|
|
|
|
|
|
USHORT ttype = BLR_WORD;
|
|
|
|
desc->dsc_scale = ttype & 0xFF; // BLOB character set
|
|
|
|
desc->dsc_flags = ttype & 0xFF00; // BLOB collation
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
2005-05-28 00:45:31 +02:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
case blr_domain_name:
|
|
|
|
case blr_domain_name2:
|
2006-08-19 04:52:39 +02:00
|
|
|
{
|
2007-01-21 16:35:35 +01:00
|
|
|
bool fullDomain = (BLR_BYTE == blr_domain_full);
|
2006-08-19 04:52:39 +02:00
|
|
|
Firebird::MetaName* name = FB_NEW(csb->csb_pool) Firebird::MetaName(csb->csb_pool);
|
|
|
|
par_name(csb, *name);
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
FieldInfo fieldInfo;
|
|
|
|
bool exist = csb->csb_map_field_info.get(*name, fieldInfo);
|
2007-01-21 16:35:35 +01:00
|
|
|
MET_get_domain(tdbb, *name, desc, (exist ? NULL : &fieldInfo));
|
2006-08-19 04:52:39 +02:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
if (!exist)
|
|
|
|
csb->csb_map_field_info.put(*name, fieldInfo);
|
|
|
|
|
|
|
|
if (itemInfo)
|
|
|
|
{
|
|
|
|
itemInfo->field = *name;
|
|
|
|
|
|
|
|
if (fullDomain)
|
|
|
|
{
|
|
|
|
itemInfo->nullable = fieldInfo.nullable;
|
|
|
|
itemInfo->fullDomain = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
itemInfo->nullable = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dtype == blr_domain_name2)
|
2006-08-19 04:52:39 +02:00
|
|
|
{
|
|
|
|
USHORT ttype = BLR_WORD;
|
|
|
|
|
|
|
|
switch (desc->dsc_dtype)
|
|
|
|
{
|
|
|
|
case dtype_cstring:
|
|
|
|
case dtype_text:
|
|
|
|
case dtype_varying:
|
|
|
|
INTL_ASSIGN_TTYPE(desc, ttype);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_blob:
|
|
|
|
desc->dsc_scale = ttype & 0xFF; // BLOB character set
|
|
|
|
desc->dsc_flags = ttype & 0xFF00; // BLOB collation
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
error(csb, isc_collation_requires_text, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
jrd_nod* dep_node = PAR_make_node(tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*) name;
|
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_field;
|
|
|
|
csb->csb_dependencies.push(dep_node);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-11-24 09:43:23 +01:00
|
|
|
default:
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_datnotsup, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
if (itemInfo)
|
2006-10-02 00:09:05 +02:00
|
|
|
{
|
2006-10-02 04:37:45 +02:00
|
|
|
if (dtype == blr_cstring2 || dtype == blr_text2 || dtype == blr_varying2 ||
|
2007-01-17 02:19:01 +01:00
|
|
|
dtype == blr_blob2 || dtype == blr_domain_name2)
|
2006-10-02 04:37:45 +02:00
|
|
|
{
|
2007-01-17 02:19:01 +01:00
|
|
|
itemInfo->explicitCollation = true;
|
2006-10-02 04:37:45 +02:00
|
|
|
}
|
2006-10-02 00:09:05 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return type_alignments[desc->dsc_dtype];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
jrd_nod* PAR_gen_field(thread_db* tdbb, USHORT stream, USHORT id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ g e n _ f i e l d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Generate a field block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-08-16 14:28:43 +02:00
|
|
|
jrd_nod* node = FB_NEW_RPT(*tdbb->getDefaultPool(), e_fld_length) jrd_nod();
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_field;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_fld_id] = (jrd_nod*) (IPTR) id;
|
|
|
|
node->nod_arg[e_fld_stream] = (jrd_nod*) (IPTR) stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-31 20:03:51 +02:00
|
|
|
jrd_nod* PAR_make_field(thread_db* tdbb, CompilerScratch* csb,
|
|
|
|
USHORT context,
|
2005-05-12 20:28:04 +02:00
|
|
|
const Firebird::MetaName& base_field)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ m a k e _ f i e l d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Make up a field node in the permanent pool. This is used
|
|
|
|
* by MET_scan_relation to handle view fields.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT stream = csb->csb_rpt[context].csb_stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-07-01 18:59:09 +02:00
|
|
|
/* CVC: This is just another case of a custom function that isn't prepared
|
|
|
|
for quoted identifiers and that causes views with fields names like "z x"
|
|
|
|
to fail miserably. Since this function was truncating field names like "z x",
|
|
|
|
MET_lookup_field() call below failed and hence the function returned NULL
|
|
|
|
so only caller MET_scan_relation() did field->fld_source = 0;
|
|
|
|
This means a field without entry in rdb$fields. This is the origin of the
|
|
|
|
mysterious message "cannot access column z x in view VF" when selecting from
|
|
|
|
such view that has field "z x". This closes Firebird Bug #227758. */
|
2005-05-12 20:28:04 +02:00
|
|
|
// solved by using MetaName& as parameter - AP
|
2004-10-13 20:17:43 +02:00
|
|
|
jrd_rel* relation = csb->csb_rpt[stream].csb_relation;
|
|
|
|
jrd_prc* procedure = csb->csb_rpt[stream].csb_procedure;
|
2002-07-01 18:59:09 +02:00
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
const SSHORT id = procedure ? find_proc_field(procedure, base_field) :
|
|
|
|
MET_lookup_field (tdbb, csb->csb_rpt[stream].csb_relation, base_field, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (id < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* If rel_fields is NULL this means that the relation is
|
|
|
|
* in a temporary state (partially loaded). In this case
|
|
|
|
* there is nothing we can do but post an error and exit.
|
|
|
|
* Note: This will most likely happen if we have a large list
|
2005-12-02 08:35:34 +01:00
|
|
|
* of deferred work which can not complete because of some
|
2001-05-23 15:26:42 +02:00
|
|
|
* error, and while we are trying to commit, we find
|
|
|
|
* that we have a dependency on something later in the list.
|
|
|
|
* IF there were no error, then the dependency woyld have
|
|
|
|
* been resolved, because we would have fully loaded the
|
|
|
|
* relation, but if it can not be loaded, then we have this
|
|
|
|
* problem. The only thing that can be done to remedy this
|
2004-03-18 06:56:06 +01:00
|
|
|
* problem is to rollback. This will clear the DeferredWork list and
|
2001-05-23 15:26:42 +02:00
|
|
|
* allow the user to remedy the original error. Note: it would
|
|
|
|
* be incorrect for us (the server) to perform the rollback
|
|
|
|
* implicitly, because this is a task for the user to do, and
|
|
|
|
* should never be decided by the server. This fixes bug 10052 */
|
2004-10-13 20:17:43 +02:00
|
|
|
|
2005-12-02 08:35:34 +01:00
|
|
|
// CVC: The code for procedures now compiles correctly, but Vlad has
|
|
|
|
// pointed out that we don't have default for output fields, therefore
|
|
|
|
// the code is commented till better times.
|
2004-10-13 20:17:43 +02:00
|
|
|
jrd_fld* field = NULL;
|
2005-12-02 08:35:34 +01:00
|
|
|
/*
|
|
|
|
Parameter* param = NULL;
|
2004-10-13 20:17:43 +02:00
|
|
|
|
|
|
|
if (procedure)
|
|
|
|
{
|
2005-12-02 08:35:34 +01:00
|
|
|
param = (*procedure->prc_output_fields)[id];
|
2004-10-13 20:17:43 +02:00
|
|
|
}
|
2005-12-02 08:35:34 +01:00
|
|
|
else
|
|
|
|
*/
|
|
|
|
if (relation)
|
2004-10-13 20:17:43 +02:00
|
|
|
{
|
|
|
|
if (!relation->rel_fields) {
|
|
|
|
ERR_post(isc_depend_on_uncommitted_rel, 0);
|
|
|
|
}
|
2005-12-02 08:35:34 +01:00
|
|
|
field = (*relation->rel_fields)[id];
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2006-01-13 10:33:40 +01:00
|
|
|
if (csb->csb_g_flags & csb_get_dependencies) {
|
|
|
|
par_dependency(tdbb, csb, stream, id, base_field);
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* temp_node = PAR_gen_field(tdbb, stream, id);
|
2005-12-02 08:35:34 +01:00
|
|
|
/*
|
|
|
|
if (param)
|
|
|
|
{
|
|
|
|
if (param->prm_default_value) //&& param->prm_not_null)
|
2001-05-23 15:26:42 +02:00
|
|
|
temp_node->nod_arg[e_fld_default_value] =
|
2005-12-02 08:35:34 +01:00
|
|
|
param->prm_default_value;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
*/
|
|
|
|
if (field)
|
|
|
|
{
|
|
|
|
if (field->fld_default_value && field->fld_not_null)
|
|
|
|
temp_node->nod_arg[e_fld_default_value] = field->fld_default_value;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return temp_node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
jrd_nod* PAR_make_list(thread_db* tdbb, NodeStack& stack)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ m a k e _ l i s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Make a list node out of a stack.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* Count the number of nodes */
|
2004-04-18 16:22:27 +02:00
|
|
|
USHORT count = stack.getCount();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, count);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_list;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod** ptr = node->nod_arg + count;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-27 18:26:52 +02:00
|
|
|
while (stack.hasData())
|
|
|
|
{
|
2004-04-18 16:22:27 +02:00
|
|
|
*--ptr = stack.pop();
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
jrd_nod* PAR_make_node(thread_db* tdbb, int size)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ m a k e _ n o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Make a node element and pass it back.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-08-16 14:28:43 +02:00
|
|
|
jrd_nod* node = FB_NEW_RPT(*tdbb->getDefaultPool(), size) jrd_nod();
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = size;
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
CompilerScratch* PAR_parse(thread_db* tdbb, const UCHAR* blr, USHORT internal_flag,
|
|
|
|
USHORT dbginfo_length, const UCHAR* dbginfo)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ p a r s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse blr, returning a compiler scratch block with the results.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-08-16 14:28:43 +02:00
|
|
|
CompilerScratch* csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), 5);
|
2001-05-23 15:26:42 +02:00
|
|
|
csb->csb_running = csb->csb_blr = blr;
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT version = *csb->csb_running++;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (internal_flag)
|
|
|
|
csb->csb_g_flags |= csb_internal;
|
|
|
|
|
|
|
|
if (version != blr_version4 && version != blr_version5)
|
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_wroblrver,
|
|
|
|
isc_arg_number, (SLONG) blr_version4,
|
|
|
|
isc_arg_number, (SLONG) version, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (version == blr_version4)
|
|
|
|
{
|
|
|
|
csb->csb_g_flags |= csb_blr_version4;
|
|
|
|
}
|
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
if (dbginfo_length > 0)
|
|
|
|
DBG_parse_debug_info(dbginfo_length, dbginfo, csb->csb_dbg_info);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = parse(tdbb, csb, OTHER);
|
2001-05-23 15:26:42 +02:00
|
|
|
csb->csb_node = node;
|
|
|
|
|
|
|
|
if (*csb->csb_running++ != (UCHAR) blr_eoc)
|
|
|
|
{
|
|
|
|
syntax_error(csb, "end_of_command");
|
|
|
|
}
|
|
|
|
|
|
|
|
return csb;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
SLONG PAR_symbol_to_gdscode(const Firebird::MetaName& name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* P A R _ s y m b o l _ t o _ g d s c o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Symbolic ASCII names are used in blr for posting and handling
|
|
|
|
* exceptions. They are also used to identify error codes
|
|
|
|
* within system triggers in a database.
|
|
|
|
*
|
|
|
|
* Returns the gds error status code for the given symbolic
|
|
|
|
* name, or 0 if not found.
|
|
|
|
*
|
|
|
|
* Symbolic names may be null or space terminated.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-10-29 17:27:47 +01:00
|
|
|
for (int i = 0; codes[i].code_number; ++i) {
|
2005-05-12 20:28:04 +02:00
|
|
|
if (name == codes[i].code_string) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return codes[i].code_number;
|
2002-10-29 17:27:47 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static void error(CompilerScratch* csb, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We've got a blr error other than a syntax error. Handle it.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-10 08:49:16 +02:00
|
|
|
ISC_STATUS *p;
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT offset;
|
|
|
|
int type;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
/* Don't bother to pass tdbb for error handling */
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-24 01:28:06 +02:00
|
|
|
va_start(args, csb);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
csb->csb_running--;
|
|
|
|
offset = csb->csb_running - csb->csb_blr;
|
|
|
|
p = tdbb->tdbb_status_vector;
|
2003-11-08 17:40:17 +01:00
|
|
|
*p++ = isc_arg_gds;
|
|
|
|
*p++ = isc_invalid_blr;
|
|
|
|
*p++ = isc_arg_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = offset;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
*p++ = isc_arg_gds;
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick up remaining args */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
while ( (*p++ = type = va_arg(args, int)) )
|
2002-10-29 17:27:47 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (type) {
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_gds:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_string:
|
|
|
|
case isc_arg_interpreted:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_cstring:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, int);
|
|
|
|
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_number:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(FALSE);
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_vms:
|
|
|
|
case isc_arg_unix:
|
|
|
|
case isc_arg_win32:
|
2002-10-31 12:58:41 +01:00
|
|
|
*p++ = va_arg(args, int);
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-10-29 17:27:47 +01:00
|
|
|
}
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Give up whatever we were doing and return to the user. */
|
|
|
|
|
|
|
|
ERR_punt();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
static SSHORT find_proc_field(const jrd_prc* procedure, const Firebird::MetaName& name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f i n d _ p r o c _ f i e l d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Look for named field in procedure output fields.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<Parameter*>* list = procedure->prc_output_fields;
|
|
|
|
vec<Parameter*>::const_iterator ptr = list->begin();
|
2005-12-04 06:32:25 +01:00
|
|
|
for (const vec<Parameter*>::const_iterator end = list->end(); ptr < end; ++ptr)
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2005-12-02 08:35:34 +01:00
|
|
|
const Parameter* param = *ptr;
|
2005-05-12 20:28:04 +02:00
|
|
|
if (name == param->prm_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
return param->prm_number;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_args(thread_db* tdbb, CompilerScratch* csb, USHORT expected)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ a r g s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a counted argument list.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT count = BLR_BYTE;
|
|
|
|
jrd_nod* node = PAR_make_node(tdbb, count);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_list;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod** ptr = node->nod_arg;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
if (count) {
|
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
*ptr++ = parse(tdbb, csb, expected);
|
2003-12-22 11:00:59 +01:00
|
|
|
} while (--count);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_cast(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ c a s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a datatype cast
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_cast_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = count_table[blr_cast];
|
|
|
|
|
2004-08-16 14:28:43 +02:00
|
|
|
Format* format = Format::newFormat(*tdbb->getDefaultPool(), 1);
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_cast_fmt] = (jrd_nod*) format;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
dsc* desc = &format->fmt_desc[0];
|
2007-01-17 02:19:01 +01:00
|
|
|
ItemInfo itemInfo;
|
|
|
|
PAR_desc(tdbb, csb, desc, &itemInfo);
|
2001-05-23 15:26:42 +02:00
|
|
|
format->fmt_length = desc->dsc_length;
|
|
|
|
|
|
|
|
node->nod_arg[e_cast_source] = parse(tdbb, csb, VALUE);
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
if (itemInfo.isSpecial())
|
|
|
|
{
|
|
|
|
ItemInfo* p = FB_NEW(*tdbb->getDefaultPool()) ItemInfo(*tdbb->getDefaultPool(), itemInfo);
|
|
|
|
node->nod_arg[e_cast_iteminfo] = (jrd_nod*) p;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (itemInfo.explicitCollation)
|
2006-08-07 18:39:21 +02:00
|
|
|
{
|
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
|
|
|
dep_node->nod_arg [e_dep_object] = (jrd_nod*)(IPTR) INTL_TEXT_TYPE(*desc);
|
|
|
|
dep_node->nod_arg [e_dep_object_type] = (jrd_nod*)(IPTR) obj_collation;
|
|
|
|
csb->csb_dependencies.push(dep_node);
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static PsqlException* par_condition(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ c o n d i t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse an error conditions list.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* dep_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG code_number;
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* allocate a node to represent the conditions list */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT code_type = BLR_BYTE;
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* don't create PsqlException if blr_raise is used,
|
2002-09-28 16:04:35 +02:00
|
|
|
just return NULL */
|
|
|
|
if (code_type == blr_raise)
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2004-08-16 14:28:43 +02:00
|
|
|
PsqlException* exception_list = FB_NEW_RPT(*tdbb->getDefaultPool(), 1) PsqlException();
|
2001-05-23 15:26:42 +02:00
|
|
|
exception_list->xcp_count = 1;
|
2007-03-12 12:48:08 +01:00
|
|
|
xcp_repeat& item = exception_list->xcp_rpt[0];
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (code_type) {
|
|
|
|
case blr_sql_code:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_sql_code;
|
|
|
|
item.xcp_code = (SSHORT) BLR_WORD;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_gds_code:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_gds_code;
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2005-05-12 20:28:04 +02:00
|
|
|
name.lower7();
|
2001-05-23 15:26:42 +02:00
|
|
|
code_number = PAR_symbol_to_gdscode(name);
|
|
|
|
if (code_number)
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_code = code_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_codnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_exception:
|
2002-09-28 16:04:35 +02:00
|
|
|
case blr_exception_msg:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_xcp_code;
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2007-03-12 12:48:08 +01:00
|
|
|
if (!(item.xcp_code = MET_lookup_exception_number(tdbb, name)))
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_xcpnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
dep_node = PAR_make_node(tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
2007-03-12 12:48:08 +01:00
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*)(IPTR) item.xcp_code;
|
2004-05-20 23:58:15 +02:00
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_exception;
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(dep_node);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(FALSE);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return exception_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static PsqlException* par_conditions(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ c o n d i t i o n s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse an error conditions list.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* dep_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG code_number;
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* allocate a node to represent the conditions list */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT n = BLR_WORD;
|
2004-08-16 14:28:43 +02:00
|
|
|
PsqlException* exception_list = FB_NEW_RPT(*tdbb->getDefaultPool(), n) PsqlException();
|
2001-05-23 15:26:42 +02:00
|
|
|
exception_list->xcp_count = n;
|
2003-12-22 11:00:59 +01:00
|
|
|
for (int i = 0; i < n; i++) {
|
|
|
|
const USHORT code_type = BLR_BYTE;
|
2007-03-12 12:48:08 +01:00
|
|
|
xcp_repeat& item = exception_list->xcp_rpt[i];
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (code_type) {
|
|
|
|
case blr_sql_code:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_sql_code;
|
|
|
|
item.xcp_code = (SSHORT) BLR_WORD;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_gds_code:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_gds_code;
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2005-05-12 20:28:04 +02:00
|
|
|
name.lower7();
|
2001-05-23 15:26:42 +02:00
|
|
|
code_number = PAR_symbol_to_gdscode(name);
|
|
|
|
if (code_number)
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_code = code_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2007-03-12 12:48:08 +01:00
|
|
|
error(csb, isc_codnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_exception:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_xcp_code;
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2007-03-12 12:48:08 +01:00
|
|
|
if (!(item.xcp_code = MET_lookup_exception_number(tdbb, name)))
|
|
|
|
error(csb, isc_xcpnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
dep_node = PAR_make_node(tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
2007-03-13 03:33:10 +01:00
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*)(IPTR) item.xcp_code;
|
2004-05-20 23:58:15 +02:00
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_exception;
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(dep_node);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_default_code:
|
2007-03-12 12:48:08 +01:00
|
|
|
item.xcp_type = xcp_default;
|
|
|
|
item.xcp_code = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(FALSE);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return exception_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static SSHORT par_context(CompilerScratch* csb, SSHORT* context_ptr)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ c o n t e x t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Introduce a new context into the system. This involves
|
|
|
|
* assigning a stream and possibly extending the compile
|
|
|
|
* scratch block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2006-12-12 17:12:15 +01:00
|
|
|
const SSHORT context = (unsigned int) BLR_BYTE;
|
|
|
|
|
|
|
|
if (context_ptr)
|
|
|
|
*context_ptr = context;
|
|
|
|
|
2006-12-22 15:34:53 +01:00
|
|
|
CompilerScratch::csb_repeat* tail = CMP_csb_element(csb, context);
|
|
|
|
|
2006-12-12 17:12:15 +01:00
|
|
|
if (tail->csb_flags & csb_used) {
|
|
|
|
if (csb->csb_g_flags & csb_reuse_context) {
|
|
|
|
return tail->csb_stream;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
error(csb, isc_ctxinuse, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-03-07 10:48:56 +01:00
|
|
|
const SSHORT stream = csb->nextStream(false);
|
2006-09-14 11:40:23 +02:00
|
|
|
if (stream >= MAX_STREAMS)
|
2004-03-07 10:48:56 +01:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_too_many_contexts, 0);
|
2003-02-27 17:28:53 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
tail->csb_flags |= csb_used;
|
|
|
|
tail->csb_stream = (UCHAR) stream;
|
|
|
|
|
2006-12-22 15:34:53 +01:00
|
|
|
CMP_csb_element(csb, stream);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void par_dependency(thread_db* tdbb,
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch* csb,
|
2003-02-27 17:28:53 +01:00
|
|
|
SSHORT stream,
|
|
|
|
SSHORT id,
|
2005-05-12 20:28:04 +02:00
|
|
|
const Firebird::MetaName& field_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ d e p e n d e n c y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Register a field, relation, procedure or exception reference
|
|
|
|
* as a dependency.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_dep_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_dependency;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (csb->csb_rpt[stream].csb_relation) {
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_dep_object] =
|
2004-01-13 10:52:19 +01:00
|
|
|
(jrd_nod*) csb->csb_rpt[stream].csb_relation;
|
2005-05-26 06:00:45 +02:00
|
|
|
// How do I determine reliably this is a view?
|
|
|
|
// At this time, rel_view_rse is still null.
|
|
|
|
//if (is_view)
|
|
|
|
// node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_view;
|
|
|
|
//else
|
|
|
|
node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-09-28 23:36:05 +02:00
|
|
|
else if (csb->csb_rpt[stream].csb_procedure) {
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_dep_object] =
|
2004-01-13 10:52:19 +01:00
|
|
|
(jrd_nod*) csb->csb_rpt[stream].csb_procedure;
|
2004-05-20 23:58:15 +02:00
|
|
|
node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
if (field_name.length() > 0) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* field_node = PAR_make_node(tdbb, 1);
|
|
|
|
node->nod_arg[e_dep_field] = field_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
field_node->nod_type = nod_literal;
|
2004-03-31 20:03:51 +02:00
|
|
|
field_node->nod_arg[0] = (jrd_nod*)
|
2005-05-12 20:28:04 +02:00
|
|
|
stringDup(*tdbb->getDefaultPool(), field_name.c_str());
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (id >= 0) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* field_node = PAR_make_node(tdbb, 1);
|
|
|
|
node->nod_arg[e_dep_field] = field_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
field_node->nod_type = nod_field;
|
2004-01-21 08:18:30 +01:00
|
|
|
field_node->nod_arg[0] = (jrd_nod*) (IPTR) id;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(node);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_exec_proc(thread_db* tdbb, CompilerScratch* csb, SSHORT blr_operator)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ e x e c _ p r o c
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse an execute procedure reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_prc* procedure = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_exec_pid) {
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT pid = BLR_WORD;
|
2004-02-20 07:43:27 +01:00
|
|
|
if (!(procedure = MET_lookup_procedure_id(tdbb, pid, false, false, 0)))
|
2004-03-31 20:03:51 +02:00
|
|
|
name.printf("id %d", pid);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
par_name(csb, name);
|
2005-05-12 20:28:04 +02:00
|
|
|
procedure = MET_lookup_procedure(tdbb, name, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (!procedure)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_prcnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_esp_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_exec_proc;
|
|
|
|
node->nod_count = count_table[blr_exec_proc];
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_esp_procedure] = (jrd_nod*) procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
par_procedure_parms(tdbb, csb, procedure, &node->nod_arg[e_esp_in_msg],
|
2005-01-21 09:36:06 +01:00
|
|
|
&node->nod_arg[e_esp_inputs], true);
|
2001-05-23 15:26:42 +02:00
|
|
|
par_procedure_parms(tdbb, csb, procedure, &node->nod_arg[e_esp_out_msg],
|
2005-01-21 09:36:06 +01:00
|
|
|
&node->nod_arg[e_esp_outputs], false);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* dep_node = PAR_make_node(tdbb, e_dep_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
dep_node->nod_type = nod_dependency;
|
2004-01-13 10:52:19 +01:00
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*) procedure;
|
2004-05-20 23:58:15 +02:00
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(dep_node);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_fetch(thread_db* tdbb, CompilerScratch* csb, jrd_nod* for_node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ f e t c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a FETCH statement, and map it into
|
|
|
|
*
|
|
|
|
* FOR x IN relation WITH x.DBKEY EQ value ...
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* Fake RecordSelExpr */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for_node->nod_arg[e_for_re] = PAR_make_node(tdbb, 1 + rse_delta + 2);
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSelExpr* rse = (RecordSelExpr*) for_node->nod_arg[e_for_re];
|
2001-05-23 15:26:42 +02:00
|
|
|
rse->nod_type = nod_rse;
|
|
|
|
rse->nod_count = 0;
|
|
|
|
rse->rse_count = 1;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* relation = parse(tdbb, csb, RELATION);
|
|
|
|
rse->rse_relation[0] = relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Fake boolean */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = rse->rse_boolean = PAR_make_node(tdbb, 2);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_eql;
|
|
|
|
node->nod_flags = nod_comparison;
|
|
|
|
node->nod_arg[1] = parse(tdbb, csb, VALUE);
|
|
|
|
node->nod_arg[0] = PAR_make_node(tdbb, 1);
|
|
|
|
node = node->nod_arg[0];
|
|
|
|
node->nod_type = nod_dbkey;
|
|
|
|
node->nod_count = 0;
|
|
|
|
node->nod_arg[0] = relation->nod_arg[e_rel_stream];
|
|
|
|
|
|
|
|
/* Pick up statement */
|
|
|
|
|
|
|
|
for_node->nod_arg[e_for_statement] = parse(tdbb, csb, STATEMENT);
|
|
|
|
|
|
|
|
return for_node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_field(thread_db* tdbb, CompilerScratch* csb, SSHORT blr_operator)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ f i e l d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a field.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2007-04-11 11:28:50 +02:00
|
|
|
const USHORT context = (unsigned int) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
// check if this is a VALUE of domain's check constraint
|
2007-09-27 13:32:56 +02:00
|
|
|
if (!csb->csb_domain_validation.isEmpty() &&
|
|
|
|
(blr_operator == blr_fid || blr_operator == blr_field) &&
|
|
|
|
context == 0)
|
2007-01-17 02:19:01 +01:00
|
|
|
{
|
2007-09-27 13:32:56 +02:00
|
|
|
if (blr_operator == blr_fid) {
|
|
|
|
SSHORT id = BLR_WORD;
|
|
|
|
fb_assert(id == 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Firebird::MetaName name;
|
|
|
|
par_name(csb, name);
|
|
|
|
}
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_domval_length);
|
|
|
|
node->nod_type = nod_domain_validation;
|
|
|
|
node->nod_count = 0;
|
|
|
|
|
|
|
|
dsc* desc = (dsc*) (node->nod_arg + e_domval_desc);
|
2007-01-21 16:35:35 +01:00
|
|
|
MET_get_domain(tdbb, csb->csb_domain_validation, desc, NULL);
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
2006-05-23 09:52:38 +02:00
|
|
|
if (context >= csb->csb_rpt.getCount())/* ||
|
2003-09-28 23:36:05 +02:00
|
|
|
!(csb->csb_rpt[context].csb_flags & csb_used) )
|
2006-05-23 09:52:38 +02:00
|
|
|
|
|
|
|
dimitr: commented out to support system triggers implementing
|
|
|
|
WITH CHECK OPTION. They reference the relation stream (#2)
|
|
|
|
directly, without a DSQL context. It breaks the layering,
|
|
|
|
but we must support legacy BLR.
|
|
|
|
*/
|
2003-09-28 23:36:05 +02:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2003-09-28 23:36:05 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2004-03-18 06:56:06 +01:00
|
|
|
SSHORT id;
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT stream = csb->csb_rpt[context].csb_stream;
|
|
|
|
SSHORT flags = 0;
|
|
|
|
bool is_column = false;
|
2007-10-14 03:32:24 +02:00
|
|
|
|
|
|
|
if (blr_operator == blr_fid)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
id = BLR_WORD;
|
|
|
|
flags = nod_id;
|
2003-12-22 11:00:59 +01:00
|
|
|
is_column = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2007-10-14 03:32:24 +02:00
|
|
|
else if (blr_operator == blr_field)
|
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* tail = &csb->csb_rpt[stream];
|
2004-03-18 06:56:06 +01:00
|
|
|
const jrd_prc* procedure = tail->csb_procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* make sure procedure has been scanned before using it */
|
|
|
|
|
|
|
|
if (procedure && (!(procedure->prc_flags & PRC_scanned)
|
|
|
|
|| (procedure->prc_flags & PRC_being_scanned)
|
2003-12-22 11:00:59 +01:00
|
|
|
|| (procedure->prc_flags & PRC_being_altered)))
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
const jrd_prc* scan_proc = MET_procedure(tdbb, procedure->prc_id, false, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (scan_proc != procedure)
|
|
|
|
procedure = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (procedure) {
|
|
|
|
par_name(csb, name);
|
|
|
|
if ((id = find_proc_field(procedure, name)) == -1)
|
2003-09-28 23:36:05 +02:00
|
|
|
error(csb,
|
2007-12-04 11:05:17 +01:00
|
|
|
isc_fldnotdef2,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_string, ERR_cstring(name),
|
2004-03-20 15:57:40 +01:00
|
|
|
isc_arg_string, procedure->prc_name.c_str(), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2004-03-18 06:56:06 +01:00
|
|
|
jrd_rel* relation = tail->csb_relation;
|
|
|
|
if (!relation)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* make sure relation has been scanned before using it */
|
|
|
|
|
|
|
|
if (!(relation->rel_flags & REL_scanned) ||
|
|
|
|
(relation->rel_flags & REL_being_scanned))
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
MET_scan_relation(tdbb, relation);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
par_name(csb, name);
|
2004-03-31 20:03:51 +02:00
|
|
|
if ((id = MET_lookup_field(tdbb, relation, name.c_str(), 0)) < 0) {
|
2003-09-28 23:36:05 +02:00
|
|
|
if (csb->csb_g_flags & csb_validation) {
|
2001-05-23 15:26:42 +02:00
|
|
|
id = 0;
|
|
|
|
flags |= nod_id;
|
2003-12-22 11:00:59 +01:00
|
|
|
is_column = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2007-12-03 16:46:39 +01:00
|
|
|
if (tdbb->getAttachment()->att_flags & ATT_gbak_attachment)
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2007-04-13 15:21:23 +02:00
|
|
|
warning(csb, isc_fldnotdef, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2005-05-12 20:28:04 +02:00
|
|
|
else if (relation->rel_name.length() > 0)
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_fldnotdef, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
2005-05-12 20:28:04 +02:00
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check for dependencies -- if a field name was given,
|
|
|
|
use it because when restoring the database the field
|
|
|
|
id's may not be valid yet */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
if (csb->csb_g_flags & csb_get_dependencies) {
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_fid)
|
2004-03-31 20:03:51 +02:00
|
|
|
par_dependency(tdbb, csb, stream, id, "");
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
par_dependency(tdbb, csb, stream, id, name);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_gen_field(tdbb, stream, id);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_flags |= flags;
|
|
|
|
|
2007-11-03 18:18:50 +01:00
|
|
|
if (is_column)
|
|
|
|
{
|
|
|
|
jrd_rel* temp_rel = csb->csb_rpt[stream].csb_relation;
|
|
|
|
|
|
|
|
if (temp_rel)
|
|
|
|
{
|
|
|
|
jrd_fld* field;
|
|
|
|
|
|
|
|
if (id < (int) temp_rel->rel_fields->count() && (field = (*temp_rel->rel_fields)[id]))
|
|
|
|
{
|
|
|
|
if (field->fld_default_value && field->fld_not_null)
|
|
|
|
node->nod_arg[e_fld_default_value] = field->fld_default_value;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (temp_rel->rel_flags & REL_system)
|
|
|
|
{
|
|
|
|
node = PAR_make_node(tdbb, 0);
|
|
|
|
node->nod_type = nod_null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_function(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ f u n c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a function reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-22 11:00:59 +01:00
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT count = par_name(csb, name);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
UserFunction* function = FUN_lookup_function(name,
|
2007-12-03 16:46:39 +01:00
|
|
|
!(tdbb->getAttachment()->att_flags & ATT_gbak_attachment));
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!function) {
|
|
|
|
if (tdbb->tdbb_flags & TDBB_prc_being_dropped) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* anode = PAR_make_node(tdbb, e_fun_length);
|
|
|
|
anode->nod_count = 1;
|
|
|
|
anode->nod_arg[e_fun_function] = NULL;
|
|
|
|
anode->nod_arg[e_fun_args] = par_args(tdbb, csb, VALUE);
|
|
|
|
return anode;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_running -= count;
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_funnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
UserFunction* homonyms;
|
2003-12-22 11:00:59 +01:00
|
|
|
for (homonyms = function; homonyms; homonyms = homonyms->fun_homonym) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (homonyms->fun_entrypoint)
|
|
|
|
break;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!homonyms)
|
2007-12-03 16:46:39 +01:00
|
|
|
if (tdbb->getAttachment()->att_flags & ATT_gbak_attachment)
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
warning(csb, isc_funnotdef,
|
|
|
|
isc_arg_string, ERR_cstring(name),
|
|
|
|
isc_arg_interpreted,
|
2001-05-23 15:26:42 +02:00
|
|
|
"module name or entrypoint could not be found", 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_running -= count;
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_funnotdef,
|
|
|
|
isc_arg_string, ERR_cstring(name),
|
|
|
|
isc_arg_interpreted,
|
2001-05-23 15:26:42 +02:00
|
|
|
"module name or entrypoint could not be found", 0);
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_fun_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 1;
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_fun_function] = (jrd_nod*) function;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_fun_args] = par_args(tdbb, csb, VALUE);
|
|
|
|
|
2002-07-01 18:59:09 +02:00
|
|
|
/* CVC: I will track ufds only if a proc is not being dropped. */
|
2003-09-28 23:36:05 +02:00
|
|
|
if (csb->csb_g_flags & csb_get_dependencies) {
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
2002-07-01 18:59:09 +02:00
|
|
|
dep_node->nod_type = nod_dependency;
|
2004-01-13 10:52:19 +01:00
|
|
|
dep_node->nod_arg [e_dep_object] = (jrd_nod*) function;
|
2004-05-20 23:58:15 +02:00
|
|
|
dep_node->nod_arg [e_dep_object_type] = (jrd_nod*)(IPTR) obj_udf;
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(dep_node);
|
2002-07-01 18:59:09 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_literal(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ l i t e r a l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a literal value.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-10-16 10:51:06 +02:00
|
|
|
SSHORT scale;
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR dtype;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
DSC desc;
|
2006-08-19 04:52:39 +02:00
|
|
|
PAR_desc(tdbb, csb, &desc);
|
2007-01-29 04:00:09 +01:00
|
|
|
const int count = lit_delta +
|
2003-12-22 11:00:59 +01:00
|
|
|
(desc.dsc_length + sizeof(jrd_nod*) - 1) / sizeof(jrd_nod*);
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, count);
|
2004-03-18 06:56:06 +01:00
|
|
|
Literal* literal = (Literal*) node;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 0;
|
|
|
|
literal->lit_desc = desc;
|
2004-03-18 06:56:06 +01:00
|
|
|
UCHAR* p = reinterpret_cast<UCHAR*>(literal->lit_data);
|
|
|
|
literal->lit_desc.dsc_address = p;
|
2001-05-23 15:26:42 +02:00
|
|
|
literal->lit_desc.dsc_flags = 0;
|
2003-10-16 10:51:06 +02:00
|
|
|
const UCHAR* q = csb->csb_running;
|
2007-01-29 04:00:09 +01:00
|
|
|
USHORT l = desc.dsc_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (desc.dsc_dtype) {
|
|
|
|
case dtype_short:
|
|
|
|
l = 2;
|
2003-11-16 02:44:51 +01:00
|
|
|
*(SSHORT *) p = (SSHORT) gds__vax_integer(q, l);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_long:
|
|
|
|
case dtype_sql_date:
|
|
|
|
case dtype_sql_time:
|
|
|
|
l = 4;
|
2006-02-10 04:28:43 +01:00
|
|
|
*(SLONG *) p = gds__vax_integer(q, l);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_timestamp:
|
|
|
|
l = 8;
|
2006-02-10 04:28:43 +01:00
|
|
|
*(SLONG *) p = gds__vax_integer(q, 4);
|
2001-05-23 15:26:42 +02:00
|
|
|
p += 4;
|
|
|
|
q += 4;
|
2006-02-10 04:28:43 +01:00
|
|
|
*(SLONG *) p = gds__vax_integer(q, 4);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_int64:
|
|
|
|
l = sizeof(SINT64);
|
2006-02-10 04:28:43 +01:00
|
|
|
*(SINT64 *) p = isc_portable_integer(q, l);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_double:
|
|
|
|
/* the double literal could potentially be used for any
|
|
|
|
numeric literal - the value is passed as if it were a
|
|
|
|
text string. Convert the numeric string to its binary
|
|
|
|
value (int64, long or double as appropriate). */
|
|
|
|
l = BLR_WORD;
|
2003-09-28 23:36:05 +02:00
|
|
|
q = csb->csb_running;
|
2001-05-23 15:26:42 +02:00
|
|
|
dtype =
|
2003-11-03 18:14:45 +01:00
|
|
|
CVT_get_numeric(q, l, &scale, (double *) p, ERR_post);
|
2001-05-23 15:26:42 +02:00
|
|
|
literal->lit_desc.dsc_dtype = dtype;
|
|
|
|
if (dtype == dtype_double)
|
|
|
|
literal->lit_desc.dsc_length = sizeof(double);
|
|
|
|
else if (dtype == dtype_long) {
|
|
|
|
literal->lit_desc.dsc_length = sizeof(SLONG);
|
|
|
|
literal->lit_desc.dsc_scale = (SCHAR) scale;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
literal->lit_desc.dsc_length = sizeof(SINT64);
|
|
|
|
literal->lit_desc.dsc_scale = (SCHAR) scale;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
case dtype_text:
|
2006-02-23 07:52:25 +01:00
|
|
|
memcpy(p, q, l);
|
2006-02-23 06:08:26 +01:00
|
|
|
break;
|
2003-12-22 11:00:59 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
default:
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(FALSE);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_running += l;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_map(thread_db* tdbb, CompilerScratch* csb, USHORT stream)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ m a p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a MAP clause for a union or global aggregate expression.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
if (BLR_BYTE != blr_map)
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, "blr_map");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
SSHORT count = BLR_WORD;
|
2004-04-18 16:22:27 +02:00
|
|
|
NodeStack map;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (--count >= 0) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* assignment = PAR_make_node(tdbb, e_asgn_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
assignment->nod_type = nod_assignment;
|
|
|
|
assignment->nod_count = e_asgn_length;
|
|
|
|
assignment->nod_arg[e_asgn_to] =
|
|
|
|
PAR_gen_field(tdbb, stream, BLR_WORD);
|
|
|
|
assignment->nod_arg[e_asgn_from] = parse(tdbb, csb, VALUE);
|
2004-04-18 16:22:27 +02:00
|
|
|
map.push(assignment);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_list(tdbb, map);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_map;
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_message(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ m e s s a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a message declaration, including operator byte.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* Get message number, register it in the compiler scratch block, and
|
2001-05-23 15:26:42 +02:00
|
|
|
allocate a node to represent the message */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT n = (unsigned int) BLR_BYTE;
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* tail = CMP_csb_element(csb, n);
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_msg_length);
|
|
|
|
tail->csb_message = node;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 0;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_msg_number] = (jrd_nod*) (IPTR) n;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (n > csb->csb_msg_number)
|
|
|
|
csb->csb_msg_number = n;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Get the number of parameters in the message and prepare to fill
|
|
|
|
out the format block */
|
|
|
|
|
|
|
|
n = BLR_WORD;
|
2004-08-16 14:28:43 +02:00
|
|
|
Format* format = Format::newFormat(*tdbb->getDefaultPool(), n);
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_msg_format] = (jrd_nod*) format;
|
2003-12-22 11:00:59 +01:00
|
|
|
ULONG offset = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
Format::fmt_desc_iterator desc, end;
|
2007-01-17 02:19:01 +01:00
|
|
|
USHORT index = 0;
|
|
|
|
|
|
|
|
for (desc = format->fmt_desc.begin(), end = desc + n; desc < end; ++desc, ++index)
|
|
|
|
{
|
|
|
|
ItemInfo itemInfo;
|
|
|
|
const USHORT alignment = PAR_desc(tdbb, csb, &*desc, &itemInfo);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (alignment)
|
|
|
|
offset = FB_ALIGN(offset, alignment);
|
2004-01-21 08:18:30 +01:00
|
|
|
desc->dsc_address = (UCHAR *) (IPTR) offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
offset += desc->dsc_length;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2007-01-21 16:35:35 +01:00
|
|
|
// ASF: Odd indexes are the nullable flag.
|
|
|
|
// So we only check even indexes, which is the actual parameter.
|
2007-01-17 02:19:01 +01:00
|
|
|
if (itemInfo.isSpecial() && index % 2 == 0)
|
|
|
|
{
|
2007-01-20 15:18:18 +01:00
|
|
|
csb->csb_dbg_info.argInfoToName.get(
|
|
|
|
Firebird::ArgumentInfo(csb->csb_msg_number, index / 2), itemInfo.name);
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
csb->csb_map_item_info.put(
|
|
|
|
Item(nod_argument, csb->csb_msg_number, index), itemInfo);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (offset > MAX_FORMAT_SIZE)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_imp_exc, isc_arg_gds, isc_blktoobig, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
format->fmt_length = (USHORT) offset;
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-03 03:09:23 +02:00
|
|
|
static jrd_nod* par_modify(thread_db* tdbb, CompilerScratch* csb, SSHORT blr_operator)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ m o d i f y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a modify statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* Parse the original and new contexts */
|
|
|
|
|
2007-04-11 11:28:50 +02:00
|
|
|
USHORT context = (unsigned int) BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (context >= csb->csb_rpt.getCount() ||
|
|
|
|
!(csb->csb_rpt[context].csb_flags & csb_used))
|
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2003-09-28 23:36:05 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT org_stream = csb->csb_rpt[context].csb_stream;
|
2004-03-07 10:48:56 +01:00
|
|
|
const SSHORT new_stream = csb->nextStream(false);
|
2006-09-14 11:40:23 +02:00
|
|
|
if (new_stream >= MAX_STREAMS)
|
2004-03-07 10:48:56 +01:00
|
|
|
{
|
|
|
|
error(csb, isc_too_many_contexts, 0);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
context = (unsigned int) BLR_BYTE;
|
|
|
|
|
|
|
|
/* Make sure the compiler scratch block is big enough to hold
|
|
|
|
everything */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* tail = CMP_csb_element(csb, context);
|
2001-05-23 15:26:42 +02:00
|
|
|
tail->csb_stream = (UCHAR) new_stream;
|
2003-09-28 23:36:05 +02:00
|
|
|
tail->csb_flags |= csb_used;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
tail = CMP_csb_element(csb, new_stream);
|
2003-09-28 23:36:05 +02:00
|
|
|
tail->csb_relation = csb->csb_rpt[org_stream].csb_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Make the node and parse the sub-expression */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_mod_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 1;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_mod_org_stream] = (jrd_nod*) (IPTR) org_stream;
|
|
|
|
node->nod_arg[e_mod_new_stream] = (jrd_nod*) (IPTR) new_stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_mod_statement] = parse(tdbb, csb, STATEMENT);
|
|
|
|
|
2006-09-03 03:09:23 +02:00
|
|
|
if (blr_operator == blr_modify2)
|
2006-11-26 16:37:50 +01:00
|
|
|
{
|
|
|
|
node->nod_count = 2;
|
2006-09-03 03:09:23 +02:00
|
|
|
node->nod_arg[e_mod_statement2] = parse(tdbb, csb, STATEMENT);
|
2006-11-26 16:37:50 +01:00
|
|
|
}
|
2006-09-03 03:09:23 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
static USHORT par_name(CompilerScratch* csb, Firebird::MetaName& name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ n a m e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a counted string, returning count.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-05-12 20:28:04 +02:00
|
|
|
size_t l = BLR_BYTE;
|
2005-05-17 20:20:07 +02:00
|
|
|
|
2004-12-22 19:34:52 +01:00
|
|
|
// Check for overly long identifiers at BLR parse stage to prevent unwanted
|
|
|
|
// surprises in deeper layers of the engine.
|
2005-05-12 20:28:04 +02:00
|
|
|
if (l > MAX_SQL_IDENTIFIER_LEN) {
|
2005-05-17 20:20:07 +02:00
|
|
|
SqlIdentifier st;
|
|
|
|
char* s = st;
|
|
|
|
l = MAX_SQL_IDENTIFIER_LEN;
|
2005-05-12 20:28:04 +02:00
|
|
|
while (l--) {
|
2005-05-17 20:20:07 +02:00
|
|
|
*s++ = BLR_BYTE;
|
2005-05-12 20:28:04 +02:00
|
|
|
}
|
2005-05-17 20:20:07 +02:00
|
|
|
*s = 0;
|
|
|
|
ERR_post(isc_identifier_too_long, isc_arg_string, ERR_cstring(st), 0);
|
2005-05-12 20:28:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
char* s = name.getBuffer(l);
|
|
|
|
|
|
|
|
while (l--)
|
|
|
|
{
|
|
|
|
*s++ = BLR_BYTE;
|
2004-12-22 19:34:52 +01:00
|
|
|
}
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
return name.length();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-13 03:53:00 +02:00
|
|
|
static size_t par_name(CompilerScratch* csb, Firebird::string& name)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ n a m e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a counted string of virtually unlimited size
|
2007-06-14 14:28:27 +02:00
|
|
|
* (up to 64K, actually <= 255), returning count.
|
2007-06-13 03:53:00 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
size_t l = BLR_BYTE;
|
2007-06-14 14:28:27 +02:00
|
|
|
char* s = name.getBuffer(l);
|
2007-06-13 03:53:00 +02:00
|
|
|
|
|
|
|
while (l--)
|
|
|
|
{
|
|
|
|
*s++ = BLR_BYTE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return name.length();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_plan(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ p l a n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse an access plan expression.
|
|
|
|
* At this stage we are just generating the
|
|
|
|
* parse tree and checking contexts
|
|
|
|
* and indices.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT node_type = (USHORT) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* a join type indicates a cross of two or more streams */
|
|
|
|
|
|
|
|
if (node_type == blr_join || node_type == blr_merge) {
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT count = (USHORT) BLR_BYTE;
|
|
|
|
jrd_nod* plan = PAR_make_node(tdbb, count);
|
2001-05-23 15:26:42 +02:00
|
|
|
plan->nod_type = (NOD_T) (USHORT) blr_table[node_type];
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
for (jrd_nod** arg = plan->nod_arg; count--;)
|
2001-05-23 15:26:42 +02:00
|
|
|
*arg++ = par_plan(tdbb, csb);
|
|
|
|
return plan;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we have hit a stream; parse the context number and access type */
|
|
|
|
|
|
|
|
if (node_type == blr_retrieve) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* plan = PAR_make_node(tdbb, e_retrieve_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
plan->nod_type = (NOD_T) (USHORT) blr_table[node_type];
|
|
|
|
|
|
|
|
/* parse the relation name and context--the relation
|
|
|
|
itself is redundant except in the case of a view,
|
|
|
|
in which case the base relation (and alias) must be specified */
|
|
|
|
|
2007-04-11 11:28:50 +02:00
|
|
|
USHORT n = (unsigned int) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n != blr_relation && n != blr_relation2 &&
|
|
|
|
n != blr_rid && n != blr_rid2)
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
|
|
|
syntax_error(csb, elements[RELATION]);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* don't have par_relation() parse the context, because
|
|
|
|
this would add a new context; while this is a reference to
|
|
|
|
an existing context */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
jrd_nod* relation_node = par_relation(tdbb, csb, n, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
plan->nod_arg[e_retrieve_relation] = relation_node;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation = (jrd_rel*) relation_node->nod_arg[e_rel_relation];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
n = BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (n >= csb->csb_rpt.getCount() || !(csb->csb_rpt[n].csb_flags & csb_used))
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT stream = csb->csb_rpt[n].csb_stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
relation_node->nod_arg[e_rel_stream] = (jrd_nod*) (IPTR) stream;
|
|
|
|
relation_node->nod_arg[e_rel_context] = (jrd_nod*) (IPTR) n;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Access plan types (sequential is default) */
|
|
|
|
|
|
|
|
node_type = (USHORT) BLR_BYTE;
|
2003-09-14 17:37:05 +02:00
|
|
|
USHORT extra_count = 0;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* access_type = 0;
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2005-05-26 06:00:45 +02:00
|
|
|
TEXT* idx_name = 0;
|
2005-01-21 09:36:06 +01:00
|
|
|
|
2003-09-14 17:37:05 +02:00
|
|
|
switch (node_type) {
|
|
|
|
case blr_navigational:
|
|
|
|
{
|
2005-05-26 06:00:45 +02:00
|
|
|
access_type = plan->nod_arg[e_retrieve_access_type] =
|
|
|
|
PAR_make_node(tdbb, e_access_type_length);
|
|
|
|
access_type->nod_type = nod_navigational;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-05-26 06:00:45 +02:00
|
|
|
/* pick up the index name and look up the appropriate ids */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
par_name(csb, name);
|
2005-05-26 06:00:45 +02:00
|
|
|
/* CVC: We can't do this. Index names are identifiers.
|
|
|
|
for (p = name; *p; *p++)
|
|
|
|
*p = UPPER (*p);
|
|
|
|
*/
|
2003-12-22 11:00:59 +01:00
|
|
|
SLONG relation_id;
|
2003-09-14 17:37:05 +02:00
|
|
|
SSHORT idx_status;
|
2003-12-22 11:00:59 +01:00
|
|
|
const SLONG index_id =
|
|
|
|
MET_lookup_index_name(tdbb, name, &relation_id, &idx_status);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (idx_status == MET_object_unknown ||
|
2003-12-22 11:00:59 +01:00
|
|
|
idx_status == MET_object_inactive)
|
|
|
|
{
|
2007-12-03 16:46:39 +01:00
|
|
|
if (tdbb->getAttachment()->att_flags & ATT_gbak_attachment)
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
warning(csb, isc_indexname, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
2005-05-12 20:28:04 +02:00
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_indexname, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
2005-05-12 20:28:04 +02:00
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2003-09-14 17:37:05 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-05-26 06:00:45 +02:00
|
|
|
/* save both the relation id and the index id, since
|
2001-05-23 15:26:42 +02:00
|
|
|
the relation could be a base relation of a view;
|
|
|
|
save the index name also, for convenience */
|
|
|
|
|
2005-05-26 06:00:45 +02:00
|
|
|
access_type->nod_arg[e_access_type_relation] = (jrd_nod*) (IPTR) relation_id;
|
|
|
|
access_type->nod_arg[e_access_type_index] = (jrd_nod*) (IPTR) index_id;
|
2006-05-23 12:04:08 +02:00
|
|
|
idx_name = stringDup(*tdbb->getDefaultPool(), name.c_str());
|
2005-05-26 06:00:45 +02:00
|
|
|
access_type->nod_arg[e_access_type_index_name] = (jrd_nod*) idx_name;
|
|
|
|
|
|
|
|
if (csb->csb_g_flags & csb_get_dependencies)
|
|
|
|
{
|
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*) idx_name;
|
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_index;
|
|
|
|
csb->csb_dependencies.push(dep_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BLR_PEEK == blr_indices)
|
|
|
|
// dimitr: FALL INTO, if the plan item is ORDER ... INDEX (...)
|
|
|
|
extra_count = 3;
|
|
|
|
else
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2005-05-26 06:00:45 +02:00
|
|
|
case blr_indices:
|
|
|
|
{
|
|
|
|
if (extra_count)
|
|
|
|
BLR_BYTE; // skip blr_indices
|
|
|
|
USHORT count = (USHORT) BLR_BYTE;
|
|
|
|
jrd_nod* temp = plan->nod_arg[e_retrieve_access_type] =
|
|
|
|
PAR_make_node(tdbb, count * e_access_type_length + extra_count);
|
|
|
|
for (USHORT i = 0; i < extra_count; i++) {
|
|
|
|
temp->nod_arg[i] = access_type->nod_arg[i];
|
|
|
|
}
|
|
|
|
temp->nod_type = (extra_count) ? nod_navigational : nod_indices;
|
|
|
|
if (extra_count)
|
|
|
|
delete access_type;
|
|
|
|
access_type = temp;
|
|
|
|
|
|
|
|
/* pick up the index names and look up the appropriate ids */
|
|
|
|
|
|
|
|
for (jrd_nod** arg = access_type->nod_arg + extra_count; count--;) {
|
|
|
|
par_name(csb, name);
|
|
|
|
/* Nickolay Samofatov: We can't do this. Index names are identifiers.
|
|
|
|
for (p = name; *p; *p++)
|
|
|
|
*p = UPPER(*p);
|
|
|
|
*/
|
|
|
|
SLONG relation_id;
|
|
|
|
SSHORT idx_status;
|
|
|
|
const SLONG index_id =
|
|
|
|
MET_lookup_index_name(tdbb, name, &relation_id, &idx_status);
|
|
|
|
|
|
|
|
if (idx_status == MET_object_unknown ||
|
|
|
|
idx_status == MET_object_inactive)
|
|
|
|
{
|
2007-12-03 16:46:39 +01:00
|
|
|
if (tdbb->getAttachment()->att_flags & ATT_gbak_attachment)
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2005-05-26 06:00:45 +02:00
|
|
|
warning(csb, isc_indexname, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2005-05-26 06:00:45 +02:00
|
|
|
else
|
2007-12-05 01:03:15 +01:00
|
|
|
{
|
2005-05-26 06:00:45 +02:00
|
|
|
error(csb, isc_indexname, isc_arg_string,
|
|
|
|
ERR_cstring(name), isc_arg_string,
|
|
|
|
relation->rel_name.c_str(), 0);
|
2007-12-05 01:03:15 +01:00
|
|
|
}
|
2005-05-26 06:00:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* save both the relation id and the index id, since
|
|
|
|
the relation could be a base relation of a view;
|
|
|
|
save the index name also, for convenience */
|
|
|
|
|
|
|
|
*arg++ = (jrd_nod*) (IPTR) relation_id;
|
|
|
|
*arg++ = (jrd_nod*) (IPTR) index_id;
|
2006-05-23 12:04:08 +02:00
|
|
|
idx_name = stringDup(*tdbb->getDefaultPool(), name.c_str());
|
2005-05-26 06:00:45 +02:00
|
|
|
*arg++ = (jrd_nod*) idx_name;
|
|
|
|
|
|
|
|
if (csb->csb_g_flags & csb_get_dependencies)
|
|
|
|
{
|
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
|
|
|
dep_node->nod_arg[e_dep_object] = (jrd_nod*) idx_name;
|
|
|
|
dep_node->nod_arg[e_dep_object_type] = (jrd_nod*)(IPTR) obj_index;
|
|
|
|
csb->csb_dependencies.push(dep_node);
|
|
|
|
}
|
|
|
|
}
|
2003-09-14 17:37:05 +02:00
|
|
|
}
|
2005-05-26 06:00:45 +02:00
|
|
|
break;
|
2003-09-14 17:37:05 +02:00
|
|
|
case blr_sequential:
|
|
|
|
break;
|
|
|
|
default:
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, "access type");
|
2003-09-14 17:37:05 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return plan;
|
|
|
|
}
|
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, "plan item");
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL; /* Added to remove compiler warning */
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_procedure(thread_db* tdbb, CompilerScratch* csb, SSHORT blr_operator)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ p r o c e d u r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse an procedural view reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_prc* procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_procedure) {
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2004-02-20 07:43:27 +01:00
|
|
|
procedure = MET_lookup_procedure(tdbb, name, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT pid = BLR_WORD;
|
2004-02-20 07:43:27 +01:00
|
|
|
if (!(procedure = MET_lookup_procedure_id(tdbb, pid, false, false, 0)))
|
2004-03-31 20:03:51 +02:00
|
|
|
name.printf("id %d", pid);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (!procedure)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_prcnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2006-07-19 16:50:33 +02:00
|
|
|
if (procedure->prc_type == prc_executable)
|
2006-07-20 05:14:15 +02:00
|
|
|
{
|
2006-07-19 16:50:33 +02:00
|
|
|
error(csb, isc_illegal_prc_type,
|
|
|
|
isc_arg_string, ERR_string(procedure->prc_name), 0);
|
2006-07-20 05:14:15 +02:00
|
|
|
}
|
2006-07-19 16:50:33 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_prc_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_type = nod_procedure;
|
|
|
|
node->nod_count = count_table[blr_procedure];
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_prc_procedure] = (jrd_nod*) (IPTR) procedure->prc_id;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT stream = par_context(csb, 0);
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_prc_stream] = (jrd_nod*) (IPTR) stream;
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_rpt[stream].csb_procedure = procedure;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
par_procedure_parms(tdbb, csb, procedure, &node->nod_arg[e_prc_in_msg],
|
2005-01-21 09:36:06 +01:00
|
|
|
&node->nod_arg[e_prc_inputs], true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
if (csb->csb_g_flags & csb_get_dependencies)
|
2004-03-31 20:03:51 +02:00
|
|
|
par_dependency(tdbb, csb, stream, (SSHORT) - 1, "");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void par_procedure_parms(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch* csb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_prc* procedure,
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod** message_ptr,
|
2005-01-21 09:36:06 +01:00
|
|
|
jrd_nod** parameter_ptr, bool input_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ p r o c e d u r e _ p a r m s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse some procedure parameters.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-22 11:00:59 +01:00
|
|
|
bool mismatch = false;
|
2004-11-17 21:41:16 +01:00
|
|
|
SLONG count = BLR_WORD;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/** Check to see if the parameter count matches **/
|
2004-01-16 13:59:16 +01:00
|
|
|
if (input_flag ?
|
|
|
|
(count < (procedure->prc_inputs - procedure->prc_defaults) ||
|
|
|
|
(count > procedure->prc_inputs) ) :
|
|
|
|
(count != procedure->prc_outputs))
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/** They don't match...Hmmm...Its OK if we were dropping the procedure **/
|
2003-12-22 11:00:59 +01:00
|
|
|
if (!(tdbb->tdbb_flags & TDBB_prc_being_dropped)) {
|
2003-09-28 23:36:05 +02:00
|
|
|
error(csb,
|
2005-06-06 15:37:31 +02:00
|
|
|
input_flag ? isc_prcmismat : isc_prc_out_param_mismatch,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_string,
|
2004-03-20 15:57:40 +01:00
|
|
|
ERR_cstring(procedure->prc_name.c_str()),
|
2004-02-20 07:43:27 +01:00
|
|
|
0);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-12-22 11:00:59 +01:00
|
|
|
mismatch = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-16 13:59:16 +01:00
|
|
|
if (count || input_flag && procedure->prc_defaults) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/** We have a few parameters. Get on with creating the message block **/
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT n = ++csb->csb_msg_number;
|
|
|
|
if (n < 2)
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_msg_number = n = 2;
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch::csb_repeat* tail = CMP_csb_element(csb, n);
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* message = PAR_make_node(tdbb, e_msg_length);
|
|
|
|
tail->csb_message = message;
|
2001-05-23 15:26:42 +02:00
|
|
|
message->nod_type = nod_message;
|
|
|
|
message->nod_count = count_table[blr_message];
|
|
|
|
*message_ptr = message;
|
|
|
|
message->nod_count = 0;
|
2004-01-21 08:18:30 +01:00
|
|
|
message->nod_arg[e_msg_number] = (jrd_nod*)(IPTR) n;
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format =
|
2001-05-23 15:26:42 +02:00
|
|
|
input_flag ? procedure->prc_input_fmt : procedure->prc_output_fmt;
|
2003-02-04 14:31:02 +01:00
|
|
|
/* dimitr: procedure (with its parameter formats) is allocated out of
|
|
|
|
its own pool (prc_request->req_pool) and can be freed during
|
|
|
|
the cache cleanup (MET_clear_cache). Since the current
|
2004-08-16 14:28:43 +02:00
|
|
|
tdbb default pool is different from the procedure's one,
|
2003-02-04 14:31:02 +01:00
|
|
|
it's dangerous to copy a pointer from one request to another.
|
|
|
|
As an experiment, I've decided to copy format by value
|
2004-03-30 06:10:52 +02:00
|
|
|
instead of copying the reference. Since Format structure
|
2003-02-04 14:31:02 +01:00
|
|
|
doesn't contain any pointers, it should be safe to use a
|
|
|
|
default assignment operator which does a simple byte copy.
|
|
|
|
This change fixes one serious bug in the current codebase.
|
|
|
|
I think that this situation can (and probably should) be
|
|
|
|
handled by the metadata cache (via incrementing prc_use_count)
|
|
|
|
to avoid unexpected cache cleanups, but that area is out of my
|
|
|
|
knowledge. So this fix should be considered a temporary solution.
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
message->nod_arg[e_msg_format] = (jrd_nod*) format;
|
2003-02-04 14:31:02 +01:00
|
|
|
*/
|
2004-08-16 14:28:43 +02:00
|
|
|
Format* fmt_copy = Format::newFormat(*tdbb->getDefaultPool(), format->fmt_count);
|
2003-02-04 14:31:02 +01:00
|
|
|
*fmt_copy = *format;
|
2004-01-13 10:52:19 +01:00
|
|
|
message->nod_arg[e_msg_format] = (jrd_nod*) fmt_copy;
|
2003-02-04 14:31:02 +01:00
|
|
|
/* --- end of fix --- */
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!mismatch)
|
|
|
|
n = format->fmt_count / 2;
|
2003-12-22 11:00:59 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* There was a parameter mismatch hence can't depend upon the format's
|
|
|
|
fmt_count. Use count instead.
|
|
|
|
*/
|
|
|
|
n = count;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
|
|
|
jrd_nod* list = *parameter_ptr = PAR_make_node(tdbb, n);
|
2001-05-23 15:26:42 +02:00
|
|
|
list->nod_type = nod_list;
|
|
|
|
list->nod_count = n;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod** ptr = list->nod_arg;
|
|
|
|
USHORT asgn_arg1, asgn_arg2;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (input_flag) {
|
|
|
|
asgn_arg1 = e_asgn_from;
|
|
|
|
asgn_arg2 = e_asgn_to;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
asgn_arg1 = e_asgn_to;
|
|
|
|
asgn_arg2 = e_asgn_from;
|
|
|
|
}
|
2004-01-16 13:59:16 +01:00
|
|
|
for (USHORT i = 0; n; count--, n--) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* asgn = PAR_make_node(tdbb, e_asgn_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
*ptr++ = asgn;
|
|
|
|
asgn->nod_type = nod_assignment;
|
|
|
|
asgn->nod_count = count_table[blr_assignment];
|
2004-01-16 13:59:16 +01:00
|
|
|
|
|
|
|
// default value for parameter
|
2004-02-02 12:02:12 +01:00
|
|
|
if ((count <= 0) && input_flag) {
|
2005-12-02 08:35:34 +01:00
|
|
|
Parameter* parameter =
|
|
|
|
(*procedure->prc_input_fields)[procedure->prc_inputs - n];
|
|
|
|
asgn->nod_arg[asgn_arg1] = parameter->prm_default_value;
|
2004-01-16 13:59:16 +01:00
|
|
|
}
|
2004-01-28 08:50:41 +01:00
|
|
|
else {
|
2004-01-16 13:59:16 +01:00
|
|
|
asgn->nod_arg[asgn_arg1] = parse(tdbb, csb, VALUE);
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* prm = asgn->nod_arg[asgn_arg2] =
|
2001-05-23 15:26:42 +02:00
|
|
|
PAR_make_node(tdbb, e_arg_length);
|
|
|
|
prm->nod_type = nod_argument;
|
|
|
|
prm->nod_count = 1;
|
|
|
|
prm->nod_arg[e_arg_message] = message;
|
2004-01-21 08:18:30 +01:00
|
|
|
prm->nod_arg[e_arg_number] = (jrd_nod*)(IPTR) i++;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* prm_f = prm->nod_arg[e_arg_flag] =
|
2001-05-23 15:26:42 +02:00
|
|
|
PAR_make_node(tdbb, e_arg_length);
|
|
|
|
prm_f->nod_type = nod_argument;
|
|
|
|
prm_f->nod_count = 0;
|
|
|
|
prm_f->nod_arg[e_arg_message] = message;
|
2004-01-21 08:18:30 +01:00
|
|
|
prm_f->nod_arg[e_arg_number] = (jrd_nod*)(IPTR) i++;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ((input_flag ? procedure->prc_inputs : procedure->prc_outputs) &&
|
|
|
|
!mismatch)
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
|
|
|
error(csb,
|
2005-06-06 15:37:31 +02:00
|
|
|
input_flag ? isc_prcmismat : isc_prc_out_param_mismatch,
|
2003-12-22 11:00:59 +01:00
|
|
|
isc_arg_string,
|
2004-03-20 15:57:40 +01:00
|
|
|
ERR_cstring(procedure->prc_name.c_str()),
|
2004-02-20 07:43:27 +01:00
|
|
|
0);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
static jrd_nod* par_relation(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-03-28 11:10:30 +02:00
|
|
|
CompilerScratch* csb, SSHORT blr_operator, bool parse_context)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ r e l a t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a relation reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* Make a relation reference node */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_rel_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 0;
|
|
|
|
|
|
|
|
/* Find relation either by id or by name */
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_rel* relation = 0;
|
2007-06-13 03:53:00 +02:00
|
|
|
Firebird::string* alias_string = 0;
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_rid || blr_operator == blr_rid2) {
|
2003-12-22 11:00:59 +01:00
|
|
|
const SSHORT id = BLR_WORD;
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_rid2) {
|
2007-06-13 03:53:00 +02:00
|
|
|
alias_string = FB_NEW(csb->csb_pool) Firebird::string(csb->csb_pool);
|
2004-03-31 20:03:51 +02:00
|
|
|
par_name(csb, *alias_string);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
if (!(relation = MET_lookup_relation_id(tdbb, id, false))) {
|
2004-03-31 20:03:51 +02:00
|
|
|
name.printf("id %d", id);
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_relnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
else if (blr_operator == blr_relation || blr_operator == blr_relation2) {
|
2001-05-23 15:26:42 +02:00
|
|
|
par_name(csb, name);
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_relation2) {
|
2007-06-13 03:53:00 +02:00
|
|
|
alias_string = FB_NEW(csb->csb_pool) Firebird::string(csb->csb_pool);
|
2004-03-31 20:03:51 +02:00
|
|
|
par_name(csb, *alias_string);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2005-05-12 20:28:04 +02:00
|
|
|
if (!(relation = MET_lookup_relation(tdbb, name)))
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_relnotdef, isc_arg_string, ERR_cstring(name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* if an alias was passed, store with the relation */
|
|
|
|
|
|
|
|
if (alias_string)
|
2004-03-31 20:03:51 +02:00
|
|
|
{
|
|
|
|
node->nod_arg[e_rel_alias] =
|
2005-05-12 20:28:04 +02:00
|
|
|
(jrd_nod*) stringDup(*tdbb->getDefaultPool(), alias_string->c_str());
|
2004-03-31 20:03:51 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Scan the relation if it hasn't already been scanned for meta data */
|
|
|
|
|
|
|
|
if ((!(relation->rel_flags & REL_scanned)
|
|
|
|
|| (relation->rel_flags & REL_being_scanned))
|
|
|
|
&& ((relation->rel_flags & REL_force_scan)
|
2003-12-22 11:00:59 +01:00
|
|
|
|| !(csb->csb_g_flags & csb_internal)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
relation->rel_flags &= ~REL_force_scan;
|
|
|
|
MET_scan_relation(tdbb, relation);
|
|
|
|
}
|
|
|
|
else if (relation->rel_flags & REL_sys_triggers)
|
|
|
|
{
|
|
|
|
MET_parse_sys_trigger(tdbb, relation);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* generate a stream for the relation reference,
|
|
|
|
assuming it is a real reference */
|
|
|
|
|
|
|
|
if (parse_context) {
|
2003-12-22 11:00:59 +01:00
|
|
|
SSHORT context;
|
|
|
|
const SSHORT stream = par_context(csb, &context);
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(stream <= MAX_STREAMS);
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_rel_stream] = (jrd_nod*) (IPTR) stream;
|
|
|
|
node->nod_arg[e_rel_context] = (jrd_nod*) (IPTR) context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
csb->csb_rpt[stream].csb_relation = relation;
|
|
|
|
csb->csb_rpt[stream].csb_alias = alias_string;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
if (csb->csb_g_flags & csb_get_dependencies)
|
2005-01-21 09:36:06 +01:00
|
|
|
par_dependency(tdbb, csb, stream, (SSHORT) -1, "");
|
2004-03-31 20:03:51 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delete alias_string;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_rel_relation] = (jrd_nod*) relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_rse(thread_db* tdbb, CompilerScratch* csb, SSHORT rse_op)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ r s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a record selection expression.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
SSHORT count = (unsigned int) BLR_BYTE;
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSelExpr* rse = (RecordSelExpr*) PAR_make_node(tdbb, count + rse_delta + 2);
|
2001-05-23 15:26:42 +02:00
|
|
|
rse->nod_count = 0;
|
|
|
|
rse->rse_count = count;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod** ptr = rse->rse_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-08-15 02:33:15 +02:00
|
|
|
while (--count >= 0) {
|
|
|
|
// AB: Added TYPE_RSE for derived table support
|
|
|
|
*ptr++ = parse(tdbb, csb, RELATION, TYPE_RSE);
|
|
|
|
//*ptr++ = parse(tdbb, csb, RELATION);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
while (true) {
|
|
|
|
const UCHAR op = BLR_BYTE;
|
|
|
|
switch (op) {
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_boolean:
|
2004-05-03 23:43:56 +02:00
|
|
|
rse->rse_boolean = parse(tdbb, csb, TYPE_BOOL);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_first:
|
|
|
|
if (rse_op == blr_rs_stream)
|
2004-03-28 11:10:30 +02:00
|
|
|
syntax_error(csb, "RecordSelExpr stream clause");
|
2001-05-23 15:26:42 +02:00
|
|
|
rse->rse_first = parse(tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
|
2002-07-01 18:59:09 +02:00
|
|
|
case blr_skip:
|
|
|
|
if (rse_op == blr_rs_stream)
|
2004-03-28 11:10:30 +02:00
|
|
|
syntax_error (csb, "RecordSelExpr stream clause");
|
2002-07-01 18:59:09 +02:00
|
|
|
rse->rse_skip = parse (tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_sort:
|
|
|
|
if (rse_op == blr_rs_stream)
|
2004-03-28 11:10:30 +02:00
|
|
|
syntax_error(csb, "RecordSelExpr stream clause");
|
|
|
|
rse->rse_sorted = par_sort(tdbb, csb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_project:
|
|
|
|
if (rse_op == blr_rs_stream)
|
2004-03-28 11:10:30 +02:00
|
|
|
syntax_error(csb, "RecordSelExpr stream clause");
|
|
|
|
rse->rse_projection = par_sort(tdbb, csb, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_join_type:
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
|
|
|
const USHORT jointype = (USHORT) BLR_BYTE;
|
|
|
|
rse->rse_jointype = jointype;
|
|
|
|
if (jointype != blr_inner
|
|
|
|
&& jointype != blr_left && jointype != blr_right
|
|
|
|
&& jointype != blr_full)
|
|
|
|
{
|
|
|
|
syntax_error(csb, "join type clause");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case blr_plan:
|
|
|
|
rse->rse_plan = par_plan(tdbb, csb);
|
|
|
|
break;
|
2002-11-18 21:27:24 +01:00
|
|
|
|
|
|
|
case blr_writelock:
|
2004-02-20 07:43:27 +01:00
|
|
|
rse->rse_writelock = true;
|
2002-11-18 21:27:24 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
/* if a receive is seen here, then it is intended to be an asynchronous
|
2004-03-28 11:10:30 +02:00
|
|
|
receive which can happen at any time during the scope of the RecordSelExpr--
|
2001-05-23 15:26:42 +02:00
|
|
|
this is intended to be a more efficient mechanism for scrolling through
|
|
|
|
a record stream, to prevent having to send a message to the engine
|
|
|
|
for each record */
|
|
|
|
|
|
|
|
case blr_receive:
|
|
|
|
BLR_PUSH;
|
|
|
|
rse->rse_async_message = parse(tdbb, csb, STATEMENT);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (op == (UCHAR) blr_end) {
|
|
|
|
/* An outer join is only allowed when the stream count is 2
|
|
|
|
and a boolean expression has been supplied */
|
|
|
|
|
|
|
|
if (!rse->rse_jointype ||
|
|
|
|
(rse->rse_count == 2 && rse->rse_boolean))
|
2003-09-28 23:36:05 +02:00
|
|
|
{
|
|
|
|
// Convert right outer joins to left joins to avoid
|
|
|
|
// RIGHT JOIN handling at lower engine levels
|
|
|
|
if (rse->rse_jointype == blr_right) {
|
|
|
|
// Swap sub-streams
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* temp = rse->rse_relation[0];
|
2003-09-28 23:36:05 +02:00
|
|
|
rse->rse_relation[0] = rse->rse_relation[1];
|
|
|
|
rse->rse_relation[1] = temp;
|
|
|
|
|
|
|
|
rse->rse_jointype = blr_left;
|
|
|
|
}
|
2004-01-13 10:52:19 +01:00
|
|
|
return (jrd_nod*) rse;
|
2003-09-28 23:36:05 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, (TEXT*)((rse_op == blr_rs_stream) ?
|
2004-03-28 11:10:30 +02:00
|
|
|
"RecordSelExpr stream clause" :
|
2001-07-12 07:46:06 +02:00
|
|
|
"record selection expression clause"));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_sort(thread_db* tdbb, CompilerScratch* csb, bool flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ s o r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a sort clause (sans header byte). This is used for
|
|
|
|
* BLR_SORT, BLR_PROJECT, and BLR_GROUP.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
SSHORT count = (unsigned int) BLR_BYTE;
|
|
|
|
jrd_nod* clause = PAR_make_node(tdbb, count * 3);
|
2001-05-23 15:26:42 +02:00
|
|
|
clause->nod_type = nod_sort;
|
|
|
|
clause->nod_count = count;
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod** ptr = clause->nod_arg;
|
|
|
|
jrd_nod** ptr2 = ptr + count;
|
|
|
|
jrd_nod** ptr3 = ptr2 + count;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (--count >= 0) {
|
2002-09-10 20:34:00 +02:00
|
|
|
if (flag) {
|
|
|
|
UCHAR code = BLR_BYTE;
|
2003-12-21 03:34:34 +01:00
|
|
|
switch (code) {
|
|
|
|
case blr_nullsfirst:
|
2004-01-13 10:52:19 +01:00
|
|
|
*ptr3++ = (jrd_nod*) (IPTR) rse_nulls_first;
|
2002-09-10 20:34:00 +02:00
|
|
|
code = BLR_BYTE;
|
2003-12-21 03:34:34 +01:00
|
|
|
break;
|
|
|
|
case blr_nullslast:
|
2004-01-13 10:52:19 +01:00
|
|
|
*ptr3++ = (jrd_nod*) (IPTR) rse_nulls_last;
|
2003-12-21 03:34:34 +01:00
|
|
|
code = BLR_BYTE;
|
|
|
|
break;
|
|
|
|
default:
|
2004-01-13 10:52:19 +01:00
|
|
|
*ptr3++ = (jrd_nod*) (IPTR) rse_nulls_default;
|
2003-12-21 01:07:03 +01:00
|
|
|
}
|
2002-09-10 20:34:00 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
*ptr2++ =
|
2004-01-13 10:52:19 +01:00
|
|
|
(jrd_nod*) (IPTR) ((code == blr_descending) ? TRUE : FALSE);
|
2002-09-10 20:34:00 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
*ptr++ = parse(tdbb, csb, VALUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return clause;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-09 18:15:29 +02:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* par_stream(thread_db* tdbb, CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ s t r e a m
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a stream expression.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSelExpr* rse = (RecordSelExpr*) PAR_make_node(tdbb, 1 + rse_delta + 2);
|
2001-05-23 15:26:42 +02:00
|
|
|
rse->nod_count = 0;
|
|
|
|
rse->rse_count = 1;
|
|
|
|
rse->rse_relation[0] = parse(tdbb, csb, RELATION);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
while (true) {
|
|
|
|
const UCHAR op = BLR_BYTE;
|
|
|
|
switch (op) {
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_boolean:
|
2004-05-03 23:43:56 +02:00
|
|
|
rse->rse_boolean = parse(tdbb, csb, TYPE_BOOL);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (op == (UCHAR) blr_end)
|
2004-01-13 10:52:19 +01:00
|
|
|
return (jrd_nod*) rse;
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, "stream_clause");
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2007-04-09 18:15:29 +02:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2007-04-12 17:56:34 +02:00
|
|
|
static jrd_nod* par_sys_function(thread_db* tdbb, CompilerScratch* csb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ s y s _ f u n c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a system function reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
Firebird::MetaName name;
|
|
|
|
const USHORT count = par_name(csb, name);
|
|
|
|
|
2007-10-12 22:25:04 +02:00
|
|
|
const SysFunction* function = SysFunction::lookup(name);
|
2007-04-12 17:56:34 +02:00
|
|
|
|
|
|
|
if (!function)
|
|
|
|
{
|
|
|
|
csb->csb_running -= count;
|
|
|
|
error(csb, isc_funnotdef, isc_arg_string, ERR_cstring(name), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_sysfun_length);
|
2007-04-16 17:31:28 +02:00
|
|
|
node->nod_count = count_table[blr_sys_function];
|
2007-04-12 17:56:34 +02:00
|
|
|
node->nod_arg[e_sysfun_args] = par_args(tdbb, csb, VALUE);
|
|
|
|
node->nod_arg[e_sysfun_function] = (jrd_nod*) function;
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-21 23:28:56 +02:00
|
|
|
static jrd_nod* par_union(thread_db* tdbb, CompilerScratch* csb, bool recursive)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ u n i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a union reference.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* Make the node, parse the context number, get a stream assigned,
|
2004-03-28 11:10:30 +02:00
|
|
|
and get the number of sub-RecordSelExpr's. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node = PAR_make_node(tdbb, e_uni_length);
|
2007-07-21 23:28:56 +02:00
|
|
|
node->nod_count = 3;
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT stream = par_context(csb, 0);
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_uni_stream] = (jrd_nod*) (IPTR) stream;
|
2007-07-21 23:28:56 +02:00
|
|
|
|
|
|
|
// assign separate context for mapped record if union is recursive
|
|
|
|
USHORT map_stream = stream;
|
|
|
|
if (recursive)
|
|
|
|
{
|
|
|
|
node->nod_flags |= nod_recurse;
|
|
|
|
map_stream = par_context(csb, 0);
|
|
|
|
node->nod_arg[e_uni_map_stream] = (jrd_nod*) (IPTR) map_stream;
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
SSHORT count = (unsigned int) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* Pick up the sub-RecordSelExpr's and maps */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
NodeStack clauses;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (--count >= 0) {
|
2004-04-18 16:22:27 +02:00
|
|
|
clauses.push(parse(tdbb, csb, TYPE_RSE));
|
2007-07-21 23:28:56 +02:00
|
|
|
clauses.push(par_map(tdbb, csb, map_stream));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
node->nod_arg[e_uni_clauses] = PAR_make_list(tdbb, clauses);
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static USHORT par_word(CompilerScratch* csb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r _ w o r d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Pick up a BLR word.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-22 11:00:59 +01:00
|
|
|
const UCHAR low = BLR_BYTE;
|
|
|
|
const UCHAR high = BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return high * 256 + low;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static jrd_nod* parse(thread_db* tdbb, CompilerScratch* csb, USHORT expected,
|
|
|
|
USHORT expected_optional)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a r s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Parse a BLR expression.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2006-10-30 21:58:06 +01:00
|
|
|
const USHORT blr_offset = csb->csb_running - csb->csb_blr;
|
2004-02-02 12:02:12 +01:00
|
|
|
const SSHORT blr_operator = BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator < 0 || blr_operator >= FB_NELEM(type_table)) {
|
2004-12-09 07:53:48 +01:00
|
|
|
// NS: This error string is correct, please do not mangle it again and again.
|
|
|
|
// The whole error message is "BLR syntax error: expected %s at offset %d, encountered %d"
|
|
|
|
syntax_error(csb, "valid BLR code");
|
2002-07-01 18:59:09 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
const SSHORT sub_type = sub_type_table[blr_operator];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
if (expected && (expected != type_table[blr_operator])) {
|
2003-08-15 02:33:15 +02:00
|
|
|
if (expected_optional) {
|
2004-02-02 12:02:12 +01:00
|
|
|
if (expected_optional != type_table[blr_operator]) {
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, elements[expected]);
|
2003-08-15 02:33:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, elements[expected]);
|
2003-08-15 02:33:15 +02:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If there is a length given in the length table, pre-allocate
|
|
|
|
the node and set its count. This saves an enormous amount of
|
|
|
|
repetitive code. */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* node;
|
|
|
|
jrd_nod** arg;
|
2004-02-02 12:02:12 +01:00
|
|
|
USHORT n = length_table[blr_operator];
|
2003-12-22 11:00:59 +01:00
|
|
|
if (n) {
|
2001-05-23 15:26:42 +02:00
|
|
|
node = PAR_make_node(tdbb, n);
|
2004-02-02 12:02:12 +01:00
|
|
|
node->nod_count = count_table[blr_operator];
|
2001-05-23 15:26:42 +02:00
|
|
|
arg = node->nod_arg;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
node = NULL;
|
|
|
|
arg = NULL;
|
2004-11-07 11:47:20 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
bool set_type = true;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Dispatch on operator type. */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
switch (blr_operator) {
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_any:
|
|
|
|
case blr_unique:
|
|
|
|
case blr_ansi_any:
|
|
|
|
case blr_ansi_all:
|
|
|
|
case blr_exists:
|
|
|
|
node->nod_arg[e_any_rse] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Boring operators -- no special handling req'd */
|
|
|
|
|
|
|
|
case blr_value_if:
|
|
|
|
case blr_substring:
|
|
|
|
case blr_matching2:
|
|
|
|
case blr_ansi_like:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
2005-05-28 00:45:31 +02:00
|
|
|
case blr_trim:
|
|
|
|
{
|
|
|
|
node->nod_count = e_trim_count;
|
|
|
|
node->nod_arg[e_trim_specification] = (jrd_nod*)(U_IPTR) BLR_BYTE;
|
|
|
|
|
|
|
|
BYTE trimWhat = BLR_BYTE;
|
|
|
|
|
|
|
|
if (trimWhat == blr_trim_characters)
|
|
|
|
node->nod_arg[e_trim_characters] = parse(tdbb, csb, sub_type);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
node->nod_arg[e_trim_characters] = NULL;
|
|
|
|
--node->nod_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
node->nod_arg[e_trim_value] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_and:
|
|
|
|
case blr_or:
|
|
|
|
|
|
|
|
case blr_prot_mask:
|
|
|
|
case blr_containing:
|
|
|
|
case blr_matching:
|
|
|
|
case blr_like:
|
|
|
|
case blr_starting:
|
|
|
|
case blr_add:
|
|
|
|
case blr_subtract:
|
|
|
|
case blr_multiply:
|
|
|
|
case blr_divide:
|
|
|
|
case blr_concatenate:
|
|
|
|
|
|
|
|
case blr_assignment:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
/* Fall into ... */
|
|
|
|
|
|
|
|
case blr_handler:
|
|
|
|
case blr_loop:
|
|
|
|
|
|
|
|
case blr_lock_state:
|
|
|
|
case blr_upcase:
|
2005-05-28 00:45:31 +02:00
|
|
|
case blr_lowcase:
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_negate:
|
|
|
|
case blr_not:
|
|
|
|
case blr_missing:
|
|
|
|
case blr_agg_count2:
|
|
|
|
case blr_agg_max:
|
|
|
|
case blr_agg_min:
|
|
|
|
case blr_agg_total:
|
|
|
|
case blr_agg_average:
|
|
|
|
case blr_agg_count_distinct:
|
|
|
|
case blr_agg_total_distinct:
|
|
|
|
case blr_agg_average_distinct:
|
|
|
|
case blr_post:
|
2002-06-14 14:09:37 +02:00
|
|
|
case blr_internal_info:
|
2001-05-23 15:26:42 +02:00
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
2006-04-24 18:56:58 +02:00
|
|
|
case blr_agg_list:
|
|
|
|
case blr_agg_list_distinct:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
*arg++ = parse(tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
|
2003-03-01 20:19:23 +01:00
|
|
|
case blr_exec_sql:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_exec_into:
|
2006-05-26 07:19:48 +02:00
|
|
|
n = BLR_WORD + 2; // e_exec_into_count - 1
|
2003-03-01 20:19:23 +01:00
|
|
|
node = PAR_make_node(tdbb, n);
|
|
|
|
arg = node->nod_arg;
|
|
|
|
*arg++ = parse(tdbb, csb, VALUE);
|
|
|
|
if (BLR_BYTE) // singleton
|
|
|
|
*arg++ = 0;
|
|
|
|
else
|
|
|
|
*arg++ = parse(tdbb, csb, STATEMENT);
|
2003-12-22 11:00:59 +01:00
|
|
|
for (n = 2/*e_exec_into_list*/; n < node->nod_count; n++)
|
2003-03-01 20:19:23 +01:00
|
|
|
*arg++ = parse(tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
|
2003-01-15 13:08:59 +01:00
|
|
|
case blr_post_arg:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_null:
|
|
|
|
case blr_agg_count:
|
|
|
|
case blr_user_name:
|
2002-07-01 18:59:09 +02:00
|
|
|
case blr_current_role:
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_current_date:
|
|
|
|
case blr_start_savepoint:
|
|
|
|
case blr_end_savepoint:
|
|
|
|
break;
|
2003-06-10 15:40:19 +02:00
|
|
|
|
2005-08-24 11:16:19 +02:00
|
|
|
case blr_current_time:
|
|
|
|
node->nod_arg[0] = (jrd_nod*) (IPTR) DEFAULT_TIME_PRECISION;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_current_timestamp:
|
|
|
|
node->nod_arg[0] = (jrd_nod*) (IPTR) DEFAULT_TIMESTAMP_PRECISION;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_current_time2:
|
|
|
|
case blr_current_timestamp2:
|
|
|
|
n = BLR_BYTE;
|
2005-10-21 11:43:26 +02:00
|
|
|
if (n > MAX_TIME_PRECISION) {
|
|
|
|
ERR_post(isc_invalid_time_precision,
|
|
|
|
isc_arg_number, MAX_TIME_PRECISION, 0);
|
2005-08-24 11:16:19 +02:00
|
|
|
}
|
|
|
|
node->nod_arg[0] = (jrd_nod*) (IPTR) n;
|
|
|
|
break;
|
|
|
|
|
2002-10-29 21:20:44 +01:00
|
|
|
case blr_user_savepoint:
|
2004-03-28 11:10:30 +02:00
|
|
|
{
|
|
|
|
*arg++ = (jrd_nod*) (IPTR) BLR_BYTE;
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2004-03-28 11:10:30 +02:00
|
|
|
par_name(csb, name);
|
2006-05-23 12:04:08 +02:00
|
|
|
*arg++ = (jrd_nod*) stringDup(*tdbb->getDefaultPool(), name.c_str());
|
2004-03-28 11:10:30 +02:00
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case blr_store:
|
|
|
|
case blr_store2:
|
|
|
|
node->nod_arg[e_sto_relation] = parse(tdbb, csb, RELATION);
|
|
|
|
node->nod_arg[e_sto_statement] = parse(tdbb, csb, sub_type);
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_store2)
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_sto_statement2] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Comparison operators */
|
|
|
|
|
|
|
|
case blr_between:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
|
2004-10-14 21:09:19 +02:00
|
|
|
case blr_equiv:
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_eql:
|
|
|
|
case blr_neq:
|
|
|
|
case blr_geq:
|
|
|
|
case blr_gtr:
|
|
|
|
case blr_leq:
|
|
|
|
case blr_lss:
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
*arg++ = parse(tdbb, csb, sub_type);
|
|
|
|
node->nod_flags = nod_comparison;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_erase:
|
|
|
|
n = BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (n >= csb->csb_rpt.getCount() || !(csb->csb_rpt[n].csb_flags & csb_used))
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_erase_stream] =
|
2004-01-21 08:18:30 +01:00
|
|
|
(jrd_nod*) (IPTR) csb->csb_rpt[n].csb_stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2002-10-29 21:20:44 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_modify:
|
2006-09-03 03:09:23 +02:00
|
|
|
case blr_modify2:
|
|
|
|
node = par_modify(tdbb, csb, blr_operator);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_exec_proc:
|
|
|
|
case blr_exec_pid:
|
2004-02-02 12:02:12 +01:00
|
|
|
node = par_exec_proc(tdbb, csb, blr_operator);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_pid:
|
|
|
|
case blr_procedure:
|
2004-02-02 12:02:12 +01:00
|
|
|
node = par_procedure(tdbb, csb, blr_operator);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_function:
|
|
|
|
node = par_function(tdbb, csb);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_index:
|
|
|
|
node->nod_arg[0] = parse(tdbb, csb, sub_type);
|
|
|
|
node->nod_arg[1] = par_args(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_for:
|
|
|
|
if (BLR_PEEK == (UCHAR) blr_stall)
|
|
|
|
node->nod_arg[e_for_stall] = parse(tdbb, csb, STATEMENT);
|
|
|
|
|
|
|
|
if (BLR_PEEK == (UCHAR) blr_rse ||
|
2006-02-07 09:08:49 +01:00
|
|
|
BLR_PEEK == (UCHAR) blr_singular)
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_for_re] = parse(tdbb, csb, TYPE_RSE);
|
|
|
|
else
|
2004-02-02 12:02:12 +01:00
|
|
|
node->nod_arg[e_for_re] = par_rse(tdbb, csb, blr_operator);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_for_statement] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
case blr_dcl_cursor:
|
2006-03-07 18:52:33 +01:00
|
|
|
{
|
|
|
|
node->nod_arg[e_dcl_cursor_number] = (jrd_nod*) (IPTR) BLR_WORD;
|
|
|
|
node->nod_arg[e_dcl_cursor_rse] = parse(tdbb, csb, TYPE_RSE);
|
|
|
|
n = BLR_WORD;
|
|
|
|
jrd_nod* temp = PAR_make_node(tdbb, n);
|
|
|
|
temp->nod_type = nod_list;
|
|
|
|
for (jrd_nod** ptr = temp->nod_arg; n; n--) {
|
|
|
|
*ptr++ = parse(tdbb, csb, VALUE);
|
|
|
|
}
|
|
|
|
node->nod_arg[e_dcl_cursor_refs] = temp;
|
|
|
|
}
|
2003-11-02 12:55:17 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_cursor_stmt:
|
|
|
|
n = BLR_BYTE;
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_cursor_stmt_op] = (jrd_nod*) (IPTR) n;
|
|
|
|
node->nod_arg[e_cursor_stmt_number] = (jrd_nod*) (IPTR) BLR_WORD;
|
2003-11-02 12:55:17 +01:00
|
|
|
switch (n) {
|
|
|
|
case blr_cursor_open:
|
|
|
|
case blr_cursor_close:
|
|
|
|
break;
|
|
|
|
case blr_cursor_fetch:
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
if (BLR_PEEK == blr_seek)
|
|
|
|
node->nod_arg[e_cursor_stmt_seek] = parse(tdbb, csb, STATEMENT);
|
|
|
|
#endif
|
2006-12-12 17:12:15 +01:00
|
|
|
csb->csb_g_flags |= csb_reuse_context;
|
2003-11-02 12:55:17 +01:00
|
|
|
node->nod_arg[e_cursor_stmt_into] = parse(tdbb, csb, STATEMENT);
|
2006-12-12 17:12:15 +01:00
|
|
|
csb->csb_g_flags &= ~csb_reuse_context;
|
2003-11-02 12:55:17 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
syntax_error(csb, "cursor operation clause");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_rse:
|
|
|
|
case blr_rs_stream:
|
2004-02-02 12:02:12 +01:00
|
|
|
node = par_rse(tdbb, csb, blr_operator);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_singular:
|
|
|
|
node = parse(tdbb, csb, TYPE_RSE);
|
2004-03-28 11:10:30 +02:00
|
|
|
((RecordSelExpr*) node)->nod_flags |= rse_singular;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_relation:
|
|
|
|
case blr_rid:
|
|
|
|
case blr_relation2:
|
|
|
|
case blr_rid2:
|
2004-03-28 11:10:30 +02:00
|
|
|
node = par_relation(tdbb, csb, blr_operator, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_union:
|
2007-07-21 23:28:56 +02:00
|
|
|
node = par_union(tdbb, csb, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2006-08-01 22:37:58 +02:00
|
|
|
case blr_recurse:
|
2007-07-21 23:28:56 +02:00
|
|
|
node = par_union(tdbb, csb, true);
|
2006-08-01 22:37:58 +02:00
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_aggregate:
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_agg_stream] = (jrd_nod*) (IPTR) par_context(csb, 0);
|
2003-12-03 09:19:24 +01:00
|
|
|
fb_assert((int) (IPTR)node->nod_arg[e_agg_stream] <= MAX_STREAMS);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_agg_rse] = parse(tdbb, csb, TYPE_RSE);
|
|
|
|
node->nod_arg[e_agg_group] = parse(tdbb, csb, OTHER);
|
|
|
|
node->nod_arg[e_agg_map] =
|
2004-01-21 08:18:30 +01:00
|
|
|
par_map(tdbb, csb, (USHORT)(IPTR) node->nod_arg[e_agg_stream]);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_group_by:
|
2004-03-28 11:10:30 +02:00
|
|
|
node = par_sort(tdbb, csb, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return (node->nod_count) ? node : NULL;
|
|
|
|
|
|
|
|
case blr_field:
|
|
|
|
case blr_fid:
|
2004-02-02 12:02:12 +01:00
|
|
|
node = par_field(tdbb, csb, blr_operator);
|
2007-10-14 03:32:24 +02:00
|
|
|
if (node->nod_type == nod_domain_validation || node->nod_type == nod_null)
|
2007-01-17 02:19:01 +01:00
|
|
|
set_type = false; // to not change nod->nod_type to nod_field
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_gen_id:
|
|
|
|
case blr_set_generator:
|
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
par_name(csb, name);
|
2004-03-31 20:03:51 +02:00
|
|
|
const SLONG tmp = MET_lookup_generator(tdbb, name.c_str());
|
2003-12-22 11:00:59 +01:00
|
|
|
if (tmp < 0) {
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_gennotdef,
|
|
|
|
isc_arg_string, ERR_cstring(name), 0);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_gen_relation] = (jrd_nod*) (IPTR) tmp;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_gen_value] = parse(tdbb, csb, VALUE);
|
2002-07-01 18:59:09 +02:00
|
|
|
|
|
|
|
/* CVC: There're thousand ways to go wrong, but I don't see any value
|
|
|
|
in posting dependencies with set generator since it's DDL, so I will
|
|
|
|
track only gen_id() in both dialects. */
|
2004-02-02 12:02:12 +01:00
|
|
|
if ((blr_operator == blr_gen_id)
|
2003-12-22 11:00:59 +01:00
|
|
|
&& (csb->csb_g_flags & csb_get_dependencies))
|
|
|
|
{
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
2002-07-01 18:59:09 +02:00
|
|
|
dep_node->nod_type = nod_dependency;
|
2004-01-21 08:18:30 +01:00
|
|
|
dep_node->nod_arg [e_dep_object] = (jrd_nod*) (IPTR) tmp;
|
2004-05-20 23:58:15 +02:00
|
|
|
dep_node->nod_arg [e_dep_object_type] = (jrd_nod*)(IPTR) obj_generator;
|
2004-04-18 16:22:27 +02:00
|
|
|
csb->csb_dependencies.push(dep_node);
|
2002-07-01 18:59:09 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_record_version:
|
|
|
|
case blr_dbkey:
|
|
|
|
n = BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (n >= csb->csb_rpt.getCount() || !(csb->csb_rpt[n].csb_flags & csb_used))
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_ctxnotdef, 0);
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[0] = (jrd_nod*) (IPTR) csb->csb_rpt[n].csb_stream;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_fetch:
|
|
|
|
par_fetch(tdbb, csb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_send:
|
|
|
|
case blr_receive:
|
|
|
|
n = BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
node->nod_arg[e_send_message] = csb->csb_rpt[n].csb_message;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_send_statement] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_message:
|
|
|
|
node = par_message(tdbb, csb);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_literal:
|
|
|
|
node = par_literal(tdbb, csb);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_cast:
|
|
|
|
node = par_cast(tdbb, csb);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_extract:
|
2004-02-20 07:43:27 +01:00
|
|
|
// This forced conversion looks strange, but extract_part fits in a byte
|
|
|
|
node->nod_arg[e_extract_part] = (jrd_nod*)(U_IPTR) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_extract_value] = parse(tdbb, csb, sub_type);
|
|
|
|
node->nod_count = e_extract_count;
|
|
|
|
break;
|
|
|
|
|
2005-06-06 20:14:10 +02:00
|
|
|
case blr_strlen:
|
2005-05-28 00:45:31 +02:00
|
|
|
// This forced conversion looks strange, but length_type fits in a byte
|
2005-06-06 20:14:10 +02:00
|
|
|
node->nod_arg[e_strlen_type] = (jrd_nod*)(U_IPTR) BLR_BYTE;
|
|
|
|
node->nod_arg[e_strlen_value] = parse(tdbb, csb, sub_type);
|
|
|
|
node->nod_count = e_strlen_count;
|
2005-05-28 00:45:31 +02:00
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case blr_dcl_variable:
|
|
|
|
{
|
2006-10-02 00:09:05 +02:00
|
|
|
dsc* desc = (dsc*) (node->nod_arg + e_dcl_desc);
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
ItemInfo itemInfo;
|
2006-10-02 00:09:05 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
n = BLR_WORD;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_dcl_id] = (jrd_nod*) (IPTR) n;
|
2007-01-17 02:19:01 +01:00
|
|
|
PAR_desc(tdbb, csb, desc, &itemInfo);
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<jrd_nod*>* vector = csb->csb_variables =
|
|
|
|
vec<jrd_nod*>::newVector(*tdbb->getDefaultPool(), csb->csb_variables, n + 1);
|
|
|
|
(*vector)[n] = node;
|
2006-10-02 00:09:05 +02:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
if (itemInfo.isSpecial())
|
|
|
|
{
|
2007-01-20 15:18:18 +01:00
|
|
|
csb->csb_dbg_info.varIndexToName.get(n, itemInfo.name);
|
2007-01-17 02:19:01 +01:00
|
|
|
csb->csb_map_item_info.put(Item(nod_variable, n), itemInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (itemInfo.explicitCollation)
|
2006-10-02 00:09:05 +02:00
|
|
|
{
|
|
|
|
jrd_nod* dep_node = PAR_make_node (tdbb, e_dep_length);
|
|
|
|
dep_node->nod_type = nod_dependency;
|
|
|
|
dep_node->nod_arg [e_dep_object] = (jrd_nod*)(IPTR) INTL_TEXT_TYPE(*desc);
|
|
|
|
dep_node->nod_arg [e_dep_object_type] = (jrd_nod*)(IPTR) obj_collation;
|
|
|
|
csb->csb_dependencies.push(dep_node);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_variable:
|
|
|
|
{
|
|
|
|
n = BLR_WORD;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_var_id] = (jrd_nod*) (IPTR) n;
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<jrd_nod*>* vector = csb->csb_variables;
|
2003-12-22 11:00:59 +01:00
|
|
|
if (!vector || n >= vector->count() ||
|
2005-12-02 08:35:34 +01:00
|
|
|
!(node->nod_arg[e_var_variable] = (*vector)[n]))
|
2003-09-28 23:36:05 +02:00
|
|
|
{
|
|
|
|
syntax_error(csb, "variable identifier");
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_parameter:
|
|
|
|
case blr_parameter2:
|
|
|
|
case blr_parameter3:
|
|
|
|
{
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* message;
|
2001-05-23 15:26:42 +02:00
|
|
|
n = (USHORT) BLR_BYTE;
|
2003-09-28 23:36:05 +02:00
|
|
|
if (n >= csb->csb_rpt.getCount() ||
|
|
|
|
!(message = csb->csb_rpt[n].csb_message))
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
|
|
|
error(csb, isc_badmsgnum, 0);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_arg_message] = message;
|
|
|
|
n = BLR_WORD;
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_arg_number] = (jrd_nod*) (IPTR) n;
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) message->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n >= format->fmt_count)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_badparnum, 0);
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator != blr_parameter) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* temp = PAR_make_node(tdbb, e_arg_length);
|
|
|
|
node->nod_arg[e_arg_flag] = temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 1;
|
|
|
|
temp->nod_count = 0;
|
|
|
|
temp->nod_type = nod_argument;
|
|
|
|
temp->nod_arg[e_arg_message] = message;
|
|
|
|
n = BLR_WORD;
|
2004-01-21 08:18:30 +01:00
|
|
|
temp->nod_arg[e_arg_number] = (jrd_nod*) (IPTR) n;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n >= format->fmt_count)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_badparnum, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_parameter3) {
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_nod* temp = PAR_make_node(tdbb, e_arg_length);
|
|
|
|
node->nod_arg[e_arg_indicator] = temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_count = 2;
|
|
|
|
temp->nod_count = 0;
|
|
|
|
temp->nod_type = nod_argument;
|
|
|
|
temp->nod_arg[e_arg_message] = message;
|
|
|
|
n = BLR_WORD;
|
2004-01-21 08:18:30 +01:00
|
|
|
temp->nod_arg[e_arg_number] = (jrd_nod*) (IPTR) n;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n >= format->fmt_count)
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_badparnum, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_stall:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_select:
|
|
|
|
case blr_begin:
|
|
|
|
{
|
2004-04-18 16:22:27 +02:00
|
|
|
NodeStack stack;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (BLR_PEEK != (UCHAR) blr_end) {
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_select && BLR_PEEK != blr_receive)
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, "blr_receive");
|
2004-04-18 16:22:27 +02:00
|
|
|
stack.push(parse(tdbb, csb, sub_type));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
BLR_BYTE; // skip blr_end
|
2001-05-23 15:26:42 +02:00
|
|
|
node = PAR_make_list(tdbb, stack);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_block:
|
|
|
|
{
|
2004-04-18 16:22:27 +02:00
|
|
|
NodeStack stack;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
node->nod_arg[e_blk_action] = parse(tdbb, csb, sub_type);
|
|
|
|
while (BLR_PEEK != (UCHAR) blr_end)
|
2004-04-18 16:22:27 +02:00
|
|
|
{
|
|
|
|
stack.push(parse(tdbb, csb, sub_type));
|
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
BLR_BYTE; // skip blr_end
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_blk_handlers] = PAR_make_list(tdbb, stack);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_error_handler:
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_err_conditions] = (jrd_nod*) par_conditions(tdbb, csb);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_err_action] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_abort:
|
2002-09-28 16:04:35 +02:00
|
|
|
{
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool flag = (BLR_PEEK == blr_exception_msg);
|
2004-01-13 10:52:19 +01:00
|
|
|
node->nod_arg[e_xcp_desc] = (jrd_nod*) par_condition(tdbb, csb);
|
2003-12-22 11:00:59 +01:00
|
|
|
if (flag)
|
|
|
|
{
|
|
|
|
node->nod_arg[e_xcp_msg] = parse(tdbb, csb, sub_type);
|
|
|
|
}
|
|
|
|
break;
|
2002-09-28 16:04:35 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case blr_if:
|
2004-05-03 23:43:56 +02:00
|
|
|
node->nod_arg[e_if_boolean] = parse(tdbb, csb, TYPE_BOOL);
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_if_true] = parse(tdbb, csb, sub_type);
|
|
|
|
if (BLR_PEEK == (UCHAR) blr_end) {
|
|
|
|
node->nod_count = 2;
|
2004-02-20 07:43:27 +01:00
|
|
|
BLR_BYTE; // skip blr_end
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
node->nod_arg[e_if_false] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_label:
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[e_lbl_label] = (jrd_nod*) (IPTR) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_lbl_statement] = parse(tdbb, csb, sub_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_leave:
|
2004-01-21 08:18:30 +01:00
|
|
|
node->nod_arg[0] = (jrd_nod*) (IPTR) BLR_BYTE;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case blr_maximum:
|
|
|
|
case blr_minimum:
|
|
|
|
case blr_count:
|
|
|
|
/* count2
|
|
|
|
case blr_count2:
|
|
|
|
*/
|
|
|
|
case blr_average:
|
|
|
|
case blr_total:
|
|
|
|
case blr_from:
|
|
|
|
case blr_via:
|
2006-02-07 09:08:49 +01:00
|
|
|
node->nod_arg[e_stat_rse] = parse(tdbb, csb, TYPE_RSE);
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator != blr_count)
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_stat_value] = parse(tdbb, csb, VALUE);
|
2004-02-02 12:02:12 +01:00
|
|
|
if (blr_operator == blr_via)
|
2001-05-23 15:26:42 +02:00
|
|
|
node->nod_arg[e_stat_default] = parse(tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
case blr_seek:
|
|
|
|
node->nod_arg[e_seek_direction] = parse(tdbb, csb, VALUE);
|
|
|
|
node->nod_arg[e_seek_offset] = parse(tdbb, csb, VALUE);
|
|
|
|
break;
|
|
|
|
#endif
|
2006-11-03 10:42:42 +01:00
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
case blr_init_variable:
|
|
|
|
{
|
|
|
|
n = BLR_WORD;
|
|
|
|
node->nod_arg[e_init_var_id] = (jrd_nod*)(U_IPTR) n;
|
|
|
|
vec<jrd_nod*>* vector = csb->csb_variables;
|
|
|
|
if (!vector || n >= vector->count() ||
|
|
|
|
!(node->nod_arg[e_init_var_variable] = (*vector)[n]))
|
|
|
|
{
|
|
|
|
syntax_error(csb, "variable identifier");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2007-04-12 17:56:34 +02:00
|
|
|
case blr_sys_function:
|
|
|
|
node = par_sys_function(tdbb, csb);
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
default:
|
2003-09-28 23:36:05 +02:00
|
|
|
syntax_error(csb, elements[expected]);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
if (set_type)
|
|
|
|
{
|
|
|
|
if (csb->csb_g_flags & csb_blr_version4)
|
|
|
|
node->nod_type = (NOD_T) (USHORT) blr_table4[(int) blr_operator];
|
|
|
|
else
|
|
|
|
node->nod_type = (NOD_T) (USHORT) blr_table[(int) blr_operator];
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
size_t pos = 0;
|
|
|
|
if (csb->csb_dbg_info.blrToSrc.find(blr_offset, pos))
|
2006-10-30 21:58:06 +01:00
|
|
|
{
|
2007-01-20 15:18:18 +01:00
|
|
|
Firebird::MapBlrToSrcItem& i = csb->csb_dbg_info.blrToSrc[pos];
|
|
|
|
jrd_nod* node_src = PAR_make_node(tdbb, e_src_info_length);
|
2006-10-30 21:58:06 +01:00
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
node_src->nod_type = nod_src_info;
|
|
|
|
node_src->nod_arg[e_src_info_line] = (jrd_nod*) (IPTR) i.mbs_src_line;
|
|
|
|
node_src->nod_arg[e_src_info_col] = (jrd_nod*) (IPTR) i.mbs_src_col;
|
|
|
|
node_src->nod_arg[e_src_info_node] = node;
|
2006-10-30 21:58:06 +01:00
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
return node_src;
|
2006-10-30 21:58:06 +01:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static void syntax_error(CompilerScratch* csb, const TEXT* string)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s y n t a x _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post a syntax error message.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
error(csb, isc_syntaxerr,
|
|
|
|
isc_arg_string, string,
|
|
|
|
isc_arg_number, (SLONG) (csb->csb_running - csb->csb_blr - 1),
|
|
|
|
isc_arg_number, (SLONG) csb->csb_running[-1], 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static void warning(CompilerScratch* csb, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w a r n i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* This is for GBAK so that we can pass warning messages
|
|
|
|
* back to the client. DO NOT USE this function until we
|
|
|
|
* fully implement warning at the engine level.
|
|
|
|
*
|
|
|
|
* We will use the status vector like a warning vector. What
|
|
|
|
* we are going to do is leave the [1] position of the vector
|
|
|
|
* as 0 so that this will not be treated as an error, and we
|
|
|
|
* will place our warning message in the consecutive positions.
|
|
|
|
* It will be up to the caller to check these positions for
|
|
|
|
* the message.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-10 08:49:16 +02:00
|
|
|
ISC_STATUS *p;
|
2001-05-23 15:26:42 +02:00
|
|
|
int type;
|
|
|
|
va_list args;
|
|
|
|
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-24 01:28:06 +02:00
|
|
|
va_start(args, csb);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
p = tdbb->tdbb_status_vector;
|
|
|
|
|
|
|
|
/* Make sure that the [1] position is 0
|
|
|
|
indicating that no error has occured */
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
*p++ = isc_arg_gds;
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = 0;
|
|
|
|
|
|
|
|
/* Now place your warning messages starting
|
|
|
|
with position [2] */
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
*p++ = isc_arg_gds;
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick up remaining args */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
while ( (*p++ = type = va_arg(args, int)) )
|
2002-10-29 17:27:47 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (type) {
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_gds:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_string:
|
|
|
|
case isc_arg_interpreted:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_cstring:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, int);
|
|
|
|
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_number:
|
2003-04-10 08:49:16 +02:00
|
|
|
*p++ = (ISC_STATUS) va_arg(args, SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(FALSE);
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_vms:
|
|
|
|
case isc_arg_unix:
|
|
|
|
case isc_arg_win32:
|
2002-10-31 12:58:41 +01:00
|
|
|
*p++ = va_arg(args, int);
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-10-29 17:27:47 +01:00
|
|
|
}
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|