8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 01:23:03 +01:00
firebird-mirror/src/jrd/exe.h

826 lines
23 KiB
C
Raw Normal View History

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
*
* 2001.07.28: Added rse_skip to class RecordSelExpr to support LIMIT.
* 2002.09.28 Dmitry Yemanov: Reworked internal_info stuff, enhanced
* exception handling in SPs/triggers,
* implemented ROWS_AFFECTED system variable
* 2002.10.21 Nickolay Samofatov: Added support for explicit pessimistic locks
* 2002.10.29 Nickolay Samofatov: Added support for savepoints
2001-05-23 15:26:42 +02:00
*/
#ifndef JRD_EXE_H
#define JRD_EXE_H
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
#include "../jrd/jrd_blks.h"
#include "../common/classes/array.h"
2001-12-24 03:51:06 +01:00
#include "gen/iberror.h"
2001-05-23 15:26:42 +02:00
#define NODE(type, name, keyword) type,
2003-08-22 12:56:55 +02:00
typedef enum nod_t {
2001-05-23 15:26:42 +02:00
#include "../jrd/nod.h"
nod_MAX
#undef NODE
} NOD_T;
#include "../jrd/dsc.h"
2003-11-04 12:35:51 +01:00
#include "../jrd/rse.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/err_proto.h"
// This macro enables DSQL tracing code
//#define CMP_DEBUG
#ifdef CMP_DEBUG
DEFINE_TRACE_ROUTINE(cmp_trace);
#define CMP_TRACE(args) cmp_trace args
#else
#define CMP_TRACE(args) /* nothing */
#endif
class str;
struct dsc;
namespace Jrd {
class jrd_rel;
class jrd_nod;
struct sort_key_def;
class SparseBitmap;
class vec;
class jrd_prc;
struct index_desc;
struct IndexDescAlloc;
2004-03-30 06:10:52 +02:00
class Format;
// 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.
class jrd_node_base : public pool_alloc_rpt<jrd_nod*, type_nod>
{
public:
jrd_nod* nod_parent;
SLONG nod_impure; /* Inpure offset from request block */
NOD_T nod_type; /* Type of node */
UCHAR nod_flags;
SCHAR nod_scale; /* Target scale factor */
USHORT nod_count; /* Number of arguments */
Firebird::Array<jrd_nod*> *nod_variables; /* Variables and arguments this node depends on */
};
2001-05-23 15:26:42 +02:00
class jrd_nod : public jrd_node_base
2001-12-24 03:51:06 +01:00
{
public:
/* 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;
}*/
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_any_and = 2; /* and node is mapping of quantified predicate */
const int nod_double = 4;
const int nod_date = 8;
const int nod_value = 16; /* full value area required in impure space */
const int nod_evaluate = 32; /* (Gateway only) */
const int nod_agg_dbkey = 64; /* dbkey of an aggregate */
const int nod_invariant = 128; /* node is recognized as being invariant */
2001-05-23 15:26:42 +02:00
/* Special RecordSelExpr node */
2001-05-23 15:26:42 +02:00
class RecordSelExpr : public jrd_node_base
2001-12-24 03:51:06 +01:00
{
public:
USHORT rse_count;
USHORT rse_jointype; /* inner, left, full */
bool rse_writelock;
RecordSource* rse_rsb;
jrd_nod* rse_first;
jrd_nod* rse_skip;
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 */
2001-05-23 15:26:42 +02:00
#ifdef SCROLLABLE_CURSORS
jrd_nod* rse_async_message; /* asynchronous message to send for scrolling */
2001-05-23 15:26:42 +02:00
#endif
jrd_nod* rse_relation[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 rse_stream = 1; /* flags RecordSelExpr-type node as a blr_stream type */
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
// Number of nodes may fit into nod_arg of normal node to get to rse_relation
const size_t rse_delta = (sizeof(RecordSelExpr) - sizeof(jrd_nod)) / sizeof(jrd_nod::blk_repeat_type);
2001-05-23 15:26:42 +02: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;
2001-05-23 15:26:42 +02:00
/* Literal value */
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) */
class AggregateSort : public pool_alloc<type_asb>
2001-12-24 03:51:06 +01:00
{
public:
jrd_nod* nod_parent;
2001-12-24 03:51:06 +01:00
SLONG nod_impure; /* Impure offset from request block */
NOD_T nod_type; /* Type of node */
UCHAR nod_flags;
SCHAR nod_scale;
USHORT nod_count;
dsc asb_desc;
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 */
struct impure_state {
2001-05-23 15:26:42 +02:00
SSHORT sta_state;
};
2001-05-23 15:26:42 +02:00
struct impure_value {
dsc vlu_desc;
USHORT vlu_flags; // Computed/invariant flags
str* vlu_string;
2001-05-23 15:26:42 +02:00
union {
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;
void* vlu_invariant; // Pre-compiled invariant object for nod_like and other string functions
2001-05-23 15:26:42 +02:00
} vlu_misc;
};
2001-05-23 15:26:42 +02:00
struct impure_value_ex : public impure_value {
2001-05-23 15:26:42 +02:00
SLONG vlux_count;
2004-01-13 10:52:19 +01:00
};
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02: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 */
2001-05-23 15:26:42 +02:00
/* Inversion (i.e. nod_index) impure area */
struct impure_inversion {
SparseBitmap* inv_bitmap;
};
2001-05-23 15:26:42 +02:00
/* AggregateSort impure area */
2001-05-23 15:26:42 +02:00
struct impure_agg_sort {
sort_context* iasb_sort_handle;
2004-01-13 10:52:19 +01:00
};
2001-05-23 15:26:42 +02:00
/* Various field positions */
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;
const int e_arg_length = 4;
const int e_msg_number = 0;
const int e_msg_format = 1;
const int e_msg_invariants = 2;
const int e_msg_next = 3;
const int e_msg_length = 4;
const int e_fld_stream = 0;
const int e_fld_id = 1;
const int e_fld_default_value = 2; /* hold column default value info if any,
(Literal*) */
2004-04-29 02:50:02 +02:00
const int e_fld_length = 3;
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;
const int e_sto_stream = 5;
const int e_sto_length = 6;
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;
const int e_mod_sub_mod = 1;
const int e_mod_validate = 2;
const int e_mod_map_view = 3;
const int e_mod_org_stream = 4;
const int e_mod_new_stream = 5;
const int e_mod_rsb = 6;
const int e_mod_length = 7;
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;
const int e_asgn_missing = 2; /* Value for comparison for missing */
const int e_asgn_missing2 = 3; /* Value for substitute for missing */
const int e_asgn_length = 4;
const int e_rel_stream = 0;
const int e_rel_relation = 1;
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 */
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_hnd_statement = 0;
const int e_hnd_length = 1;
const int e_val_boolean = 0;
const int e_val_value = 1;
const int e_val_length = 2;
const int e_uni_stream = 0; /* Stream for union */
const int e_uni_clauses = 1; /* RecordSelExpr's for union */
const int e_uni_length = 2;
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
/* Statistical expressions */
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
/* Execute stored procedure */
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
/* Stored procedure view */
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
/* Function expression */
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
/* Generate id */
2004-04-29 02:50:02 +02:00
const int e_gen_value = 0;
const int e_gen_relation = 1;
const int e_gen_id = 1; /* Generator id (replaces e_gen_relation) */
const int e_gen_length = 2;
2001-05-23 15:26:42 +02:00
/* Protection mask */
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
2003-03-05 13:50:44 +01:00
/* Exception */
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
/* 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;
const int e_var_length = 2;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_dcl_id = 0;
const int e_dcl_invariants = 1;
const int e_dcl_desc = 2;
const int e_dcl_length = (2 + sizeof (DSC)/sizeof (::Jrd::jrd_nod*)); /* Room for descriptor */
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_dep_object = 0; /* node for registering dependencies */
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
2004-04-29 02:50:02 +02:00
const int e_scl_field = 0; /* Scalar expression (blr_index) */
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
/* Datatype cast operator */
2004-04-29 02:50:02 +02:00
const int e_cast_source = 0;
const int e_cast_fmt = 1;
const int e_cast_length = 2;
2001-05-23 15:26:42 +02:00
/* IDAPI semantics nodes */
2004-04-29 02:50:02 +02:00
const int e_index_index = 0; /* set current index (blr_set_index) */
const int e_index_stream = 1;
const int e_index_rsb = 2;
const int e_index_length = 3;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_seek_offset = 0; /* for seeking through a stream */
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
2004-04-29 02:50:02 +02:00
const int e_find_args = 0; /* for finding a key value in a stream */
const int e_find_operator = 1;
const int e_find_direction = 2;
const int e_find_stream = 3;
const int e_find_rsb = 4;
const int e_find_length = 5;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_bookmark_id = 0; /* nod_bookmark */
const int e_bookmark_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_setmark_id = 0; /* nod_set_bookmark */
const int e_setmark_stream = 1;
const int e_setmark_rsb = 2;
const int e_setmark_length = 3;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_getmark_stream = 0; /* nod_get_bookmark */
const int e_getmark_rsb = 1;
const int e_getmark_length = 2;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_relmark_id = 0; /* nod_release_bookmark */
const int e_relmark_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_lockrel_relation= 0; /* nod_lock_relation */
const int e_lockrel_level = 1;
const int e_lockrel_length = 2;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_lockrec_level = 0; /* nod_lock_record */
const int e_lockrec_stream = 1;
const int e_lockrec_rsb = 2;
const int e_lockrec_length = 3;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_brange_number = 0; /* nod_begin_range */
const int e_brange_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_erange_number = 0; /* nod_end_range */
const int e_erange_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_drange_number = 0; /* nod_delete_range */
const int e_drange_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_rellock_lock = 0; /* nod_release_lock */
const int e_rellock_length = 1;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_find_dbkey_dbkey = 0; /* double duty for nod_find_dbkey and nod_find_dbkey_version */
const int e_find_dbkey_version = 1;
const int e_find_dbkey_stream = 2;
const int e_find_dbkey_rsb = 3;
const int e_find_dbkey_length = 4;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_range_relation_number = 0; /* nod_range_relation */
const int e_range_relation_relation = 1;
const int e_range_relation_length = 2;
2001-05-23 15:26:42 +02:00
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
2004-04-29 02:50:02 +02:00
const int e_reset_from_stream = 0;
const int e_reset_to_stream = 1;
const int e_reset_from_rsb = 2;
const int e_reset_length = 3;
2001-05-23 15:26:42 +02:00
2004-04-29 02:50:02 +02:00
const int e_card_stream = 0;
const int e_card_rsb = 1;
const int e_card_length = 2;
2001-05-23 15:26:42 +02:00
/* SQL Date supporting nodes */
2004-04-29 02:50:02 +02:00
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 */
const int e_current_date_length = 1;
const int e_current_time_length = 1;
const int e_current_timestamp_length= 1;
const int e_dcl_cursor_number = 0;
const int e_dcl_cursor_rse = 1;
const int e_dcl_cursor_rsb = 2;
const int e_dcl_cursor_length = 3;
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
// Request resources
struct Resource
{
enum rsc_s
{
rsc_relation,
rsc_procedure,
rsc_index
};
enum rsc_s rsc_type;
USHORT rsc_id; /* Id of the resource */
jrd_rel* rsc_rel; /* Relation block */
jrd_prc* rsc_prc; /* Procedure block */
static bool greaterThan(const Resource& i1, const Resource& i2) {
// A few places of the engine depend on fact that rsc_type
// 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;
}
Resource(rsc_s type, USHORT id, jrd_rel* rel, jrd_prc* prc) :
rsc_type(type), rsc_id(id), rsc_rel(rel), rsc_prc(prc) { }
};
typedef Firebird::SortedArray<Resource, Firebird::EmptyStorage<Resource>,
Resource, Firebird::DefaultKeyValue<Resource>, Resource> ResourceList;
/* Access items */
struct AccessItem
{
const TEXT* acc_security_name;
SLONG acc_view_id;
const TEXT* acc_name;
const TEXT* acc_type;
USHORT acc_mask;
static int strcmp_null(const char* s1, const char* s2) {
return s1 == NULL ? s2 != NULL : s2 == NULL ? -1 : strcmp(s1, s2);
}
static bool greaterThan(const AccessItem& i1, const AccessItem& i2) {
int v;
if ((v = strcmp_null(i1.acc_security_name, i2.acc_security_name)) != 0)
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;
if ((v = strcmp(i1.acc_type, i2.acc_type)) != 0)
return v > 0;
if ((v = strcmp(i1.acc_name, i2.acc_name)) != 0)
return v > 0;
return false; // Equal
}
AccessItem(const TEXT* security_name, SLONG view_id, const TEXT* name,
const TEXT* type, USHORT mask)
: acc_security_name(security_name), acc_view_id(view_id), acc_name(name),
acc_type(type), acc_mask(mask)
{}
};
typedef Firebird::SortedArray<AccessItem, Firebird::EmptyStorage<AccessItem>,
AccessItem, Firebird::DefaultKeyValue<AccessItem>, AccessItem> AccessItemList;
// Triggers and procedures the request accesses
struct ExternalAccess
{
enum exa_act {
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
ExternalAccess(USHORT prc_id) :
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)
{ }
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;
return false; // Equal
}
};
typedef Firebird::SortedArray<ExternalAccess, Firebird::EmptyStorage<ExternalAccess>,
ExternalAccess, Firebird::DefaultKeyValue<ExternalAccess>, ExternalAccess> ExternalAccessList;
2001-05-23 15:26:42 +02:00
/* Compile scratch block */
/*
* 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.
*/
// CVC: Mike comment seems to apply only when the conversion to C++
// was being done. It's almost impossible that a repeating structure of
// the compiler scratch block be available to outsiders.
2001-05-23 15:26:42 +02:00
typedef Firebird::SortedArray<SLONG> VarInvariantArray;
typedef Firebird::Array<VarInvariantArray*> MsgInvariantArray;
class CompilerScratch : public pool_alloc<type_csb>
2001-12-24 03:51:06 +01:00
{
public:
CompilerScratch(MemoryPool& p, size_t len)
: /*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),
#endif
csb_async_message(0),
csb_count(0),
csb_n_stream(0),
csb_msg_number(0),
csb_impure(0),
csb_g_flags(0),*/
csb_external(p),
csb_access(p),
csb_resources(p),
csb_dependencies(p),
csb_fors(p),
csb_invariants(p),
csb_current_nodes(p),
2004-03-31 20:03:51 +02:00
csb_pool(p),
csb_rpt(p, len)
2001-12-24 03:51:06 +01:00
{}
static CompilerScratch* newCsb(MemoryPool& p, size_t len)
{ return FB_NEW(p) CompilerScratch(p, len); }
2001-12-24 03:51:06 +01:00
int nextStream(bool check = true)
{
if (csb_n_stream >= MAX_STREAMS && check)
{
ERR_post(isc_too_many_contexts, 0);
}
return csb_n_stream++;
}
const UCHAR* csb_blr;
const UCHAR* csb_running;
jrd_nod* csb_node;
ExternalAccessList csb_external; /* Access to outside procedures/triggers to be checked */
AccessItemList csb_access; /* Access items to be checked */
vec* csb_variables; /* Vector of variables, if any */
ResourceList csb_resources; /* Resources (relations and indexes) */
NodeStack csb_dependencies; /* objects this request depends upon */
Firebird::Array<RecordSource*> csb_fors; /* stack of fors */
Firebird::Array<jrd_nod*> csb_invariants; /* stack of invariant nodes */
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
RecordSelExpr* csb_current_rse; /* this holds the RecordSelExpr currently being processed;
unlike the current_rses stack, it references any expanded view RecordSelExpr */
2001-05-23 15:26:42 +02:00
#endif
jrd_nod* csb_async_message; /* asynchronous message to send to request */
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;
2004-03-31 20:03:51 +02:00
MemoryPool& csb_pool; /* Memory pool to be used by csb */
struct csb_repeat
{
// 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;
2004-03-31 20:03:51 +02:00
Firebird::string* csb_alias; /* SQL alias name for this instance of relation */
jrd_prc* csb_procedure;
jrd_rel* csb_view; /* parent view */
IndexDescAlloc* csb_idx; /* Packed description of indices */
jrd_nod* csb_message; /* Msg for send/receive */
2004-03-30 06:10:52 +02:00
Format* csb_format; /* Default Format for stream */
SparseBitmap* csb_fields; /* Fields referenced */
float csb_cardinality; /* Cardinality of relation */
jrd_nod* csb_plan; /* user-specified plan for this relation */
UCHAR* csb_map; /* Stream map for views */
RecordSource** csb_rsb_ptr; /* point to rsb for nod_stream */
};
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
2004-04-29 02:50:02 +02:00
const int csb_internal = 0x1; /* "csb_g_flag" switch */
const int csb_get_dependencies = 0x2;
const int csb_ignore_perm = 0x4; /* ignore permissions checks */
const int csb_blr_version4 = 0x8; /* The blr is of version 4 */
const int csb_active = 1;
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_validation = 32; /* We're in a validation expression (RDB hack) */
const int csb_store = 64; /* we are processing a store statement */
const int csb_modify = 128; /* we are processing a modify */
const int csb_compute = 256; /* compute cardinality for this stream */
const int csb_erase = 512; /* we are processing an erase */
const int csb_unmatched = 1024; /* stream has conjuncts unmatched by any index */
const int csb_dbkey = 8192; /* Dbkey as been requested (Gateway only) */
const int csb_update = 16384; /* Erase or modify for relation */
const int csb_made_river = 32768; /* stream has been included in a river */
2001-05-23 15:26:42 +02:00
/* Exception condition list */
struct xcp_repeat {
SSHORT xcp_type;
SLONG xcp_code;
};
2001-12-24 03:51:06 +01:00
class PsqlException : public pool_alloc_rpt<xcp_repeat, type_xcp>
2001-12-24 03:51:06 +01:00
{
public:
SLONG xcp_count;
xcp_repeat xcp_rpt[1];
};
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
class StatusXcp {
ISC_STATUS_ARRAY status;
public:
StatusXcp();
void clear();
void init(const ISC_STATUS*);
void copyTo(ISC_STATUS*) const;
bool success() const;
SLONG as_gdscode() const;
SLONG as_sqlcode() const;
};
2004-04-29 02:50:02 +02:00
const int XCP_MESSAGE_LENGTH = 78; // must correspond to the size of
// RDB$EXCEPTIONS.RDB$MESSAGE
} //namespace Jrd
#endif // JRD_EXE_H