2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: exe.h
|
|
|
|
* DESCRIPTION: Execution block definitions
|
|
|
|
*
|
|
|
|
* 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-06-29 15:03:13 +02:00
|
|
|
*
|
2004-03-28 11:10:30 +02:00
|
|
|
* 2001.07.28: Added rse_skip to class RecordSelExpr 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 Nickolay Samofatov: Added support for savepoints
|
2007-04-13 03:37:44 +02:00
|
|
|
* Adriano dos Santos Fernandes
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2003-10-03 03:53:34 +02:00
|
|
|
#ifndef JRD_EXE_H
|
|
|
|
#define JRD_EXE_H
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-04-16 20:45:40 +02:00
|
|
|
#include "../jrd/blb.h"
|
2006-05-30 15:29:24 +02:00
|
|
|
#include "../jrd/Relation.h"
|
2003-09-28 23:36:05 +02:00
|
|
|
#include "../common/classes/array.h"
|
2005-05-12 20:28:04 +02:00
|
|
|
#include "../common/classes/MetaName.h"
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2004-03-07 10:48:56 +01:00
|
|
|
#include "gen/iberror.h"
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/dsc.h"
|
2003-11-04 12:35:51 +01:00
|
|
|
#include "../jrd/rse.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-07 10:48:56 +01:00
|
|
|
#include "../jrd/err_proto.h"
|
2004-06-29 06:37:59 +02:00
|
|
|
#include "../jrd/scl.h"
|
2004-09-28 08:28:38 +02:00
|
|
|
#include "../jrd/sbm.h"
|
2004-03-07 10:48:56 +01:00
|
|
|
|
2006-10-30 21:58:06 +01:00
|
|
|
#include "../jrd/DebugInterface.h"
|
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
// This macro enables DSQL tracing code
|
2004-06-27 00:17:33 +02:00
|
|
|
//#define CMP_DEBUG
|
2003-09-28 23:36:05 +02:00
|
|
|
|
|
|
|
#ifdef CMP_DEBUG
|
|
|
|
DEFINE_TRACE_ROUTINE(cmp_trace);
|
|
|
|
#define CMP_TRACE(args) cmp_trace args
|
|
|
|
#else
|
|
|
|
#define CMP_TRACE(args) /* nothing */
|
|
|
|
#endif
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
class VaryingString;
|
2004-03-28 11:10:30 +02:00
|
|
|
struct dsc;
|
2004-03-20 15:57:40 +01:00
|
|
|
|
|
|
|
namespace Jrd {
|
|
|
|
|
2008-03-02 09:45:41 +01:00
|
|
|
#define NODE(type, name, keyword) type,
|
|
|
|
|
2008-02-28 14:48:16 +01:00
|
|
|
enum nod_t {
|
|
|
|
#include "../jrd/nod.h"
|
|
|
|
nod_MAX
|
|
|
|
#undef NODE
|
|
|
|
};
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
class jrd_rel;
|
|
|
|
class jrd_nod;
|
|
|
|
struct sort_key_def;
|
2005-12-02 08:35:34 +01:00
|
|
|
template <typename T> class vec;
|
2004-03-28 11:10:30 +02:00
|
|
|
class jrd_prc;
|
2006-10-10 21:40:33 +02:00
|
|
|
class Collation;
|
2004-03-28 11:10:30 +02:00
|
|
|
struct index_desc;
|
2004-04-04 10:25:40 +02:00
|
|
|
struct IndexDescAlloc;
|
2004-03-30 06:10:52 +02:00
|
|
|
class Format;
|
2004-03-11 06:04:26 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
// NOTE: The definition of structures RecordSelExpr and lit must be defined in
|
2003-12-31 06:36:12 +01:00
|
|
|
// exactly the same way as structure jrd_nod through item nod_count.
|
|
|
|
// Now, inheritance takes care of those common data members.
|
2004-03-11 06:04:26 +01:00
|
|
|
class jrd_node_base : public pool_alloc_rpt<jrd_nod*, type_nod>
|
2003-12-27 05:37:23 +01:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
jrd_nod* nod_parent;
|
|
|
|
SLONG nod_impure; /* Inpure offset from request block */
|
2009-04-03 12:07:55 +02:00
|
|
|
nod_t nod_type; /* Type of node */
|
2006-08-01 22:37:58 +02:00
|
|
|
USHORT nod_flags;
|
2003-12-27 05:37:23 +01:00
|
|
|
SCHAR nod_scale; /* Target scale factor */
|
|
|
|
USHORT nod_count; /* Number of arguments */
|
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-27 05:37:23 +01:00
|
|
|
|
|
|
|
class jrd_nod : public jrd_node_base
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2003-01-16 18:47:10 +01:00
|
|
|
/* jrd_nod()
|
2001-12-24 03:51:06 +01:00
|
|
|
: nod_parent(0),
|
|
|
|
nod_impure(0),
|
|
|
|
nod_type(nod_nop),
|
|
|
|
nod_flags(0),
|
|
|
|
nod_scale(0),
|
|
|
|
nod_count(0)
|
|
|
|
{
|
|
|
|
nod_arg[0] = 0;
|
2003-01-16 18:47:10 +01:00
|
|
|
}*/
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* nod_arg[1];
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int nod_comparison = 1;
|
|
|
|
const int nod_id = 1; /* marks a field node as a blr_fid guy */
|
|
|
|
const int nod_quad = 2; /* compute in quad (default is long) */
|
|
|
|
const int nod_double = 4;
|
|
|
|
const int nod_date = 8;
|
|
|
|
const int nod_value = 16; /* full value area required in impure space */
|
2004-11-17 20:33:11 +01:00
|
|
|
const int nod_deoptimize = 32; /* boolean which requires deoptimization */
|
2004-04-29 02:50:02 +02:00
|
|
|
const int nod_agg_dbkey = 64; /* dbkey of an aggregate */
|
|
|
|
const int nod_invariant = 128; /* node is recognized as being invariant */
|
2006-08-01 22:37:58 +02:00
|
|
|
const int nod_recurse = 256; /* union node is a recursive union */
|
2008-08-08 03:44:42 +02:00
|
|
|
const int nod_unique_sort = 512; // sorts using unique key - for distinct and group by
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* Special RecordSelExpr node */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
class RecordSelExpr : public jrd_node_base
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2004-03-18 06:56:06 +01:00
|
|
|
USHORT rse_count;
|
|
|
|
USHORT rse_jointype; /* inner, left, full */
|
2004-02-20 07:43:27 +01:00
|
|
|
bool rse_writelock;
|
2008-09-10 17:46:58 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rse_rsb;
|
2008-09-10 17:46:58 +02:00
|
|
|
#endif
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* rse_first;
|
2007-01-21 08:46:41 +01:00
|
|
|
jrd_nod* rse_skip;
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* rse_boolean;
|
|
|
|
jrd_nod* rse_sorted;
|
|
|
|
jrd_nod* rse_projection;
|
|
|
|
jrd_nod* rse_aggregate; /* singleton aggregate for optimizing to index */
|
|
|
|
jrd_nod* rse_plan; /* user-specified access plan */
|
2004-06-26 00:12:20 +02:00
|
|
|
VarInvariantArray *rse_invariants; /* Invariant nodes bound to top-level RSE */
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* rse_async_message; /* asynchronous message to send for scrolling */
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* rse_relation[1];
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
|
2006-02-07 09:08:49 +01:00
|
|
|
// First one is obsolete: was used for PC_ENGINE
|
|
|
|
//const int rse_stream = 1; // flags RecordSelExpr-type node as a blr_stream type
|
2004-04-29 02:50:02 +02:00
|
|
|
const int rse_singular = 2; /* flags RecordSelExpr-type node as from a singleton select */
|
|
|
|
const int rse_variant = 4; /* flags RecordSelExpr as variant (not invariant?) */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-12 08:00:52 +01:00
|
|
|
// Number of nodes may fit into nod_arg of normal node to get to rse_relation
|
2004-03-28 11:10:30 +02:00
|
|
|
const size_t rse_delta = (sizeof(RecordSelExpr) - sizeof(jrd_nod)) / sizeof(jrd_nod::blk_repeat_type);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-21 03:34:34 +01:00
|
|
|
// Types of nulls placement for each column in sort order
|
2004-04-29 02:50:02 +02:00
|
|
|
const int rse_nulls_default = 0;
|
|
|
|
const int rse_nulls_first = 1;
|
|
|
|
const int rse_nulls_last = 2;
|
2003-12-21 03:34:34 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Literal value */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
class Literal : public jrd_node_base
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
dsc lit_desc;
|
2003-10-11 01:56:57 +02:00
|
|
|
SINT64 lit_data[1]; // Defined this way to prevent SIGBUS error in 64-bit ports
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const size_t lit_delta = ((sizeof(Literal) - sizeof(jrd_nod) - sizeof(SINT64)) / sizeof(jrd_nod**));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* Aggregate Sort Block (for DISTINCT aggregates) */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
class AggregateSort : public pool_alloc<type_asb>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* nod_parent;
|
2001-12-24 03:51:06 +01:00
|
|
|
SLONG nod_impure; /* Impure offset from request block */
|
2009-04-03 12:07:55 +02:00
|
|
|
nod_t nod_type; /* Type of node */
|
2001-12-24 03:51:06 +01:00
|
|
|
UCHAR nod_flags;
|
|
|
|
SCHAR nod_scale;
|
|
|
|
USHORT nod_count;
|
|
|
|
dsc asb_desc;
|
2007-05-11 04:07:36 +02:00
|
|
|
USHORT asb_length;
|
2008-02-09 16:19:21 +01:00
|
|
|
bool asb_intl;
|
2004-03-11 06:04:26 +01:00
|
|
|
sort_key_def* asb_key_desc; /* for the aggregate */
|
2001-12-24 03:51:06 +01:00
|
|
|
UCHAR asb_key_data[1];
|
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const size_t asb_delta = ((sizeof(AggregateSort) - sizeof(jrd_nod)) / sizeof (jrd_nod**));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* Various structures in the impure area */
|
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
struct impure_state
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
SSHORT sta_state;
|
2004-03-28 11:10:30 +02:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2008-04-24 12:56:57 +02:00
|
|
|
struct impure_value
|
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
dsc vlu_desc;
|
2002-11-24 14:47:17 +01:00
|
|
|
USHORT vlu_flags; // Computed/invariant flags
|
2005-05-12 20:28:04 +02:00
|
|
|
VaryingString* vlu_string;
|
2008-04-24 12:56:57 +02:00
|
|
|
union
|
|
|
|
{
|
2007-04-16 17:31:28 +02:00
|
|
|
UCHAR vlu_uchar;
|
2001-05-23 15:26:42 +02:00
|
|
|
SSHORT vlu_short;
|
|
|
|
SLONG vlu_long;
|
|
|
|
SINT64 vlu_int64;
|
|
|
|
SQUAD vlu_quad;
|
|
|
|
SLONG vlu_dbkey[2];
|
|
|
|
float vlu_float;
|
|
|
|
double vlu_double;
|
|
|
|
GDS_TIMESTAMP vlu_timestamp;
|
|
|
|
GDS_TIME vlu_sql_time;
|
|
|
|
GDS_DATE vlu_sql_date;
|
2006-04-16 20:45:40 +02:00
|
|
|
bid vlu_bid;
|
2003-12-27 05:37:23 +01:00
|
|
|
void* vlu_invariant; // Pre-compiled invariant object for nod_like and other string functions
|
2001-05-23 15:26:42 +02:00
|
|
|
} vlu_misc;
|
2008-04-27 04:39:51 +02:00
|
|
|
|
2008-04-24 12:56:57 +02:00
|
|
|
void make_long(const SLONG val, const signed char scale = 0);
|
|
|
|
void make_int64(const SINT64 val, const signed char scale = 0);
|
2008-12-05 01:56:15 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2008-04-24 12:56:57 +02:00
|
|
|
// Do not use these methods where dsc_sub_type is not explicitly set to zero.
|
|
|
|
inline void impure_value::make_long(const SLONG val, const signed char scale)
|
|
|
|
{
|
|
|
|
this->vlu_misc.vlu_long = val;
|
|
|
|
this->vlu_desc.dsc_dtype = dtype_long;
|
|
|
|
this->vlu_desc.dsc_length = sizeof(SLONG);
|
|
|
|
this->vlu_desc.dsc_scale = scale;
|
|
|
|
this->vlu_desc.dsc_sub_type = 0;
|
|
|
|
this->vlu_desc.dsc_address = reinterpret_cast<UCHAR*>(&this->vlu_misc.vlu_long);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void impure_value::make_int64(const SINT64 val, const signed char scale)
|
|
|
|
{
|
|
|
|
this->vlu_misc.vlu_int64 = val;
|
|
|
|
this->vlu_desc.dsc_dtype = dtype_int64;
|
|
|
|
this->vlu_desc.dsc_length = sizeof(SINT64);
|
|
|
|
this->vlu_desc.dsc_scale = scale;
|
|
|
|
this->vlu_desc.dsc_sub_type = 0;
|
|
|
|
this->vlu_desc.dsc_address = reinterpret_cast<UCHAR*>(&this->vlu_misc.vlu_int64);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct impure_value_ex : public impure_value
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG vlux_count;
|
2007-05-05 21:28:46 +02:00
|
|
|
blb* vlu_blob;
|
2004-01-13 10:52:19 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2007-02-06 15:25:10 +01:00
|
|
|
const int VLU_computed = 1; // An invariant sub-query has been computed
|
|
|
|
const int VLU_null = 2; // An invariant sub-query computed to null
|
|
|
|
const int VLU_checked = 4; // Constraint already checked in first read or assignment to argument/variable
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Inversion (i.e. nod_index) impure area */
|
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
struct impure_inversion
|
|
|
|
{
|
2004-09-28 08:28:38 +02:00
|
|
|
RecordBitmap* inv_bitmap;
|
2004-03-28 11:10:30 +02:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
/* AggregateSort impure area */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
struct impure_agg_sort
|
|
|
|
{
|
2004-03-19 07:14:53 +01:00
|
|
|
sort_context* iasb_sort_handle;
|
2004-01-13 10:52:19 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Various field positions
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_for_re = 0;
|
|
|
|
const int e_for_statement = 1;
|
|
|
|
const int e_for_stall = 2;
|
|
|
|
const int e_for_rsb = 3;
|
|
|
|
const int e_for_length = 4;
|
|
|
|
|
|
|
|
const int e_arg_flag = 0;
|
|
|
|
const int e_arg_indicator = 1;
|
|
|
|
const int e_arg_message = 2;
|
|
|
|
const int e_arg_number = 3;
|
2007-12-08 14:41:26 +01:00
|
|
|
const int e_arg_info = 4;
|
|
|
|
const int e_arg_length = 5;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
2007-02-06 15:25:10 +01:00
|
|
|
const int e_msg_number = 0;
|
|
|
|
const int e_msg_format = 1;
|
|
|
|
const int e_msg_next = 2;
|
|
|
|
const int e_msg_impure_flags = 3;
|
|
|
|
const int e_msg_length = 4;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_fld_stream = 0;
|
|
|
|
const int e_fld_id = 1;
|
2008-11-30 19:55:52 +01:00
|
|
|
const int e_fld_format = 2; // relation or procedure latest format when compiling
|
|
|
|
const int e_fld_default_value = 3; // hold column default value info if any, (Literal*)
|
|
|
|
const int e_fld_length = 4;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_sto_statement = 0;
|
|
|
|
const int e_sto_statement2 = 1;
|
|
|
|
const int e_sto_sub_store = 2;
|
|
|
|
const int e_sto_validate = 3;
|
|
|
|
const int e_sto_relation = 4;
|
2005-02-01 21:05:02 +01:00
|
|
|
const int e_sto_length = 5;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_erase_statement = 0;
|
|
|
|
const int e_erase_sub_erase = 1;
|
|
|
|
const int e_erase_stream = 2;
|
|
|
|
const int e_erase_rsb = 3;
|
|
|
|
const int e_erase_length = 4;
|
|
|
|
|
|
|
|
const int e_sav_operation = 0;
|
|
|
|
const int e_sav_name = 1;
|
|
|
|
const int e_sav_length = 2;
|
|
|
|
|
|
|
|
const int e_mod_statement = 0;
|
2006-09-03 03:09:23 +02:00
|
|
|
const int e_mod_statement2 = 1;
|
|
|
|
const int e_mod_sub_mod = 2;
|
|
|
|
const int e_mod_validate = 3;
|
|
|
|
const int e_mod_map_view = 4;
|
|
|
|
const int e_mod_org_stream = 5;
|
|
|
|
const int e_mod_new_stream = 6;
|
|
|
|
const int e_mod_rsb = 7;
|
|
|
|
const int e_mod_length = 8;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_send_statement = 0;
|
|
|
|
const int e_send_message = 1;
|
|
|
|
const int e_send_length = 2;
|
|
|
|
|
|
|
|
const int e_asgn_from = 0;
|
|
|
|
const int e_asgn_to = 1;
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_asgn_missing = 2; // Value for comparison for missing
|
|
|
|
const int e_asgn_missing2 = 3; // Value for substitute for missing
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_asgn_length = 4;
|
|
|
|
|
|
|
|
const int e_rel_stream = 0;
|
|
|
|
const int e_rel_relation = 1;
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_rel_view = 2; // parent view for posting access
|
|
|
|
const int e_rel_alias = 3; // SQL alias for the relation
|
|
|
|
const int e_rel_context = 4; // user-specified context number for the relation reference
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_rel_length = 5;
|
|
|
|
|
|
|
|
const int e_idx_retrieval = 0;
|
|
|
|
const int e_idx_length = 1;
|
|
|
|
|
|
|
|
const int e_lbl_statement = 0;
|
|
|
|
const int e_lbl_label = 1;
|
|
|
|
const int e_lbl_length = 2;
|
|
|
|
|
|
|
|
const int e_any_rse = 0;
|
|
|
|
const int e_any_rsb = 1;
|
|
|
|
const int e_any_length = 2;
|
|
|
|
|
|
|
|
const int e_if_boolean = 0;
|
|
|
|
const int e_if_true = 1;
|
|
|
|
const int e_if_false = 2;
|
|
|
|
const int e_if_length = 3;
|
|
|
|
|
|
|
|
const int e_val_boolean = 0;
|
|
|
|
const int e_val_value = 1;
|
|
|
|
const int e_val_length = 2;
|
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_uni_stream = 0; // Stream for union
|
|
|
|
const int e_uni_clauses = 1; // RecordSelExpr's for union
|
2007-07-21 23:28:56 +02:00
|
|
|
const int e_uni_map_stream = 2; // stream for next level record of recursive union
|
|
|
|
const int e_uni_length = 3;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_agg_stream = 0;
|
|
|
|
const int e_agg_rse = 1;
|
|
|
|
const int e_agg_group = 2;
|
|
|
|
const int e_agg_map = 3;
|
|
|
|
const int e_agg_length = 4;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Statistical expressions
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_stat_rse = 0;
|
|
|
|
const int e_stat_value = 1;
|
|
|
|
const int e_stat_default = 2;
|
|
|
|
const int e_stat_rsb = 3;
|
|
|
|
const int e_stat_length = 4;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Execute stored procedure
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_esp_inputs = 0;
|
|
|
|
const int e_esp_in_msg = 1;
|
|
|
|
const int e_esp_outputs = 2;
|
|
|
|
const int e_esp_out_msg = 3;
|
|
|
|
const int e_esp_procedure = 4;
|
|
|
|
const int e_esp_length = 5;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Stored procedure view
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_prc_inputs = 0;
|
|
|
|
const int e_prc_in_msg = 1;
|
|
|
|
const int e_prc_stream = 2;
|
|
|
|
const int e_prc_procedure = 3;
|
|
|
|
const int e_prc_length = 4;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Function expression
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_fun_args = 0;
|
|
|
|
const int e_fun_function = 1;
|
|
|
|
const int e_fun_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Generate id
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_gen_value = 0;
|
|
|
|
const int e_gen_relation = 1;
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_gen_id = 1; // Generator id (replaces e_gen_relation)
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_gen_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Protection mask
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_pro_class = 0;
|
|
|
|
const int e_pro_relation = 1;
|
|
|
|
const int e_pro_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Exception
|
2003-03-05 13:50:44 +01:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_xcp_desc = 0;
|
|
|
|
const int e_xcp_msg = 1;
|
|
|
|
const int e_xcp_length = 2;
|
2003-03-05 13:50:44 +01:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Variable declaration
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_var_id = 0;
|
|
|
|
const int e_var_variable = 1;
|
2007-12-08 14:41:26 +01:00
|
|
|
const int e_var_info = 2;
|
|
|
|
const int e_var_length = 3;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_dcl_id = 0;
|
2004-06-26 00:12:20 +02:00
|
|
|
const int e_dcl_desc = 1;
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_dcl_length = (1 + sizeof (DSC) / sizeof(::Jrd::jrd_nod*)); // Room for descriptor
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_dep_object = 0; // node for registering dependencies
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_dep_object_type = 1;
|
|
|
|
const int e_dep_field = 2;
|
|
|
|
const int e_dep_length = 3;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_scl_field = 0; // Scalar expression (blr_index)
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_scl_subscripts = 1;
|
|
|
|
const int e_scl_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_blk_action = 0;
|
|
|
|
const int e_blk_handlers = 1;
|
|
|
|
const int e_blk_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_err_action = 0;
|
|
|
|
const int e_err_conditions = 1;
|
|
|
|
const int e_err_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Datatype cast operator
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_cast_source = 0;
|
|
|
|
const int e_cast_fmt = 1;
|
2007-01-17 02:19:01 +01:00
|
|
|
const int e_cast_iteminfo = 2;
|
|
|
|
const int e_cast_length = 3;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2006-02-07 09:08:49 +01:00
|
|
|
// CVC: These belong to SCROLLABLE_CURSORS, but I can't mark them with the macro
|
|
|
|
// because e_seek_length is used in blrtable.h.
|
2006-02-02 08:40:42 +01:00
|
|
|
const int e_seek_offset = 0; // for seeking through a stream
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_seek_direction = 1;
|
|
|
|
const int e_seek_rse = 2;
|
|
|
|
const int e_seek_length = 3;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2005-01-21 09:36:06 +01:00
|
|
|
// This is for the plan node
|
2004-04-29 02:50:02 +02:00
|
|
|
const int e_retrieve_relation = 0;
|
|
|
|
const int e_retrieve_access_type = 1;
|
|
|
|
const int e_retrieve_length = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-01-21 09:36:06 +01:00
|
|
|
// This is for the plan's access_type subnode
|
|
|
|
const int e_access_type_relation = 0;
|
|
|
|
const int e_access_type_index = 1;
|
|
|
|
const int e_access_type_index_name = 2;
|
|
|
|
const int e_access_type_length = 3;
|
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// SQL Date supporting nodes
|
|
|
|
const int e_extract_value = 0; // Node
|
|
|
|
const int e_extract_part = 1; // Integer
|
|
|
|
const int e_extract_count = 1; // Number of nodes
|
|
|
|
const int e_extract_length = 2; // Number of entries in nod_args
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_current_date_length = 1;
|
|
|
|
const int e_current_time_length = 1;
|
|
|
|
const int e_current_timestamp_length= 1;
|
|
|
|
|
2006-03-29 19:47:50 +02:00
|
|
|
const int e_dcl_cursor_rse = 0;
|
|
|
|
const int e_dcl_cursor_refs = 1;
|
|
|
|
const int e_dcl_cursor_number = 2;
|
|
|
|
const int e_dcl_cursor_rsb = 3;
|
2006-03-07 18:52:33 +01:00
|
|
|
const int e_dcl_cursor_length = 4;
|
2004-04-29 02:50:02 +02:00
|
|
|
|
|
|
|
const int e_cursor_stmt_op = 0;
|
|
|
|
const int e_cursor_stmt_number = 1;
|
|
|
|
const int e_cursor_stmt_seek = 2;
|
|
|
|
const int e_cursor_stmt_into = 3;
|
|
|
|
const int e_cursor_stmt_length = 4;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-06-06 20:14:10 +02:00
|
|
|
const int e_strlen_value = 0;
|
|
|
|
const int e_strlen_type = 1;
|
|
|
|
const int e_strlen_count = 1;
|
|
|
|
const int e_strlen_length = 2;
|
2005-05-28 00:45:31 +02:00
|
|
|
|
|
|
|
const int e_trim_value = 0;
|
|
|
|
const int e_trim_characters = 1;
|
|
|
|
const int e_trim_specification = 2;
|
|
|
|
const int e_trim_count = 2;
|
|
|
|
const int e_trim_length = 3;
|
|
|
|
|
2006-10-30 21:58:06 +01:00
|
|
|
// nod_src_info
|
2007-01-17 02:19:01 +01:00
|
|
|
const int e_src_info_line = 0;
|
|
|
|
const int e_src_info_col = 1;
|
|
|
|
const int e_src_info_node = 2;
|
|
|
|
const int e_src_info_length = 3;
|
|
|
|
|
|
|
|
// nod_init_variable
|
|
|
|
const int e_init_var_id = 0;
|
|
|
|
const int e_init_var_variable = 1;
|
2007-12-08 14:41:26 +01:00
|
|
|
const int e_init_var_info = 2;
|
|
|
|
const int e_init_var_length = 3;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
// nod_domain_validation
|
|
|
|
const int e_domval_desc = 0;
|
|
|
|
const int e_domval_length = sizeof (DSC) / sizeof(::Jrd::jrd_nod*); // Room for descriptor
|
2006-10-30 21:58:06 +01:00
|
|
|
|
2007-04-12 17:56:34 +02:00
|
|
|
// System function expression
|
|
|
|
const int e_sysfun_args = 0;
|
|
|
|
const int e_sysfun_function = 1;
|
2007-04-16 17:31:28 +02:00
|
|
|
const int e_sysfun_count = 1;
|
2007-04-12 17:56:34 +02:00
|
|
|
const int e_sysfun_length = 2;
|
|
|
|
|
2008-04-09 22:18:47 +02:00
|
|
|
// nod_exec_stmt
|
|
|
|
const int e_exec_stmt_stmt_sql = 0;
|
|
|
|
const int e_exec_stmt_data_src = 1;
|
|
|
|
const int e_exec_stmt_user = 2;
|
|
|
|
const int e_exec_stmt_password = 3;
|
|
|
|
const int e_exec_stmt_proc_block = 4;
|
|
|
|
const int e_exec_stmt_fixed_count = 5;
|
|
|
|
|
|
|
|
const int e_exec_stmt_extra_inputs = 0;
|
|
|
|
const int e_exec_stmt_extra_input_names = 1;
|
|
|
|
const int e_exec_stmt_extra_outputs = 2;
|
|
|
|
const int e_exec_stmt_extra_tran = 3;
|
2008-06-08 22:42:27 +02:00
|
|
|
const int e_exec_stmt_extra_privs = 4;
|
|
|
|
const int e_exec_stmt_extra_count = 5;
|
2008-04-09 22:18:47 +02:00
|
|
|
|
2008-09-01 15:18:02 +02:00
|
|
|
// nod_stmt_expr
|
|
|
|
const int e_stmt_expr_stmt = 0;
|
|
|
|
const int e_stmt_expr_expr = 1;
|
|
|
|
const int e_stmt_expr_length = 2;
|
|
|
|
|
2008-09-07 21:44:48 +02:00
|
|
|
// nod_derived_expr
|
|
|
|
const int e_derived_expr_expr = 0;
|
|
|
|
const int e_derived_expr_stream_count = 1;
|
|
|
|
const int e_derived_expr_stream_list = 2;
|
|
|
|
const int e_derived_expr_count = 1;
|
|
|
|
const int e_derived_expr_length = 3;
|
|
|
|
|
2004-04-18 04:50:38 +02:00
|
|
|
// Request resources
|
|
|
|
|
|
|
|
struct Resource
|
|
|
|
{
|
|
|
|
enum rsc_s
|
|
|
|
{
|
|
|
|
rsc_relation,
|
|
|
|
rsc_procedure,
|
2006-08-07 18:39:21 +02:00
|
|
|
rsc_index,
|
2006-10-10 21:40:33 +02:00
|
|
|
rsc_collation
|
2004-04-18 04:50:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
enum rsc_s rsc_type;
|
|
|
|
USHORT rsc_id; /* Id of the resource */
|
|
|
|
jrd_rel* rsc_rel; /* Relation block */
|
|
|
|
jrd_prc* rsc_prc; /* Procedure block */
|
2006-10-10 21:40:33 +02:00
|
|
|
Collation* rsc_coll; /* Collation block */
|
2004-04-18 04:50:38 +02:00
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
static bool greaterThan(const Resource& i1, const Resource& i2)
|
|
|
|
{
|
2008-12-05 01:56:15 +01:00
|
|
|
// A few places of the engine depend on fact that rsc_type
|
2004-04-18 04:50:38 +02:00
|
|
|
// is the first field in ResourceList ordering
|
|
|
|
if (i1.rsc_type != i2.rsc_type)
|
|
|
|
return i1.rsc_type > i2.rsc_type;
|
|
|
|
if (i1.rsc_type == rsc_index) {
|
|
|
|
// Sort by relation ID for now
|
|
|
|
if (i1.rsc_rel->rel_id != i2.rsc_rel->rel_id)
|
|
|
|
return i1.rsc_rel->rel_id > i2.rsc_rel->rel_id;
|
|
|
|
}
|
|
|
|
return i1.rsc_id > i2.rsc_id;
|
|
|
|
}
|
|
|
|
|
2008-12-20 09:12:19 +01:00
|
|
|
Resource(rsc_s type, USHORT id, jrd_rel* rel, jrd_prc* prc, Collation* coll)
|
|
|
|
: rsc_type(type), rsc_id(id), rsc_rel(rel), rsc_prc(prc), rsc_coll(coll)
|
|
|
|
{ }
|
2004-04-18 04:50:38 +02:00
|
|
|
};
|
|
|
|
|
2008-12-05 01:56:15 +01:00
|
|
|
typedef Firebird::SortedArray<Resource, Firebird::EmptyStorage<Resource>,
|
2004-04-18 04:50:38 +02:00
|
|
|
Resource, Firebird::DefaultKeyValue<Resource>, Resource> ResourceList;
|
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Access items
|
2005-05-12 20:28:04 +02:00
|
|
|
// In case we start to use MetaName with required pool parameter,
|
|
|
|
// access item to be reworked!
|
2004-04-18 04:50:38 +02:00
|
|
|
|
|
|
|
struct AccessItem
|
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName acc_security_name;
|
|
|
|
SLONG acc_view_id;
|
2008-01-16 09:54:50 +01:00
|
|
|
Firebird::MetaName acc_name, acc_r_name;
|
2005-05-12 20:28:04 +02:00
|
|
|
const TEXT* acc_type;
|
|
|
|
SecurityClass::flags_t acc_mask;
|
2004-04-18 04:50:38 +02:00
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
static bool greaterThan(const AccessItem& i1, const AccessItem& i2)
|
|
|
|
{
|
2004-04-18 04:50:38 +02:00
|
|
|
int v;
|
2006-03-28 11:36:22 +02:00
|
|
|
|
|
|
|
// Relations and procedures should be sorted before
|
|
|
|
// columns, hence such a tricky inverted condition
|
2008-12-05 01:56:15 +01:00
|
|
|
if ((v = -strcmp(i1.acc_type, i2.acc_type)) != 0)
|
2006-03-28 11:36:22 +02:00
|
|
|
return v > 0;
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
if ((v = i1.acc_security_name.compare(i2.acc_security_name)) != 0)
|
2004-04-18 04:50:38 +02:00
|
|
|
return v > 0;
|
|
|
|
|
|
|
|
if (i1.acc_view_id != i2.acc_view_id)
|
|
|
|
return i1.acc_view_id > i2.acc_view_id;
|
|
|
|
|
|
|
|
if (i1.acc_mask != i2.acc_mask)
|
|
|
|
return i1.acc_mask > i2.acc_mask;
|
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
if ((v = i1.acc_name.compare(i2.acc_name)) != 0)
|
2004-04-18 04:50:38 +02:00
|
|
|
return v > 0;
|
|
|
|
|
2008-01-16 09:54:50 +01:00
|
|
|
if ((v = i1.acc_r_name.compare(i2.acc_r_name)) != 0)
|
|
|
|
return v > 0;
|
|
|
|
|
2004-04-18 04:50:38 +02:00
|
|
|
return false; // Equal
|
|
|
|
}
|
|
|
|
|
2008-12-05 01:56:15 +01:00
|
|
|
AccessItem(const Firebird::MetaName& security_name, SLONG view_id,
|
|
|
|
const Firebird::MetaName& name, const TEXT* type,
|
2008-01-16 09:54:50 +01:00
|
|
|
SecurityClass::flags_t mask, const Firebird::MetaName& relName)
|
2008-12-20 09:12:19 +01:00
|
|
|
: acc_security_name(security_name), acc_view_id(view_id), acc_name(name),
|
|
|
|
acc_r_name(relName), acc_type(type), acc_mask(mask)
|
2004-04-18 04:50:38 +02:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
2008-12-05 01:56:15 +01:00
|
|
|
typedef Firebird::SortedArray<AccessItem, Firebird::EmptyStorage<AccessItem>,
|
2004-04-18 04:50:38 +02:00
|
|
|
AccessItem, Firebird::DefaultKeyValue<AccessItem>, AccessItem> AccessItemList;
|
|
|
|
|
|
|
|
// Triggers and procedures the request accesses
|
|
|
|
struct ExternalAccess
|
|
|
|
{
|
2008-05-10 05:44:57 +02:00
|
|
|
enum exa_act
|
|
|
|
{
|
2004-04-18 04:50:38 +02:00
|
|
|
exa_procedure,
|
|
|
|
exa_insert,
|
|
|
|
exa_update,
|
|
|
|
exa_delete
|
|
|
|
};
|
|
|
|
exa_act exa_action;
|
|
|
|
USHORT exa_prc_id;
|
|
|
|
USHORT exa_rel_id;
|
|
|
|
USHORT exa_view_id;
|
|
|
|
|
|
|
|
// Procedure
|
2008-12-05 01:56:15 +01:00
|
|
|
ExternalAccess(USHORT prc_id) :
|
2004-04-18 04:50:38 +02:00
|
|
|
exa_action(exa_procedure), exa_prc_id(prc_id), exa_rel_id(0), exa_view_id(0)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
// Trigger
|
|
|
|
ExternalAccess(exa_act action, USHORT rel_id, USHORT view_id) :
|
|
|
|
exa_action(action), exa_prc_id(0), exa_rel_id(rel_id), exa_view_id(view_id)
|
|
|
|
{ }
|
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
static bool greaterThan(const ExternalAccess& i1, const ExternalAccess& i2)
|
|
|
|
{
|
2004-05-09 07:48:33 +02:00
|
|
|
if (i1.exa_action != i2.exa_action)
|
|
|
|
return i1.exa_action > i2.exa_action;
|
|
|
|
if (i1.exa_prc_id != i2.exa_prc_id)
|
|
|
|
return i1.exa_prc_id > i2.exa_prc_id;
|
|
|
|
if (i1.exa_rel_id != i2.exa_rel_id)
|
|
|
|
return i1.exa_rel_id > i2.exa_rel_id;
|
|
|
|
if (i1.exa_view_id != i2.exa_view_id)
|
|
|
|
return i1.exa_view_id > i2.exa_view_id;
|
2004-04-18 04:50:38 +02:00
|
|
|
return false; // Equal
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2008-12-05 01:56:15 +01:00
|
|
|
typedef Firebird::SortedArray<ExternalAccess, Firebird::EmptyStorage<ExternalAccess>,
|
2004-04-18 04:50:38 +02:00
|
|
|
ExternalAccess, Firebird::DefaultKeyValue<ExternalAccess>, ExternalAccess> ExternalAccessList;
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
// The three structs below are used for domains DEFAULT and constraints in PSQL
|
|
|
|
struct Item
|
|
|
|
{
|
2009-04-03 12:07:55 +02:00
|
|
|
Item(nod_t aType, UCHAR aSubType, USHORT aIndex)
|
2007-01-17 02:19:01 +01:00
|
|
|
: type(aType),
|
|
|
|
subType(aSubType),
|
|
|
|
index(aIndex)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2009-04-03 12:07:55 +02:00
|
|
|
Item(nod_t aType, USHORT aIndex = 0)
|
2007-01-17 02:19:01 +01:00
|
|
|
: type(aType),
|
|
|
|
subType(0),
|
|
|
|
index(aIndex)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2009-04-03 12:07:55 +02:00
|
|
|
nod_t type;
|
2007-01-17 02:19:01 +01:00
|
|
|
UCHAR subType;
|
|
|
|
USHORT index;
|
|
|
|
|
|
|
|
bool operator >(const Item& x) const
|
|
|
|
{
|
|
|
|
if (type == x.type)
|
|
|
|
{
|
|
|
|
if (subType == x.subType)
|
|
|
|
return index > x.index;
|
2007-01-21 08:46:41 +01:00
|
|
|
|
|
|
|
return subType > x.subType;
|
2007-01-17 02:19:01 +01:00
|
|
|
}
|
2007-01-21 08:46:41 +01:00
|
|
|
|
|
|
|
return type > x.type;
|
2007-01-17 02:19:01 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FieldInfo
|
|
|
|
{
|
2008-01-16 09:54:50 +01:00
|
|
|
FieldInfo()
|
|
|
|
: nullable(false), defaultValue(0), validation(0)
|
|
|
|
{}
|
|
|
|
|
2007-01-17 02:19:01 +01:00
|
|
|
bool nullable;
|
|
|
|
jrd_nod* defaultValue;
|
|
|
|
jrd_nod* validation;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ItemInfo
|
|
|
|
{
|
|
|
|
ItemInfo(MemoryPool& p, const ItemInfo& o)
|
|
|
|
: name(p, o.name),
|
|
|
|
field(p, o.field),
|
|
|
|
nullable(o.nullable),
|
|
|
|
explicitCollation(o.explicitCollation),
|
|
|
|
fullDomain(o.fullDomain)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2008-03-10 09:16:57 +01:00
|
|
|
explicit ItemInfo(MemoryPool& p)
|
2007-01-17 02:19:01 +01:00
|
|
|
: name(p),
|
|
|
|
field(p),
|
|
|
|
nullable(true),
|
|
|
|
explicitCollation(false),
|
|
|
|
fullDomain(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ItemInfo()
|
|
|
|
: name(),
|
|
|
|
field(),
|
|
|
|
nullable(true),
|
|
|
|
explicitCollation(false),
|
|
|
|
fullDomain(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isSpecial() const
|
|
|
|
{
|
|
|
|
return !nullable || fullDomain;
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::MetaName name;
|
|
|
|
Firebird::MetaName field;
|
|
|
|
bool nullable;
|
|
|
|
bool explicitCollation;
|
|
|
|
bool fullDomain;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef Firebird::GenericMap<Firebird::Pair<Firebird::Left<Firebird::MetaName, FieldInfo> > > MapFieldInfo;
|
|
|
|
typedef Firebird::GenericMap<Firebird::Pair<Firebird::Right<Item, ItemInfo> > > MapItemInfo;
|
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Compile scratch block
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* TMN: I had to move the enclosed csb_repeat outside this class,
|
|
|
|
* since it's part of the C API. Compiling as C++ would enclose it.
|
|
|
|
*/
|
2004-03-18 06:56:06 +01:00
|
|
|
// CVC: Mike comment seems to apply only when the conversion to C++
|
|
|
|
// was being done. It's almost impossible that a repeating structure of
|
2004-03-28 11:10:30 +02:00
|
|
|
// the compiler scratch block be available to outsiders.
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
class CompilerScratch : public pool_alloc<type_csb>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2007-08-26 11:05:29 +02:00
|
|
|
CompilerScratch(MemoryPool& p, size_t len, const Firebird::MetaName& domain_validation = Firebird::MetaName())
|
2003-01-16 18:47:10 +01:00
|
|
|
: /*csb_blr(0),
|
2001-12-24 03:51:06 +01:00
|
|
|
csb_running(0),
|
|
|
|
csb_node(0),
|
|
|
|
csb_variables(0),
|
|
|
|
csb_dependencies(0),
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2001-12-24 03:51:06 +01:00
|
|
|
csb_current_rse(0),
|
|
|
|
csb_async_message(0),
|
2006-02-03 09:37:14 +01:00
|
|
|
#endif
|
2001-12-24 03:51:06 +01:00
|
|
|
csb_count(0),
|
|
|
|
csb_n_stream(0),
|
|
|
|
csb_msg_number(0),
|
|
|
|
csb_impure(0),
|
2003-01-16 18:47:10 +01:00
|
|
|
csb_g_flags(0),*/
|
2004-04-18 04:50:38 +02:00
|
|
|
csb_external(p),
|
|
|
|
csb_access(p),
|
|
|
|
csb_resources(p),
|
2004-04-18 16:22:27 +02:00
|
|
|
csb_dependencies(p),
|
2004-03-14 14:40:14 +01:00
|
|
|
csb_fors(p),
|
2006-03-10 18:46:18 +01:00
|
|
|
csb_exec_sta(p),
|
2004-03-14 14:40:14 +01:00
|
|
|
csb_invariants(p),
|
|
|
|
csb_current_nodes(p),
|
2004-03-31 20:03:51 +02:00
|
|
|
csb_pool(p),
|
2007-01-20 15:18:18 +01:00
|
|
|
csb_dbg_info(p),
|
2007-01-17 02:19:01 +01:00
|
|
|
csb_map_field_info(p),
|
|
|
|
csb_map_item_info(p),
|
2007-01-20 15:07:18 +01:00
|
|
|
csb_domain_validation(domain_validation),
|
|
|
|
csb_rpt(p, len)
|
2001-12-24 03:51:06 +01:00
|
|
|
{}
|
|
|
|
|
2007-08-26 11:05:29 +02:00
|
|
|
static CompilerScratch* newCsb(MemoryPool& p, size_t len, const Firebird::MetaName& domain_validation = Firebird::MetaName())
|
2008-05-10 05:44:57 +02:00
|
|
|
{
|
|
|
|
return FB_NEW(p) CompilerScratch(p, len, domain_validation);
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2008-05-19 15:47:48 +02:00
|
|
|
UCHAR getBlrByte()
|
|
|
|
{
|
|
|
|
return *csb_running++;
|
|
|
|
}
|
|
|
|
|
2004-03-07 10:48:56 +01:00
|
|
|
int nextStream(bool check = true)
|
|
|
|
{
|
|
|
|
if (csb_n_stream >= MAX_STREAMS && check)
|
|
|
|
{
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post(Firebird::Arg::Gds(isc_too_many_contexts));
|
2004-03-07 10:48:56 +01:00
|
|
|
}
|
|
|
|
return csb_n_stream++;
|
|
|
|
}
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
const UCHAR* csb_blr;
|
|
|
|
const UCHAR* csb_running;
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* csb_node;
|
2004-04-18 04:50:38 +02:00
|
|
|
ExternalAccessList csb_external; /* Access to outside procedures/triggers to be checked */
|
|
|
|
AccessItemList csb_access; /* Access items to be checked */
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<jrd_nod*>* csb_variables; /* Vector of variables, if any */
|
2004-04-18 04:50:38 +02:00
|
|
|
ResourceList csb_resources; /* Resources (relations and indexes) */
|
2004-04-18 16:22:27 +02:00
|
|
|
NodeStack csb_dependencies; /* objects this request depends upon */
|
|
|
|
Firebird::Array<RecordSource*> csb_fors; /* stack of fors */
|
2007-01-21 08:46:41 +01:00
|
|
|
Firebird::Array<jrd_nod*> csb_exec_sta; // Array of exec_into nodes
|
2004-02-20 07:43:27 +01:00
|
|
|
Firebird::Array<jrd_nod*> csb_invariants; /* stack of invariant nodes */
|
2004-03-28 11:10:30 +02:00
|
|
|
Firebird::Array<jrd_node_base*> csb_current_nodes; /* RecordSelExpr's and other invariant candidates within whose scope we are */
|
2001-12-24 03:51:06 +01:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSelExpr* csb_current_rse; /* this holds the RecordSelExpr currently being processed;
|
|
|
|
unlike the current_rses stack, it references any expanded view RecordSelExpr */
|
2002-11-21 00:18:16 +01:00
|
|
|
jrd_nod* csb_async_message; /* asynchronous message to send to request */
|
2006-02-03 09:37:14 +01:00
|
|
|
#endif
|
2004-03-18 06:56:06 +01:00
|
|
|
USHORT csb_n_stream; /* Next available stream */
|
|
|
|
USHORT csb_msg_number; /* Highest used message number */
|
|
|
|
SLONG csb_impure; /* Next offset into impure area */
|
|
|
|
USHORT csb_g_flags;
|
2007-01-17 02:19:01 +01:00
|
|
|
MemoryPool& csb_pool; // Memory pool to be used by csb
|
2007-01-20 15:18:18 +01:00
|
|
|
Firebird::DbgInfo csb_dbg_info; // Debug information
|
2007-01-17 02:19:01 +01:00
|
|
|
MapFieldInfo csb_map_field_info; // Map field name to field info
|
|
|
|
MapItemInfo csb_map_item_info; // Map item to item info
|
|
|
|
Firebird::MetaName csb_domain_validation; // Parsing domain constraint in PSQL
|
2003-09-28 23:36:05 +02:00
|
|
|
|
2008-05-18 03:02:35 +02:00
|
|
|
// used in cmp.cpp/pass1
|
|
|
|
jrd_rel* csb_view;
|
|
|
|
USHORT csb_view_stream;
|
|
|
|
bool csb_validate_expr;
|
2008-09-01 15:18:02 +02:00
|
|
|
USHORT csb_remap_variable;
|
2008-05-18 03:02:35 +02:00
|
|
|
|
2007-01-21 08:46:41 +01:00
|
|
|
struct csb_repeat
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
|
|
|
// We must zero-initialize this one
|
|
|
|
csb_repeat()
|
|
|
|
: csb_stream(0),
|
|
|
|
csb_view_stream(0),
|
|
|
|
csb_flags(0),
|
|
|
|
csb_indices(0),
|
|
|
|
csb_relation(0),
|
|
|
|
csb_alias(0),
|
|
|
|
csb_procedure(0),
|
|
|
|
csb_view(0),
|
|
|
|
csb_idx(0),
|
|
|
|
csb_message(0),
|
|
|
|
csb_format(0),
|
|
|
|
csb_fields(0),
|
|
|
|
csb_cardinality(0.0f), // TMN: Non-natural cardinality?!
|
|
|
|
csb_plan(0),
|
|
|
|
csb_map(0),
|
|
|
|
csb_rsb_ptr(0)
|
|
|
|
{}
|
|
|
|
|
|
|
|
UCHAR csb_stream; /* Map user context to internal stream */
|
|
|
|
UCHAR csb_view_stream; /* stream number for view relation, below */
|
|
|
|
USHORT csb_flags;
|
|
|
|
USHORT csb_indices; /* Number of indices */
|
|
|
|
|
|
|
|
jrd_rel* csb_relation;
|
2007-06-13 03:53:00 +02:00
|
|
|
Firebird::string* csb_alias; /* SQL alias name for this instance of relation */
|
2004-03-20 15:57:40 +01:00
|
|
|
jrd_prc* csb_procedure;
|
2004-03-18 06:56:06 +01:00
|
|
|
jrd_rel* csb_view; /* parent view */
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
IndexDescAlloc* csb_idx; /* Packed description of indices */
|
2004-03-18 06:56:06 +01:00
|
|
|
jrd_nod* csb_message; /* Msg for send/receive */
|
2004-03-30 06:10:52 +02:00
|
|
|
Format* csb_format; /* Default Format for stream */
|
2004-09-28 08:28:38 +02:00
|
|
|
UInt32Bitmap* csb_fields; /* Fields referenced */
|
2004-03-18 06:56:06 +01:00
|
|
|
float csb_cardinality; /* Cardinality of relation */
|
|
|
|
jrd_nod* csb_plan; /* user-specified plan for this relation */
|
|
|
|
UCHAR* csb_map; /* Stream map for views */
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource** csb_rsb_ptr; /* point to rsb for nod_stream */
|
2004-03-18 06:56:06 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
typedef csb_repeat* rpt_itr;
|
2004-03-20 04:07:21 +01:00
|
|
|
typedef const csb_repeat* rpt_const_itr;
|
2004-03-31 20:03:51 +02:00
|
|
|
Firebird::HalfStaticArray<csb_repeat, 5> csb_rpt;
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-09-08 12:08:18 +02:00
|
|
|
const int csb_internal = 1; // "csb_g_flag" switch
|
|
|
|
const int csb_get_dependencies = 2; // we are retrieving dependencies
|
|
|
|
const int csb_ignore_perm = 4; // ignore permissions checks
|
|
|
|
const int csb_blr_version4 = 8; // the BLR is of version 4
|
|
|
|
const int csb_pre_trigger = 16; // this is a BEFORE trigger
|
|
|
|
const int csb_post_trigger = 32; // this is an AFTER trigger
|
|
|
|
const int csb_validation = 64; // we're in a validation expression (RDB hack)
|
2006-12-12 17:12:15 +01:00
|
|
|
const int csb_reuse_context = 128; // allow context reusage
|
2005-09-08 12:08:18 +02:00
|
|
|
|
|
|
|
const int csb_active = 1; // stream is active
|
|
|
|
const int csb_used = 2; // context has already been defined (BLR parsing only)
|
|
|
|
const int csb_view_update = 4; // view update w/wo trigger is in progress
|
|
|
|
const int csb_trigger = 8; // NEW or OLD context in trigger
|
|
|
|
const int csb_no_dbkey = 16; // stream doesn't have a dbkey
|
|
|
|
const int csb_store = 32; // we are processing a store statement
|
|
|
|
const int csb_modify = 64; // we are processing a modify
|
2007-10-29 18:56:28 +01:00
|
|
|
const int csb_sub_stream = 128; // a sub-stream of the RSE being processed
|
2005-09-08 12:08:18 +02:00
|
|
|
const int csb_erase = 256; // we are processing an erase
|
|
|
|
const int csb_unmatched = 512; // stream has conjuncts unmatched by any index
|
|
|
|
const int csb_update = 1024; // erase or modify for relation
|
|
|
|
const int csb_made_river = 2048; // stream has been included in a river
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-02 08:40:42 +01:00
|
|
|
// Exception condition list
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
struct xcp_repeat
|
|
|
|
{
|
2002-09-28 16:04:35 +02:00
|
|
|
SSHORT xcp_type;
|
|
|
|
SLONG xcp_code;
|
|
|
|
};
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
class PsqlException : public pool_alloc_rpt<xcp_repeat, type_xcp>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2007-01-21 08:46:41 +01:00
|
|
|
public:
|
2001-12-24 03:51:06 +01:00
|
|
|
SLONG xcp_count;
|
2007-01-21 08:46:41 +01:00
|
|
|
xcp_repeat xcp_rpt[1];
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 02:50:02 +02:00
|
|
|
const int xcp_sql_code = 1;
|
|
|
|
const int xcp_gds_code = 2;
|
|
|
|
const int xcp_xcp_code = 3;
|
|
|
|
const int xcp_default = 4;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2008-05-10 05:44:57 +02:00
|
|
|
class StatusXcp
|
|
|
|
{
|
2003-11-02 12:55:17 +01:00
|
|
|
ISC_STATUS_ARRAY status;
|
|
|
|
|
|
|
|
public:
|
2004-03-28 11:10:30 +02:00
|
|
|
StatusXcp();
|
2003-11-02 12:55:17 +01:00
|
|
|
|
|
|
|
void clear();
|
|
|
|
void init(const ISC_STATUS*);
|
2003-11-05 10:02:33 +01:00
|
|
|
void copyTo(ISC_STATUS*) const;
|
2003-11-02 12:55:17 +01:00
|
|
|
bool success() const;
|
|
|
|
SLONG as_gdscode() const;
|
|
|
|
SLONG as_sqlcode() const;
|
|
|
|
};
|
|
|
|
|
2008-12-05 01:56:15 +01:00
|
|
|
// must correspond to the size of RDB$EXCEPTIONS.RDB$MESSAGE
|
2004-11-18 14:58:46 +01:00
|
|
|
// minus size of vary::vary_length (USHORT) since RDB$MESSAGE
|
|
|
|
// declared as varchar
|
|
|
|
const int XCP_MESSAGE_LENGTH = 1023 - sizeof(USHORT);
|
2005-08-24 11:16:19 +02:00
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
} //namespace Jrd
|
2002-11-14 08:35:44 +01:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
#endif // JRD_EXE_H
|