8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 00:03:03 +01:00
firebird-mirror/src/dsql/gen.cpp

1689 lines
43 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: Dynamic SQL runtime support
2003-10-05 08:27:16 +02:00
* MODULE: gen.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Routines to generate BLR.
*
* 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.
2002-06-29 08:56:51 +02:00
* Contributor(s): ______________________________________
* 2001.6.21 Claudio Valderrama: BREAK and SUBSTRING.
* 2001.07.28 John Bellardo: Added code to generate blr_skip.
* 2002.07.30 Arno Brinkman: Added code, procedures to generate COALESCE, CASE
* 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
* 2003.10.05 Dmitry Yemanov: Added support for explicit cursors in PSQL
* 2004.01.16 Vlad Horsun: Added support for default parameters and
* EXECUTE BLOCK statement
2007-04-13 03:37:44 +02:00
* Adriano dos Santos Fernandes
2001-05-23 15:26:42 +02:00
*/
#include "firebird.h"
2001-05-23 15:26:42 +02:00
#include <string.h>
#include <stdio.h>
#include "../dsql/dsql.h"
2008-02-28 14:48:16 +01:00
#include "../dsql/node.h"
#include "../dsql/DdlNodes.h"
2010-10-24 02:26:00 +02:00
#include "../dsql/ExprNodes.h"
#include "../dsql/StmtNodes.h"
#include "../jrd/RecordSourceNodes.h"
2003-11-08 00:27:24 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/align.h"
2005-05-28 00:45:31 +02:00
#include "../jrd/constants.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/intl.h"
2008-02-28 14:48:16 +01:00
#include "../jrd/jrd.h"
#include "../jrd/val.h"
2001-05-23 15:26:42 +02:00
#include "../dsql/ddl_proto.h"
#include "../dsql/errd_proto.h"
#include "../dsql/gen_proto.h"
#include "../dsql/make_proto.h"
#include "../dsql/metd_proto.h"
#include "../dsql/utld_proto.h"
2005-05-28 00:45:31 +02:00
#include "../jrd/thread_proto.h"
2010-10-12 10:02:57 +02:00
#include "../common/dsc_proto.h"
#include "../yvalve/why_proto.h"
#include "gen/iberror.h"
#include "../common/StatusArg.h"
2001-05-23 15:26:42 +02:00
2008-02-28 14:48:16 +01:00
using namespace Jrd;
using namespace Dsql;
using namespace Firebird;
2008-02-28 14:48:16 +01:00
static void gen_coalesce(DsqlCompilerScratch*, const dsql_nod*);
static void gen_error_condition(DsqlCompilerScratch*, const dsql_nod*);
static void gen_plan(DsqlCompilerScratch*, const dsql_nod*);
static void gen_searched_case(DsqlCompilerScratch*, const dsql_nod*);
2009-12-21 15:20:26 +01:00
static void gen_select(DsqlCompilerScratch*, dsql_nod*);
static void gen_simple_case(DsqlCompilerScratch*, const dsql_nod*);
static void gen_statement(DsqlCompilerScratch*, const dsql_nod*);
static void gen_table_lock(DsqlCompilerScratch*, const dsql_nod*, USHORT);
static void gen_union(DsqlCompilerScratch*, const dsql_nod*);
2001-05-23 15:26:42 +02:00
void GEN_hidden_variables(DsqlCompilerScratch* dsqlScratch, bool inExpression)
{
/**************************************
*
* G E N _ h i d d e n _ v a r i a b l e s
*
**************************************
*
* Function
* Emit BLR for hidden variables.
*
**************************************/
if (dsqlScratch->hiddenVars.isEmpty())
return;
if (inExpression)
{
dsqlScratch->appendUChar(blr_stmt_expr);
if (dsqlScratch->hiddenVars.getCount() > 1)
dsqlScratch->appendUChar(blr_begin);
}
for (DsqlNodStack::const_iterator i(dsqlScratch->hiddenVars); i.hasData(); ++i)
{
2010-11-02 18:05:01 +01:00
const dsql_nod* varNod = i.object()->nod_arg[e_hidden_var_var];
const VariableNode* varNode = ExprNode::as<VariableNode>(varNod);
const dsql_var* var = varNode->dsqlVar;
dsqlScratch->appendUChar(blr_dcl_variable);
dsqlScratch->appendUShort(var->var_variable_number);
2010-11-02 18:05:01 +01:00
GEN_descriptor(dsqlScratch, &varNode->varDesc, true);
}
if (inExpression && dsqlScratch->hiddenVars.getCount() > 1)
dsqlScratch->appendUChar(blr_end);
// Clear it for GEN_expr not regenerate them.
dsqlScratch->hiddenVars.clear();
}
/**
GEN_expr
@brief Generate blr for an arbitrary expression.
@param dsqlScratch
@param node
**/
void GEN_expr(DsqlCompilerScratch* dsqlScratch, dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
case nod_class_exprnode:
{
ExprNode* exprNode = reinterpret_cast<ExprNode*>(node->nod_arg[0]);
if (exprNode->is<RseNode>())
{
GEN_rse(dsqlScratch, node);
return;
}
exprNode->genBlr(dsqlScratch);
// Check whether the node we just processed is for a dialect 3
// operation which gives a different result than the corresponding
// operation in dialect 1. If it is, and if the client dialect is 2,
// issue a warning about the difference.
// ASF: Shouldn't we check nod_gen_id2 too?
if (exprNode->dsqlCompatDialectVerb &&
dsqlScratch->clientDialect == SQL_DIALECT_V6_TRANSITION)
{
dsc desc;
MAKE_desc(dsqlScratch, &desc, node);
if (desc.dsc_dtype == dtype_int64)
{
ERRD_post_warning(
Arg::Warning(isc_dsql_dialect_warning_expr) <<
Arg::Str(exprNode->dsqlCompatDialectVerb));
}
}
return;
}
2001-05-23 15:26:42 +02:00
case nod_alias:
GEN_expr(dsqlScratch, node->nod_arg[e_alias_value]);
2001-05-23 15:26:42 +02:00
return;
case nod_dom_value:
dsqlScratch->appendUChar(blr_fid);
2010-06-22 02:53:35 +02:00
dsqlScratch->appendUChar(0); // Context
dsqlScratch->appendUShort(0); // Field id
2001-05-23 15:26:42 +02:00
return;
case nod_coalesce:
gen_coalesce(dsqlScratch, node);
return;
case nod_simple_case:
gen_simple_case(dsqlScratch, node);
return;
case nod_searched_case:
gen_searched_case(dsqlScratch, node);
return;
2008-07-03 09:12:36 +02:00
case nod_assign:
dsqlScratch->appendUChar(blr_assignment);
GEN_expr(dsqlScratch, node->nod_arg[0]);
GEN_expr(dsqlScratch, node->nod_arg[1]);
return;
case nod_hidden_var:
dsqlScratch->appendUChar(blr_stmt_expr);
// If it was not pre-declared, declare it now.
if (dsqlScratch->hiddenVars.hasData())
{
2010-11-02 18:05:01 +01:00
const VariableNode* varNode = ExprNode::as<VariableNode>(node->nod_arg[e_hidden_var_var]);
const dsql_var* var = varNode->dsqlVar;
dsqlScratch->appendUChar(blr_begin);
dsqlScratch->appendUChar(blr_dcl_variable);
dsqlScratch->appendUShort(var->var_variable_number);
2010-11-02 18:05:01 +01:00
GEN_descriptor(dsqlScratch, &varNode->varDesc, true);
}
dsqlScratch->appendUChar(blr_assignment);
GEN_expr(dsqlScratch, node->nod_arg[e_hidden_var_expr]);
GEN_expr(dsqlScratch, node->nod_arg[e_hidden_var_var]);
if (dsqlScratch->hiddenVars.hasData())
dsqlScratch->appendUChar(blr_end);
GEN_expr(dsqlScratch, node->nod_arg[e_hidden_var_var]);
return;
2001-05-23 15:26:42 +02:00
default:
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-901) <<
Arg::Gds(isc_dsql_internal_err) <<
// expression evaluation not supported
Arg::Gds(isc_expression_eval_err) <<
Arg::Gds(isc_dsql_eval_unknode) << Arg::Num(node->nod_type));
2001-05-23 15:26:42 +02:00
}
}
/**
GEN_port
@brief Generate a port from a message. Feel free to rearrange the
order of parameters.
@param dsqlScratch
@param message
**/
void GEN_port(DsqlCompilerScratch* dsqlScratch, dsql_msg* message)
2001-05-23 15:26:42 +02:00
{
2008-02-28 14:48:16 +01:00
thread_db* tdbb = JRD_get_thread_data();
2001-05-23 15:26:42 +02:00
dsqlScratch->appendUChar(blr_message);
dsqlScratch->appendUChar(message->msg_number);
dsqlScratch->appendUShort(message->msg_parameter);
2001-05-23 15:26:42 +02:00
ULONG offset = 0;
for (size_t i = 0; i < message->msg_parameters.getCount(); ++i)
{
dsql_par* parameter = message->msg_parameters[i];
parameter->par_parameter = (USHORT) i;
2001-05-23 15:26:42 +02:00
2007-07-08 10:43:30 +02:00
const USHORT fromCharSet = parameter->par_desc.getCharSet();
const USHORT toCharSet = (fromCharSet == CS_NONE || fromCharSet == CS_BINARY) ?
fromCharSet : tdbb->getCharSet();
if (parameter->par_desc.dsc_dtype <= dtype_any_text &&
tdbb->getCharSet() != CS_NONE && tdbb->getCharSet() != CS_BINARY)
2005-05-28 00:45:31 +02:00
{
USHORT adjust = 0;
2005-05-28 00:45:31 +02:00
if (parameter->par_desc.dsc_dtype == dtype_varying)
adjust = sizeof(USHORT);
2005-05-28 00:45:31 +02:00
else if (parameter->par_desc.dsc_dtype == dtype_cstring)
adjust = 1;
parameter->par_desc.dsc_length -= adjust;
2005-05-28 00:45:31 +02:00
const USHORT fromCharSetBPC = METD_get_charset_bpc(dsqlScratch->getTransaction(), fromCharSet);
const USHORT toCharSetBPC = METD_get_charset_bpc(dsqlScratch->getTransaction(), toCharSet);
2005-05-28 00:45:31 +02:00
2009-11-28 20:39:23 +01:00
parameter->par_desc.setTextType(INTL_CS_COLL_TO_TTYPE(toCharSet,
(fromCharSet == toCharSet ? INTL_GET_COLLATE(&parameter->par_desc) : 0)));
2005-05-28 00:45:31 +02:00
parameter->par_desc.dsc_length =
UTLD_char_length_to_byte_length(parameter->par_desc.dsc_length / fromCharSetBPC, toCharSetBPC);
2005-05-28 00:45:31 +02:00
parameter->par_desc.dsc_length += adjust;
2005-05-28 00:45:31 +02:00
}
else if (parameter->par_desc.dsc_dtype == dtype_blob &&
2007-07-08 10:43:30 +02:00
parameter->par_desc.dsc_sub_type == isc_blob_text &&
tdbb->getCharSet() != CS_NONE && tdbb->getCharSet() != CS_BINARY)
{
if (fromCharSet != toCharSet)
parameter->par_desc.setTextType(toCharSet);
}
2005-05-28 00:45:31 +02:00
2009-04-18 16:13:26 +02:00
// For older clients - generate an error should they try and
// access data types which did not exist in the older dialect
if (dsqlScratch->clientDialect <= SQL_DIALECT_V5)
{
2009-01-06 06:53:34 +01:00
switch (parameter->par_desc.dsc_dtype)
{
2009-04-18 16:13:26 +02:00
// In V6.0 - older clients, which we distinguish by
// their use of SQL DIALECT 0 or 1, are forbidden
// from selecting values of new datatypes
case dtype_sql_date:
case dtype_sql_time:
case dtype_int64:
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-804) <<
Arg::Gds(isc_dsql_datatype_err) <<
2009-01-06 06:53:34 +01:00
Arg::Gds(isc_sql_dialect_datatype_unsupport) <<
Arg::Num(dsqlScratch->clientDialect) <<
2009-01-06 06:53:34 +01:00
Arg::Str(DSC_dtype_tostring(parameter->par_desc.dsc_dtype)));
break;
default:
// No special action for other data types
break;
}
}
2009-01-07 10:30:57 +01:00
const USHORT align = type_alignments[parameter->par_desc.dsc_dtype];
2001-05-23 15:26:42 +02:00
if (align)
offset = FB_ALIGN(offset, align);
parameter->par_desc.dsc_address = (UCHAR*)(IPTR) offset;
offset += parameter->par_desc.dsc_length;
GEN_descriptor(dsqlScratch, &parameter->par_desc, false);
2001-05-23 15:26:42 +02:00
}
2009-11-16 10:18:24 +01:00
if (offset > MAX_FORMAT_SIZE)
{
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-204) <<
Arg::Gds(isc_imp_exc) <<
Arg::Gds(isc_blktoobig));
}
message->msg_length = (USHORT) offset;
dsqlScratch->ports.add(message);
2001-05-23 15:26:42 +02:00
}
/**
GEN_request
@brief Generate complete blr for a dsqlScratch.
@param dsqlScratch
@param node
**/
void GEN_request(DsqlCompilerScratch* scratch, dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
DsqlCompiledStatement* statement = scratch->getStatement();
if (statement->getType() == DsqlCompiledStatement::TYPE_CREATE_DB ||
statement->getType() == DsqlCompiledStatement::TYPE_DDL)
2008-02-28 14:48:16 +01:00
{
DDL_generate(scratch, node);
2001-05-23 15:26:42 +02:00
return;
}
if (statement->getFlags() & DsqlCompiledStatement::FLAG_BLR_VERSION4)
scratch->appendUChar(blr_version4);
2001-05-23 15:26:42 +02:00
else
scratch->appendUChar(blr_version5);
if (statement->getType() == DsqlCompiledStatement::TYPE_SAVEPOINT)
{
// Do not generate BEGIN..END block around savepoint statement
// to avoid breaking of savepoint logic
statement->setSendMsg(NULL);
statement->setReceiveMsg(NULL);
GEN_statement(scratch, node);
}
else
{
scratch->appendUChar(blr_begin);
2001-05-23 15:26:42 +02:00
GEN_hidden_variables(scratch, false);
switch (statement->getType())
2004-10-26 09:21:47 +02:00
{
case DsqlCompiledStatement::TYPE_SELECT:
case DsqlCompiledStatement::TYPE_SELECT_UPD:
2009-12-21 15:20:26 +01:00
gen_select(scratch, node);
2008-01-16 07:52:43 +01:00
break;
case DsqlCompiledStatement::TYPE_EXEC_BLOCK:
case DsqlCompiledStatement::TYPE_SELECT_BLOCK:
GEN_statement(scratch, node);
2008-01-16 07:52:43 +01:00
break;
default:
{
dsql_msg* message = statement->getSendMsg();
2008-01-16 07:52:43 +01:00
if (!message->msg_parameter)
statement->setSendMsg(NULL);
2009-11-16 10:18:24 +01:00
else
{
GEN_port(scratch, message);
scratch->appendUChar(blr_receive);
scratch->appendUChar(message->msg_number);
2008-01-16 07:52:43 +01:00
}
message = statement->getReceiveMsg();
2008-01-16 07:52:43 +01:00
if (!message->msg_parameter)
statement->setReceiveMsg(NULL);
2008-01-16 07:52:43 +01:00
else
GEN_port(scratch, message);
GEN_statement(scratch, node);
2001-05-23 15:26:42 +02:00
}
}
scratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
}
scratch->appendUChar(blr_eoc);
2001-05-23 15:26:42 +02:00
}
/**
GEN_start_transaction
@brief Generate tpb for set transaction. Use blr string of dsqlScratch.
If a value is not specified, default is not STUFF'ed, let the
engine handle it.
Do not allow an option to be specified more than once.
@param dsqlScratch
@param tran_node
**/
void GEN_start_transaction( DsqlCompilerScratch* dsqlScratch, const dsql_nod* tran_node)
2001-05-23 15:26:42 +02:00
{
SSHORT count = tran_node->nod_count;
2001-05-23 15:26:42 +02:00
if (!count)
return;
const dsql_nod* node = tran_node->nod_arg[0];
2001-05-23 15:26:42 +02:00
if (!node)
return;
// Find out isolation level - if specified. This is required for
// specifying the correct lock level in reserving clause.
2001-05-23 15:26:42 +02:00
2003-11-08 00:27:24 +01:00
USHORT lock_level = isc_tpb_shared;
2001-05-23 15:26:42 +02:00
2009-01-07 10:30:57 +01:00
if (count = node->nod_count)
{
2009-11-16 10:18:24 +01:00
while (count--)
{
const dsql_nod* ptr = node->nod_arg[count];
2001-05-23 15:26:42 +02:00
2008-06-06 04:25:35 +02:00
if (!ptr || ptr->nod_type != nod_isolation)
2001-05-23 15:26:42 +02:00
continue;
2009-01-06 06:53:34 +01:00
lock_level = (ptr->nod_flags & NOD_CONSISTENCY) ? isc_tpb_protected : isc_tpb_shared;
2001-05-23 15:26:42 +02:00
}
}
bool sw_access = false, sw_wait = false, sw_isolation = false,
sw_reserve = false, sw_lock_timeout = false;
int misc_flags = 0;
2009-04-18 16:13:26 +02:00
// Stuff some version info.
2001-05-23 15:26:42 +02:00
if (count = node->nod_count)
dsqlScratch->appendUChar(isc_tpb_version1);
2001-05-23 15:26:42 +02:00
while (count--)
{
const dsql_nod* ptr = node->nod_arg[count];
2001-05-23 15:26:42 +02:00
if (!ptr)
continue;
2009-01-06 06:53:34 +01:00
switch (ptr->nod_type)
{
2001-05-23 15:26:42 +02:00
case nod_access:
if (sw_access)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
2001-05-23 15:26:42 +02:00
sw_access = true;
2001-05-23 15:26:42 +02:00
if (ptr->nod_flags & NOD_READ_ONLY)
dsqlScratch->appendUChar(isc_tpb_read);
2001-05-23 15:26:42 +02:00
else
dsqlScratch->appendUChar(isc_tpb_write);
2001-05-23 15:26:42 +02:00
break;
case nod_wait:
if (sw_wait)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
2001-05-23 15:26:42 +02:00
sw_wait = true;
2001-05-23 15:26:42 +02:00
if (ptr->nod_flags & NOD_NO_WAIT)
dsqlScratch->appendUChar(isc_tpb_nowait);
2001-05-23 15:26:42 +02:00
else
dsqlScratch->appendUChar(isc_tpb_wait);
2001-05-23 15:26:42 +02:00
break;
case nod_isolation:
if (sw_isolation)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
2001-05-23 15:26:42 +02:00
sw_isolation = true;
2001-05-23 15:26:42 +02:00
if (ptr->nod_flags & NOD_CONCURRENCY)
dsqlScratch->appendUChar(isc_tpb_concurrency);
2001-05-23 15:26:42 +02:00
else if (ptr->nod_flags & NOD_CONSISTENCY)
dsqlScratch->appendUChar(isc_tpb_consistency);
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(isc_tpb_read_committed);
2001-05-23 15:26:42 +02:00
2009-01-06 06:53:34 +01:00
if (ptr->nod_count && ptr->nod_arg[0] && ptr->nod_arg[0]->nod_type == nod_version)
2004-10-26 09:21:47 +02:00
{
2001-05-23 15:26:42 +02:00
if (ptr->nod_arg[0]->nod_flags & NOD_VERSION)
dsqlScratch->appendUChar(isc_tpb_rec_version);
2001-05-23 15:26:42 +02:00
else
dsqlScratch->appendUChar(isc_tpb_no_rec_version);
2001-05-23 15:26:42 +02:00
}
else
dsqlScratch->appendUChar(isc_tpb_no_rec_version);
2001-05-23 15:26:42 +02:00
}
break;
case nod_reserve:
{
if (sw_reserve)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
sw_reserve = true;
const dsql_nod* reserve = ptr->nod_arg[0];
2009-11-16 10:18:24 +01:00
if (reserve)
{
const dsql_nod* const* temp = reserve->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* end = temp + reserve->nod_count; temp < end; temp++)
{
gen_table_lock(dsqlScratch, *temp, lock_level);
}
}
}
2009-01-07 10:30:57 +01:00
break;
case nod_tra_misc:
if (misc_flags & ptr->nod_flags)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
misc_flags |= ptr->nod_flags;
if (ptr->nod_flags & NOD_NO_AUTO_UNDO)
dsqlScratch->appendUChar(isc_tpb_no_auto_undo);
else if (ptr->nod_flags & NOD_IGNORE_LIMBO)
dsqlScratch->appendUChar(isc_tpb_ignore_limbo);
else if (ptr->nod_flags & NOD_RESTART_REQUESTS)
dsqlScratch->appendUChar(isc_tpb_restart_requests);
break;
case nod_lock_timeout:
if (sw_lock_timeout)
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_dup_option));
sw_lock_timeout = true;
2010-10-24 02:26:00 +02:00
if (ptr->nod_count == 1 && ExprNode::is<LiteralNode>(ptr->nod_arg[0]))
{
2010-10-24 02:26:00 +02:00
const int lck_timeout = (int) ExprNode::as<LiteralNode>(ptr->nod_arg[0])->getSlong();
dsqlScratch->appendUChar(isc_tpb_lock_timeout);
dsqlScratch->appendUChar(2);
dsqlScratch->appendUShort(lck_timeout);
}
break;
2001-05-23 15:26:42 +02:00
default:
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-104) <<
Arg::Gds(isc_dsql_tran_err));
2001-05-23 15:26:42 +02:00
}
}
// Store DYN data in the statement.
2010-06-17 18:06:54 +02:00
dsqlScratch->getStatement()->setDdlData(dsqlScratch->getBlrData());
2001-05-23 15:26:42 +02:00
}
/**
GEN_statement
@brief Generate blr for an arbitrary expression.
@param dsqlScratch
@param node
**/
void GEN_statement( DsqlCompilerScratch* dsqlScratch, dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
dsql_nod* temp;
dsql_nod** ptr;
const dsql_nod* const* end;
2003-11-10 10:16:38 +01:00
dsql_str* string;
2001-05-23 15:26:42 +02:00
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
2001-05-23 15:26:42 +02:00
case nod_assign:
dsqlScratch->appendUChar(blr_assignment);
GEN_expr(dsqlScratch, node->nod_arg[0]);
GEN_expr(dsqlScratch, node->nod_arg[1]);
2001-05-23 15:26:42 +02:00
return;
case nod_block:
dsqlScratch->appendUChar(blr_block);
GEN_statement(dsqlScratch, node->nod_arg[e_blk_action]);
2009-11-16 10:18:24 +01:00
if (node->nod_count > 1)
{
2001-05-23 15:26:42 +02:00
temp = node->nod_arg[e_blk_errs];
2009-01-06 06:53:34 +01:00
for (ptr = temp->nod_arg, end = ptr + temp->nod_count; ptr < end; ptr++)
{
GEN_statement(dsqlScratch, *ptr);
}
2001-05-23 15:26:42 +02:00
}
dsqlScratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
return;
case nod_class_stmtnode:
2009-11-16 10:18:24 +01:00
{
DmlNode* dmlNode = reinterpret_cast<DmlNode*>(node->nod_arg[0]);
dmlNode->genBlr(dsqlScratch);
2009-11-16 10:18:24 +01:00
}
return;
2008-01-16 07:52:43 +01:00
2001-05-23 15:26:42 +02:00
case nod_set_generator:
case nod_set_generator2:
dsqlScratch->appendUChar(blr_set_generator);
2003-11-10 10:16:38 +01:00
string = (dsql_str*) node->nod_arg[e_gen_id_name];
dsqlScratch->appendNullString(string->str_data);
GEN_expr(dsqlScratch, node->nod_arg[e_gen_id_value]);
2001-05-23 15:26:42 +02:00
return;
case nod_list:
if (!(node->nod_flags & NOD_SIMPLE_LIST))
dsqlScratch->appendUChar(blr_begin);
2009-01-06 06:53:34 +01:00
for (ptr = node->nod_arg, end = ptr + node->nod_count; ptr < end; ptr++)
{
GEN_statement(dsqlScratch, *ptr);
}
if (!(node->nod_flags & NOD_SIMPLE_LIST))
dsqlScratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
return;
case nod_erase:
case nod_erase_current:
2001-05-23 15:26:42 +02:00
case nod_modify:
case nod_modify_current:
case nod_store:
gen_statement(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
return;
case nod_on_error:
dsqlScratch->appendUChar(blr_error_handler);
2001-05-23 15:26:42 +02:00
temp = node->nod_arg[e_err_errs];
dsqlScratch->appendUShort(temp->nod_count);
2009-01-06 06:53:34 +01:00
for (ptr = temp->nod_arg, end = ptr + temp->nod_count; ptr < end; ptr++)
{
gen_error_condition(dsqlScratch, *ptr);
}
GEN_statement(dsqlScratch, node->nod_arg[e_err_action]);
2001-05-23 15:26:42 +02:00
return;
case nod_start_savepoint:
dsqlScratch->appendUChar(blr_start_savepoint);
2001-05-23 15:26:42 +02:00
return;
case nod_end_savepoint:
dsqlScratch->appendUChar(blr_end_savepoint);
2001-05-23 15:26:42 +02:00
return;
2001-05-23 15:26:42 +02:00
case nod_while:
dsqlScratch->appendUChar(blr_label);
2010-06-22 02:53:35 +02:00
dsqlScratch->appendUChar((int)(IPTR) node->nod_arg[e_while_label]->nod_arg[e_label_number]);
dsqlScratch->appendUChar(blr_loop);
dsqlScratch->appendUChar(blr_begin);
dsqlScratch->appendUChar(blr_if);
GEN_expr(dsqlScratch, node->nod_arg[e_while_cond]);
GEN_statement(dsqlScratch, node->nod_arg[e_while_action]);
dsqlScratch->appendUChar(blr_leave);
2010-06-22 02:53:35 +02:00
dsqlScratch->appendUChar((int)(IPTR) node->nod_arg[e_while_label]->nod_arg[e_label_number]);
dsqlScratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
return;
case nod_sqlcode:
case nod_gdscode:
dsqlScratch->appendUChar(blr_abort);
gen_error_condition(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
return;
case nod_cursor:
dsqlScratch->appendUChar(blr_dcl_cursor);
2010-06-22 02:53:35 +02:00
dsqlScratch->appendUShort((int)(IPTR) node->nod_arg[e_cur_number]);
if (node->nod_arg[e_cur_scroll])
dsqlScratch->appendUChar(blr_scrollable);
GEN_rse(dsqlScratch, node->nod_arg[e_cur_rse]);
temp = ExprNode::as<RseNode>(node->nod_arg[e_cur_rse])->dsqlSelectList;
dsqlScratch->appendUShort(temp->nod_count);
ptr = temp->nod_arg;
end = ptr + temp->nod_count;
while (ptr < end)
GEN_expr(dsqlScratch, *ptr++);
return;
case nod_src_info:
dsqlScratch->putDebugSrcInfo(node->nod_line, node->nod_column);
GEN_statement(dsqlScratch, node->nod_arg[e_src_info_stmt]);
return;
2001-05-23 15:26:42 +02:00
default:
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-901) <<
Arg::Gds(isc_dsql_internal_err) <<
// gen.c: node not supported
Arg::Gds(isc_node_err));
2001-05-23 15:26:42 +02:00
}
}
/**
gen_coalesce
@brief Generate BLR for coalesce function
Generate the blr values, begin with a cast and then :
blr_value_if
blr_missing
blr for expression 1
blr_value_if
blr_missing
blr for expression n-1
expression n
blr for expression n-1
@param dsqlScratch
@param node
**/
static void gen_coalesce( DsqlCompilerScratch* dsqlScratch, const dsql_nod* node)
{
// blr_value_if is used for building the coalesce function
dsql_nod* list = node->nod_arg[0];
dsqlScratch->appendUChar(blr_cast);
GEN_descriptor(dsqlScratch, &node->nod_desc, true);
dsql_nod* const* ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + (list->nod_count - 1); ptr < end; ptr++)
{
// IF (expression IS NULL) THEN
dsqlScratch->appendUChar(blr_value_if);
dsqlScratch->appendUChar(blr_missing);
GEN_expr(dsqlScratch, *ptr);
}
// Return values
GEN_expr(dsqlScratch, *ptr);
list = node->nod_arg[1];
const dsql_nod* const* const begin = list->nod_arg;
ptr = list->nod_arg + list->nod_count;
// if all expressions are NULL return NULL
for (ptr--; ptr >= begin; ptr--)
{
GEN_expr(dsqlScratch, *ptr);
}
}
/**
GEN_descriptor
@brief Generate a blr descriptor from an internal descriptor.
@param dsqlScratch
@param desc
@param texttype
**/
void GEN_descriptor( DsqlCompilerScratch* dsqlScratch, const dsc* desc, bool texttype)
2001-05-23 15:26:42 +02:00
{
2009-01-06 06:53:34 +01:00
switch (desc->dsc_dtype)
{
2001-05-23 15:26:42 +02:00
case dtype_text:
2009-11-16 10:18:24 +01:00
if (texttype || desc->dsc_ttype() == ttype_binary || desc->dsc_ttype() == ttype_none)
{
dsqlScratch->appendUChar(blr_text2);
dsqlScratch->appendUShort(desc->dsc_ttype());
2001-05-23 15:26:42 +02:00
}
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(blr_text2); // automatic transliteration
dsqlScratch->appendUShort(ttype_dynamic);
2001-05-23 15:26:42 +02:00
}
dsqlScratch->appendUShort(desc->dsc_length);
2001-05-23 15:26:42 +02:00
break;
case dtype_varying:
2009-11-16 10:18:24 +01:00
if (texttype || desc->dsc_ttype() == ttype_binary || desc->dsc_ttype() == ttype_none)
{
dsqlScratch->appendUChar(blr_varying2);
dsqlScratch->appendUShort(desc->dsc_ttype());
2001-05-23 15:26:42 +02:00
}
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(blr_varying2); // automatic transliteration
dsqlScratch->appendUShort(ttype_dynamic);
2001-05-23 15:26:42 +02:00
}
dsqlScratch->appendUShort(desc->dsc_length - sizeof(USHORT));
2001-05-23 15:26:42 +02:00
break;
case dtype_short:
dsqlScratch->appendUChar(blr_short);
dsqlScratch->appendUChar(desc->dsc_scale);
2001-05-23 15:26:42 +02:00
break;
case dtype_long:
dsqlScratch->appendUChar(blr_long);
dsqlScratch->appendUChar(desc->dsc_scale);
2001-05-23 15:26:42 +02:00
break;
case dtype_quad:
dsqlScratch->appendUChar(blr_quad);
dsqlScratch->appendUChar(desc->dsc_scale);
2001-05-23 15:26:42 +02:00
break;
case dtype_int64:
dsqlScratch->appendUChar(blr_int64);
dsqlScratch->appendUChar(desc->dsc_scale);
2001-05-23 15:26:42 +02:00
break;
case dtype_real:
dsqlScratch->appendUChar(blr_float);
2001-05-23 15:26:42 +02:00
break;
case dtype_double:
dsqlScratch->appendUChar(blr_double);
2001-05-23 15:26:42 +02:00
break;
case dtype_sql_date:
dsqlScratch->appendUChar(blr_sql_date);
2001-05-23 15:26:42 +02:00
break;
case dtype_sql_time:
dsqlScratch->appendUChar(blr_sql_time);
2001-05-23 15:26:42 +02:00
break;
case dtype_timestamp:
dsqlScratch->appendUChar(blr_timestamp);
2001-05-23 15:26:42 +02:00
break;
case dtype_array:
dsqlScratch->appendUChar(blr_quad);
dsqlScratch->appendUChar(0);
2001-05-23 15:26:42 +02:00
break;
case dtype_blob:
dsqlScratch->appendUChar(blr_blob2);
dsqlScratch->appendUShort(desc->dsc_sub_type);
dsqlScratch->appendUShort(desc->getTextType());
break;
2010-12-18 03:17:06 +01:00
case dtype_boolean:
dsqlScratch->appendUChar(blr_bool);
break;
2001-05-23 15:26:42 +02:00
default:
// don't understand dtype
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-804) <<
Arg::Gds(isc_dsql_datatype_err));
2001-05-23 15:26:42 +02:00
}
}
/**
gen_error_condition
@brief Generate blr for an error condtion
@param dsqlScratch
@param node
**/
static void gen_error_condition( DsqlCompilerScratch* dsqlScratch, const dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
2003-11-10 10:16:38 +01:00
const dsql_str* string;
2001-05-23 15:26:42 +02:00
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
2001-05-23 15:26:42 +02:00
case nod_sqlcode:
dsqlScratch->appendUChar(blr_sql_code);
dsqlScratch->appendUShort((USHORT)(IPTR) node->nod_arg[0]);
2001-05-23 15:26:42 +02:00
return;
case nod_gdscode:
dsqlScratch->appendUChar(blr_gds_code);
2003-11-10 10:16:38 +01:00
string = (dsql_str*) node->nod_arg[0];
dsqlScratch->appendNullString(string->str_data);
2001-05-23 15:26:42 +02:00
return;
case nod_exception:
dsqlScratch->appendUChar(blr_exception);
2003-11-10 10:16:38 +01:00
string = (dsql_str*) node->nod_arg[0];
dsqlScratch->appendNullString(string->str_data);
2001-05-23 15:26:42 +02:00
return;
case nod_default:
dsqlScratch->appendUChar(blr_default_code);
2001-05-23 15:26:42 +02:00
return;
default:
2003-11-04 00:59:24 +01:00
fb_assert(false);
2001-05-23 15:26:42 +02:00
return;
}
}
2010-10-09 03:57:37 +02:00
// Generate a parameter reference.
void GEN_parameter( DsqlCompilerScratch* dsqlScratch, const dsql_par* parameter)
2001-05-23 15:26:42 +02:00
{
const dsql_msg* message = parameter->par_message;
2001-05-23 15:26:42 +02:00
2004-02-02 12:02:12 +01:00
const dsql_par* null = parameter->par_null;
2009-11-16 10:18:24 +01:00
if (null != NULL)
{
dsqlScratch->appendUChar(blr_parameter2);
dsqlScratch->appendUChar(message->msg_number);
dsqlScratch->appendUShort(parameter->par_parameter);
dsqlScratch->appendUShort(null->par_parameter);
2001-05-23 15:26:42 +02:00
return;
}
dsqlScratch->appendUChar(blr_parameter);
dsqlScratch->appendUChar(message->msg_number);
dsqlScratch->appendUShort(parameter->par_parameter);
2001-05-23 15:26:42 +02:00
}
/**
gen_plan
@brief Generate blr for an access plan expression.
@param dsqlScratch
@param plan_expression
**/
static void gen_plan( DsqlCompilerScratch* dsqlScratch, const dsql_nod* plan_expression)
2001-05-23 15:26:42 +02:00
{
2009-04-18 16:13:26 +02:00
// stuff the join type
2001-05-23 15:26:42 +02:00
const dsql_nod* list = plan_expression->nod_arg[0];
2009-11-16 10:18:24 +01:00
if (list->nod_count > 1)
{
dsqlScratch->appendUChar(blr_join);
dsqlScratch->appendUChar(list->nod_count);
2001-05-23 15:26:42 +02:00
}
2009-04-18 16:13:26 +02:00
// stuff one or more plan items
2001-05-23 15:26:42 +02:00
const dsql_nod* const* ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
{
const dsql_nod* node = *ptr;
2009-11-16 10:18:24 +01:00
if (node->nod_type == nod_plan_expr)
{
gen_plan(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
continue;
}
// if we're here, it must be a nod_plan_item
2001-05-23 15:26:42 +02:00
dsqlScratch->appendUChar(blr_retrieve);
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// stuff the relation--the relation id itself is redundant except
// when there is a need to differentiate the base tables of views
2001-05-23 15:26:42 +02:00
/*const*/ dsql_nod* arg = node->nod_arg[0];
ExprNode::as<RelationSourceNode>(arg)->genBlr(dsqlScratch);
2001-05-23 15:26:42 +02:00
// now stuff the access method for this stream
2003-11-10 10:16:38 +01:00
const dsql_str* index_string;
2001-05-23 15:26:42 +02:00
arg = node->nod_arg[1];
2009-01-06 06:53:34 +01:00
switch (arg->nod_type)
{
2001-05-23 15:26:42 +02:00
case nod_natural:
dsqlScratch->appendUChar(blr_sequential);
2001-05-23 15:26:42 +02:00
break;
case nod_index_order:
dsqlScratch->appendUChar(blr_navigational);
2003-11-10 10:16:38 +01:00
index_string = (dsql_str*) arg->nod_arg[0];
dsqlScratch->appendNullString(index_string->str_data);
if (!arg->nod_arg[1])
break;
// dimitr: FALL INTO, if the plan item is ORDER ... INDEX (...)
2001-05-23 15:26:42 +02:00
case nod_index:
{
dsqlScratch->appendUChar(blr_indices);
2009-01-06 06:53:34 +01:00
arg = (arg->nod_type == nod_index) ? arg->nod_arg[0] : arg->nod_arg[1];
dsqlScratch->appendUChar(arg->nod_count);
const dsql_nod* const* ptr2 = arg->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end2 = ptr2 + arg->nod_count; ptr2 < end2; ptr2++)
{
2003-11-10 10:16:38 +01:00
index_string = (dsql_str*) * ptr2;
dsqlScratch->appendNullString(index_string->str_data);
}
break;
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
break;
}
}
}
/**
GEN_rse
@brief Generate a record selection expression.
@param dsqlScratch
@param rse
**/
void GEN_rse( DsqlCompilerScratch* dsqlScratch, const dsql_nod* rseNod)
2001-05-23 15:26:42 +02:00
{
const RseNode* rse = ExprNode::as<RseNode>(rseNod);
if (rseNod->nod_flags & NOD_SELECT_EXPR_SINGLETON)
dsqlScratch->appendUChar(blr_singular);
2001-05-23 15:26:42 +02:00
if (rse->dsqlExplicitJoin)
{
dsqlScratch->appendUChar(blr_rs_stream);
fb_assert(rse->dsqlStreams->nod_count == 2);
}
else
dsqlScratch->appendUChar(blr_rse);
2001-05-23 15:26:42 +02:00
dsql_nod* list = rse->dsqlStreams;
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// Handle source streams
2001-05-23 15:26:42 +02:00
if (ExprNode::is<UnionSourceNode>(list))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(1);
gen_union(dsqlScratch, rseNod);
2001-05-23 15:26:42 +02:00
}
2009-01-06 06:53:34 +01:00
else if (list->nod_type == nod_list)
{
dsqlScratch->appendUChar(list->nod_count);
dsql_nod* const* ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
{
dsql_nod* node = *ptr;
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
case nod_class_exprnode:
GEN_expr(dsqlScratch, node);
2009-01-06 06:53:34 +01:00
break;
2003-08-15 01:34:37 +02:00
}
2001-05-23 15:26:42 +02:00
}
}
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(1);
GEN_expr(dsqlScratch, list);
2001-05-23 15:26:42 +02:00
}
if (rse->flags & RseNode::FLAG_WRITELOCK)
dsqlScratch->appendUChar(blr_writelock);
dsql_nod* node;
if ((node = rse->dsqlFirst))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_first);
GEN_expr(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
}
if ((node = rse->dsqlSkip))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_skip);
2010-01-20 08:34:31 +01:00
GEN_expr(dsqlScratch, node);
2006-06-02 10:03:22 +02:00
}
2002-06-29 08:56:51 +02:00
if (rse->rse_jointype != blr_inner)
{
dsqlScratch->appendUChar(blr_join_type);
dsqlScratch->appendUChar(rse->rse_jointype);
}
if ((node = rse->dsqlWhere))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_boolean);
GEN_expr(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
}
if ((list = rse->dsqlOrder))
GEN_sort(dsqlScratch, list);
2001-05-23 15:26:42 +02:00
if ((list = rse->dsqlDistinct))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_project);
dsqlScratch->appendUChar(list->nod_count);
dsql_nod** ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
GEN_expr(dsqlScratch, *ptr);
2001-05-23 15:26:42 +02:00
}
2009-04-18 16:13:26 +02:00
// if the user specified an access plan to use, add it here
2001-05-23 15:26:42 +02:00
if ((node = rse->dsqlPlan) != NULL)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_plan);
gen_plan(dsqlScratch, node);
2001-05-23 15:26:42 +02:00
}
dsqlScratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
}
/**
gen_searched_case
@brief Generate BLR for CASE function (searched)
@param dsqlScratch
@param node
**/
static void gen_searched_case( DsqlCompilerScratch* dsqlScratch, const dsql_nod* node)
{
// blr_value_if is used for building the case expression
dsqlScratch->appendUChar(blr_cast);
GEN_descriptor(dsqlScratch, &node->nod_desc, true);
2009-01-06 06:53:34 +01:00
const SSHORT count = node->nod_arg[e_searched_case_search_conditions]->nod_count;
dsql_nod* boolean_list = node->nod_arg[e_searched_case_search_conditions];
dsql_nod* results_list = node->nod_arg[e_searched_case_results];
dsql_nod* const* bptr = boolean_list->nod_arg;
dsql_nod* const* rptr = results_list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = bptr + count; bptr < end; bptr++, rptr++)
{
dsqlScratch->appendUChar(blr_value_if);
GEN_expr(dsqlScratch, *bptr);
GEN_expr(dsqlScratch, *rptr);
}
// else_result
GEN_expr(dsqlScratch, node->nod_arg[e_searched_case_results]->nod_arg[count]);
}
/**
gen_select
@brief Generate BLR for a SELECT statement.
@param dsqlScratch
@param rse
**/
static void gen_select(DsqlCompilerScratch* dsqlScratch, dsql_nod* rseNod)
2001-05-23 15:26:42 +02:00
{
dsql_ctx* context;
2011-01-19 08:54:19 +01:00
RseNode* const rse = ExprNode::as<RseNode>(rseNod);
fb_assert(rse);
2001-05-23 15:26:42 +02:00
2011-01-19 08:54:19 +01:00
DsqlCompiledStatement* const statement = dsqlScratch->getStatement();
2009-12-21 15:20:26 +01:00
2009-04-18 16:13:26 +02:00
// Set up parameter for things in the select list
const dsql_nod* list = rse->dsqlSelectList;
dsql_nod* const* ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
{
dsql_par* parameter = MAKE_parameter(statement->getReceiveMsg(), true, true, 0, *ptr);
parameter->par_node = *ptr;
MAKE_desc(dsqlScratch, &parameter->par_desc, *ptr);
}
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// Set up parameter to handle EOF
2001-05-23 15:26:42 +02:00
2011-01-19 08:54:19 +01:00
dsql_par* const parameter_eof =
MAKE_parameter(statement->getReceiveMsg(), false, false, 0, NULL);
statement->setEof(parameter_eof);
parameter_eof->par_desc.dsc_dtype = dtype_short;
parameter_eof->par_desc.dsc_scale = 0;
parameter_eof->par_desc.dsc_length = sizeof(SSHORT);
2009-04-18 16:13:26 +02:00
// Save DBKEYs for possible update later
2001-05-23 15:26:42 +02:00
GenericMap<NonPooled<dsql_par*, dsql_ctx*> > paramContexts(*getDefaultMemoryPool());
if (!rse->dsqlDistinct)
2009-01-06 06:53:34 +01:00
{
2011-01-19 08:54:19 +01:00
list = rse->dsqlStreams;
2006-08-17 14:08:49 +02:00
dsql_nod* const* ptr2 = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end2 = ptr2 + list->nod_count; ptr2 < end2; ptr2++)
{
2011-01-19 08:54:19 +01:00
dsql_nod* const item = *ptr2;
RelationSourceNode* relNode;
if (item && (relNode = ExprNode::as<RelationSourceNode>(item)))
2009-01-06 06:53:34 +01:00
{
context = relNode->dsqlContext;
2011-01-19 08:54:19 +01:00
const dsql_rel* const relation = context->ctx_relation;
2009-11-16 10:18:24 +01:00
if (relation)
2009-01-06 06:53:34 +01:00
{
// Set up dbkey
2011-01-19 08:54:19 +01:00
dsql_par* parameter =
MAKE_parameter(statement->getReceiveMsg(), false, false, 0, NULL);
parameter->par_dbkey_relname = relation->rel_name;
paramContexts.put(parameter, context);
2001-05-23 15:26:42 +02:00
parameter->par_desc.dsc_dtype = dtype_text;
parameter->par_desc.dsc_ttype() = ttype_binary;
2009-01-06 06:53:34 +01:00
parameter->par_desc.dsc_length = relation->rel_dbkey_length;
2001-05-23 15:26:42 +02:00
// Set up record version - for post v33 databases
2001-05-23 15:26:42 +02:00
parameter = MAKE_parameter(statement->getReceiveMsg(), false, false, 0, NULL);
parameter->par_rec_version_relname = relation->rel_name;
paramContexts.put(parameter, context);
parameter->par_desc.dsc_dtype = dtype_text;
parameter->par_desc.dsc_ttype() = ttype_binary;
2009-01-06 06:53:34 +01:00
parameter->par_desc.dsc_length = relation->rel_dbkey_length / 2;
2001-05-23 15:26:42 +02:00
}
}
}
2001-05-23 15:26:42 +02:00
}
2009-04-18 16:13:26 +02:00
// Generate definitions for the messages
2001-05-23 15:26:42 +02:00
GEN_port(dsqlScratch, statement->getReceiveMsg());
dsql_msg* message = statement->getSendMsg();
2001-05-23 15:26:42 +02:00
if (message->msg_parameter)
GEN_port(dsqlScratch, message);
2001-05-23 15:26:42 +02:00
else
statement->setSendMsg(NULL);
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// If there is a send message, build a RECEIVE
2001-05-23 15:26:42 +02:00
if ((message = statement->getSendMsg()) != NULL)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_receive);
dsqlScratch->appendUChar(message->msg_number);
2001-05-23 15:26:42 +02:00
}
2009-04-18 16:13:26 +02:00
// Generate FOR loop
2001-05-23 15:26:42 +02:00
message = statement->getReceiveMsg();
2001-05-23 15:26:42 +02:00
dsqlScratch->appendUChar(blr_for);
dsqlScratch->appendUChar(blr_stall);
GEN_rse(dsqlScratch, rseNod);
dsqlScratch->appendUChar(blr_send);
dsqlScratch->appendUChar(message->msg_number);
dsqlScratch->appendUChar(blr_begin);
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// Build body of FOR loop
2001-05-23 15:26:42 +02:00
SSHORT constant;
dsc constant_desc;
2011-01-19 08:54:19 +01:00
constant_desc.makeShort(0, &constant);
// Add invalid usage here
dsqlScratch->appendUChar(blr_assignment);
2001-05-23 15:26:42 +02:00
constant = 1;
2010-10-24 02:26:00 +02:00
LiteralNode::genConstant(dsqlScratch, &constant_desc, false);
2010-10-09 03:57:37 +02:00
GEN_parameter(dsqlScratch, statement->getEof());
2001-05-23 15:26:42 +02:00
for (size_t i = 0; i < message->msg_parameters.getCount(); ++i)
2004-10-26 09:21:47 +02:00
{
2011-01-19 08:54:19 +01:00
dsql_par* const parameter = message->msg_parameters[i];
2009-11-16 10:18:24 +01:00
if (parameter->par_node)
{
dsqlScratch->appendUChar(blr_assignment);
GEN_expr(dsqlScratch, parameter->par_node);
2010-10-09 03:57:37 +02:00
GEN_parameter(dsqlScratch, parameter);
2001-05-23 15:26:42 +02:00
}
if (parameter->par_dbkey_relname.hasData() && paramContexts.get(parameter, context))
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_assignment);
dsqlScratch->appendUChar(blr_dbkey);
GEN_stuff_context(dsqlScratch, context);
2010-10-09 03:57:37 +02:00
GEN_parameter(dsqlScratch, parameter);
}
if (parameter->par_rec_version_relname.hasData() && paramContexts.get(parameter, context))
{
dsqlScratch->appendUChar(blr_assignment);
dsqlScratch->appendUChar(blr_record_version);
GEN_stuff_context(dsqlScratch, context);
2010-10-09 03:57:37 +02:00
GEN_parameter(dsqlScratch, parameter);
2001-05-23 15:26:42 +02:00
}
}
dsqlScratch->appendUChar(blr_end);
dsqlScratch->appendUChar(blr_send);
dsqlScratch->appendUChar(message->msg_number);
dsqlScratch->appendUChar(blr_assignment);
2001-05-23 15:26:42 +02:00
constant = 0;
2010-10-24 02:26:00 +02:00
LiteralNode::genConstant(dsqlScratch, &constant_desc, false);
2010-10-09 03:57:37 +02:00
GEN_parameter(dsqlScratch, statement->getEof());
2001-05-23 15:26:42 +02:00
}
/**
gen_simple_case
@brief Generate BLR for CASE function (simple)
@param dsqlScratch
@param node
**/
static void gen_simple_case( DsqlCompilerScratch* dsqlScratch, const dsql_nod* node)
{
// blr_value_if is used for building the case expression
dsqlScratch->appendUChar(blr_cast);
GEN_descriptor(dsqlScratch, &node->nod_desc, true);
2008-08-26 12:28:58 +02:00
const SSHORT count = node->nod_arg[e_simple_case_when_operands]->nod_count;
dsql_nod* when_list = node->nod_arg[e_simple_case_when_operands];
dsql_nod* results_list = node->nod_arg[e_simple_case_results];
dsql_nod* const* wptr = when_list->nod_arg;
dsql_nod* const* rptr = results_list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = wptr + count; wptr < end; wptr++, rptr++)
{
dsqlScratch->appendUChar(blr_value_if);
dsqlScratch->appendUChar(blr_eql);
if (wptr == when_list->nod_arg || !node->nod_arg[e_simple_case_case_operand2])
GEN_expr(dsqlScratch, node->nod_arg[e_simple_case_case_operand]);
else
GEN_expr(dsqlScratch, node->nod_arg[e_simple_case_case_operand2]);
GEN_expr(dsqlScratch, *wptr);
GEN_expr(dsqlScratch, *rptr);
}
// else_result
GEN_expr(dsqlScratch, node->nod_arg[e_simple_case_results]->nod_arg[count]);
}
// Generate a sort clause.
void GEN_sort( DsqlCompilerScratch* dsqlScratch, dsql_nod* list)
2001-05-23 15:26:42 +02:00
{
dsqlScratch->appendUChar(blr_sort);
dsqlScratch->appendUChar(list->nod_count);
2001-05-23 15:26:42 +02:00
dsql_nod* const* ptr = list->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
{
dsql_nod* nulls_placement = (*ptr)->nod_arg[e_order_nulls];
2010-10-24 02:26:00 +02:00
2009-01-06 06:53:34 +01:00
if (nulls_placement)
{
2010-10-24 02:26:00 +02:00
switch (ExprNode::as<LiteralNode>(nulls_placement)->getSlong())
2009-01-06 06:53:34 +01:00
{
case NOD_NULLS_FIRST:
dsqlScratch->appendUChar(blr_nullsfirst);
break;
case NOD_NULLS_LAST:
dsqlScratch->appendUChar(blr_nullslast);
break;
}
}
2002-09-10 20:28:23 +02:00
if ((*ptr)->nod_arg[e_order_flag])
dsqlScratch->appendUChar(blr_descending);
2001-05-23 15:26:42 +02:00
else
dsqlScratch->appendUChar(blr_ascending);
GEN_expr(dsqlScratch, (*ptr)->nod_arg[e_order_field]);
2001-05-23 15:26:42 +02:00
}
}
/**
gen_statement
@brief Generate BLR for DML statements.
@param dsqlScratch
@param node
**/
static void gen_statement(DsqlCompilerScratch* dsqlScratch, const dsql_nod* node)
{
dsql_nod* rse = NULL;
const dsql_msg* message = NULL;
bool innerSend = dsqlScratch->flags & DsqlCompilerScratch::FLAG_UPDATE_OR_INSERT;
2010-06-30 10:55:16 +02:00
const bool merge = dsqlScratch->flags & DsqlCompilerScratch::FLAG_MERGE;
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
case nod_store:
rse = node->nod_arg[e_sto_rse];
break;
case nod_modify:
rse = node->nod_arg[e_mod_rse];
break;
case nod_erase:
rse = node->nod_arg[e_era_rse];
break;
default:
innerSend = true;
break;
}
if (dsqlScratch->getStatement()->getType() == DsqlCompiledStatement::TYPE_EXEC_PROCEDURE &&
!innerSend && !merge)
{
if ((message = dsqlScratch->getStatement()->getReceiveMsg()))
{
dsqlScratch->appendUChar(blr_send);
dsqlScratch->appendUChar(message->msg_number);
}
}
2009-11-16 10:18:24 +01:00
if (rse)
{
dsqlScratch->appendUChar(blr_for);
GEN_expr(dsqlScratch, rse);
}
if (dsqlScratch->getStatement()->getType() == DsqlCompiledStatement::TYPE_EXEC_PROCEDURE)
{
if ((message = dsqlScratch->getStatement()->getReceiveMsg()))
{
dsqlScratch->appendUChar(blr_begin);
if (innerSend && !merge)
{
dsqlScratch->appendUChar(blr_send);
dsqlScratch->appendUChar(message->msg_number);
}
2006-09-03 03:09:23 +02:00
}
}
dsql_nod* temp;
2008-07-02 11:46:44 +02:00
const dsql_ctx* context;
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
case nod_store:
dsqlScratch->appendUChar(node->nod_arg[e_sto_return] ? blr_store2 : blr_store);
GEN_expr(dsqlScratch, node->nod_arg[e_sto_relation]);
GEN_statement(dsqlScratch, node->nod_arg[e_sto_statement]);
if (node->nod_arg[e_sto_return])
GEN_statement(dsqlScratch, node->nod_arg[e_sto_return]);
break;
case nod_modify:
dsqlScratch->appendUChar(node->nod_arg[e_mod_return] ? blr_modify2 : blr_modify);
temp = node->nod_arg[e_mod_source];
context = ExprNode::as<RelationSourceNode>(temp)->dsqlContext;
GEN_stuff_context(dsqlScratch, context);
temp = node->nod_arg[e_mod_update];
context = ExprNode::as<RelationSourceNode>(temp)->dsqlContext;
GEN_stuff_context(dsqlScratch, context);
GEN_statement(dsqlScratch, node->nod_arg[e_mod_statement]);
if (node->nod_arg[e_mod_return])
GEN_statement(dsqlScratch, node->nod_arg[e_mod_return]);
break;
case nod_modify_current:
dsqlScratch->appendUChar(node->nod_arg[e_mdc_return] ? blr_modify2 : blr_modify);
context = (dsql_ctx*) node->nod_arg[e_mdc_context];
GEN_stuff_context(dsqlScratch, context);
temp = node->nod_arg[e_mdc_update];
context = ExprNode::as<RelationSourceNode>(temp)->dsqlContext;
GEN_stuff_context(dsqlScratch, context);
GEN_statement(dsqlScratch, node->nod_arg[e_mdc_statement]);
if (node->nod_arg[e_mdc_return])
GEN_statement(dsqlScratch, node->nod_arg[e_mdc_return]);
break;
case nod_erase:
temp = node->nod_arg[e_era_relation];
context = ExprNode::as<RelationSourceNode>(temp)->dsqlContext;
2009-11-16 10:18:24 +01:00
if (node->nod_arg[e_era_return])
{
dsqlScratch->appendUChar(blr_begin);
GEN_statement(dsqlScratch, node->nod_arg[e_era_return]);
dsqlScratch->appendUChar(blr_erase);
GEN_stuff_context(dsqlScratch, context);
dsqlScratch->appendUChar(blr_end);
}
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(blr_erase);
GEN_stuff_context(dsqlScratch, context);
}
break;
case nod_erase_current:
context = (dsql_ctx*) node->nod_arg[e_erc_context];
2009-11-16 10:18:24 +01:00
if (node->nod_arg[e_erc_return])
{
dsqlScratch->appendUChar(blr_begin);
GEN_statement(dsqlScratch, node->nod_arg[e_erc_return]);
dsqlScratch->appendUChar(blr_erase);
GEN_stuff_context(dsqlScratch, context);
dsqlScratch->appendUChar(blr_end);
}
2009-11-16 10:18:24 +01:00
else
{
dsqlScratch->appendUChar(blr_erase);
GEN_stuff_context(dsqlScratch, context);
}
break;
default:
fb_assert(false);
}
if (message)
dsqlScratch->appendUChar(blr_end);
}
/**
gen_table_lock
@brief Generate tpb for table lock.
If lock level is specified, it overrrides the transaction lock level.
@param dsqlScratch
@param tbl_lock
@param lock_level
**/
static void gen_table_lock( DsqlCompilerScratch* dsqlScratch, const dsql_nod* tbl_lock, USHORT lock_level)
2001-05-23 15:26:42 +02:00
{
2008-06-03 08:14:59 +02:00
if (!tbl_lock || tbl_lock->nod_type != nod_table_lock)
2001-05-23 15:26:42 +02:00
return;
const dsql_nod* tbl_names = tbl_lock->nod_arg[e_lock_tables];
SSHORT flags = 0;
2001-05-23 15:26:42 +02:00
if (tbl_lock->nod_arg[e_lock_mode])
flags = tbl_lock->nod_arg[e_lock_mode]->nod_flags;
if (flags & NOD_PROTECTED)
2003-11-08 00:27:24 +01:00
lock_level = isc_tpb_protected;
2001-05-23 15:26:42 +02:00
else if (flags & NOD_SHARED)
2003-11-08 00:27:24 +01:00
lock_level = isc_tpb_shared;
2001-05-23 15:26:42 +02:00
2009-01-07 10:30:57 +01:00
const USHORT lock_mode = (flags & NOD_WRITE) ? isc_tpb_lock_write : isc_tpb_lock_read;
2001-05-23 15:26:42 +02:00
2006-06-02 10:03:22 +02:00
const dsql_nod* const* ptr = tbl_names->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + tbl_names->nod_count; ptr < end; ptr++)
{
2001-05-23 15:26:42 +02:00
if ((*ptr)->nod_type != nod_relation_name)
continue;
dsqlScratch->appendUChar(lock_mode);
2001-05-23 15:26:42 +02:00
// stuff table name
2003-11-10 10:16:38 +01:00
const dsql_str* temp = (dsql_str*) ((*ptr)->nod_arg[e_rln_name]);
dsqlScratch->appendNullString(temp->str_data);
2001-05-23 15:26:42 +02:00
dsqlScratch->appendUChar(lock_level);
2001-05-23 15:26:42 +02:00
}
}
/**
gen_union
@brief Generate a union of substreams.
@param dsqlScratch
@param union_node
**/
static void gen_union( DsqlCompilerScratch* dsqlScratch, const dsql_nod* union_node)
2001-05-23 15:26:42 +02:00
{
const RseNode* unionRse = ExprNode::as<RseNode>(union_node);
if (unionRse->dsqlStreams->nod_flags & NOD_UNION_RECURSIVE)
dsqlScratch->appendUChar(blr_recurse);
else
dsqlScratch->appendUChar(blr_union);
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// Obtain the context for UNION from the first dsql_map* node
dsql_nod* items = unionRse->dsqlSelectList;
dsql_nod* map_item = items->nod_arg[0];
// AB: First item could be a virtual field generated by derived table.
DerivedFieldNode* derivedField;
if ((derivedField = ExprNode::as<DerivedFieldNode>(map_item)))
map_item = derivedField->dsqlValue;
dsql_ctx* union_context = ExprNode::as<DsqlMapNode>(map_item)->context;
GEN_stuff_context(dsqlScratch, union_context);
// secondary context number must be present once in generated blr
union_context->ctx_flags &= ~CTX_recursive;
2001-05-23 15:26:42 +02:00
dsql_nod* streams = ExprNode::as<UnionSourceNode>(unionRse->dsqlStreams)->dsqlClauses;
dsqlScratch->appendUChar(streams->nod_count); // number of substreams
2001-05-23 15:26:42 +02:00
2006-06-02 10:03:22 +02:00
dsql_nod** ptr = streams->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const end = ptr + streams->nod_count; ptr < end; ptr++)
{
dsql_nod* sub_rse = *ptr;
GEN_rse(dsqlScratch, sub_rse);
items = ExprNode::as<RseNode>(sub_rse)->dsqlSelectList;
dsqlScratch->appendUChar(blr_map);
dsqlScratch->appendUShort(items->nod_count);
USHORT count = 0;
dsql_nod** iptr = items->nod_arg;
2009-01-06 06:53:34 +01:00
for (const dsql_nod* const* const iend = iptr + items->nod_count; iptr < iend; iptr++)
{
dsqlScratch->appendUShort(count);
GEN_expr(dsqlScratch, *iptr);
2001-05-23 15:26:42 +02:00
count++;
}
}
}
// Write a context number into the BLR buffer. Check for possible overflow.
void GEN_stuff_context(DsqlCompilerScratch* dsqlScratch, const dsql_ctx* context)
{
if (context->ctx_context > MAX_UCHAR)
ERRD_post(Arg::Gds(isc_too_many_contexts));
dsqlScratch->appendUChar(context->ctx_context);
if (context->ctx_flags & CTX_recursive)
{
if (context->ctx_recursive > MAX_UCHAR)
ERRD_post(Arg::Gds(isc_too_many_contexts));
dsqlScratch->appendUChar(context->ctx_recursive);
}
}