8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-31 11:23:02 +01:00
firebird-mirror/src/dsql/gen.cpp

666 lines
18 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"
#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 Firebird;
2008-02-28 14:48:16 +01:00
2012-04-25 03:42:47 +02:00
static void gen_plan(DsqlCompilerScratch*, const PlanNode*);
2001-05-23 15:26:42 +02:00
void GEN_hidden_variables(DsqlCompilerScratch* dsqlScratch)
{
/**************************************
*
* 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->hiddenVariables.isEmpty())
return;
for (Array<dsql_var*>::const_iterator i = dsqlScratch->hiddenVariables.begin();
i != dsqlScratch->hiddenVariables.end();
++i)
{
const dsql_var* var = *i;
dsqlScratch->appendUChar(blr_dcl_variable);
dsqlScratch->appendUShort(var->number);
GEN_descriptor(dsqlScratch, &var->desc, true);
}
// Clear it for GEN_expr not regenerate them.
dsqlScratch->hiddenVariables.clear();
}
/**
GEN_expr
@brief Generate blr for an arbitrary expression.
@param dsqlScratch
@param node
**/
2012-04-25 03:42:47 +02:00
void GEN_expr(DsqlCompilerScratch* dsqlScratch, ExprNode* node)
2001-05-23 15:26:42 +02:00
{
2012-04-25 03:42:47 +02:00
RseNode* rseNode = node->as<RseNode>();
if (rseNode)
2009-01-06 06:53:34 +01:00
{
2012-04-25 03:42:47 +02:00
GEN_rse(dsqlScratch, rseNode);
return;
}
2011-02-18 01:52:10 +01:00
2012-04-25 03:42:47 +02:00
node->genBlr(dsqlScratch);
2012-04-25 03:42:47 +02:00
// 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.
2012-04-25 03:42:47 +02:00
// ASF: Shouldn't we check nod_gen_id2 too?
2012-04-25 03:42:47 +02:00
if (node->kind == DmlNode::KIND_VALUE && node->dsqlCompatDialectVerb &&
dsqlScratch->clientDialect == SQL_DIALECT_V6_TRANSITION)
{
dsc desc;
MAKE_desc(dsqlScratch, &desc, static_cast<ValueExprNode*>(node));
2012-04-25 03:42:47 +02:00
if (desc.dsc_dtype == dtype_int64)
2011-02-18 01:52:10 +01:00
{
2012-04-25 03:42:47 +02:00
ERRD_post_warning(
Arg::Warning(isc_dsql_dialect_warning_expr) <<
Arg::Str(node->dsqlCompatDialectVerb));
}
2011-02-18 01:52:10 +01:00
}
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)));
2012-03-17 03:26:59 +01:00
parameter->par_desc.dsc_length = UTLD_char_length_to_byte_length(
parameter->par_desc.dsc_length / fromCharSetBPC, toCharSetBPC, adjust);
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
if (parameter->par_desc.dsc_dtype == dtype_text && parameter->par_index != 0)
{
// We should convert par_desc from text to varying so the user can receive it with
// correct length when requesting it as varying. See CORE-2606.
// But we flag it to describe as text.
parameter->par_is_text = true;
parameter->par_desc.dsc_dtype = dtype_varying;
parameter->par_desc.dsc_length += sizeof(USHORT);
}
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, true);
2001-05-23 15:26:42 +02:00
}
if (offset > MAX_MESSAGE_SIZE)
2009-11-16 10:18:24 +01:00
{
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-204) <<
Arg::Gds(isc_imp_exc) <<
Arg::Gds(isc_blktoobig));
}
message->msg_length = offset;
dsqlScratch->ports.add(message);
2001-05-23 15:26:42 +02:00
}
2012-02-20 03:02:04 +01:00
// Generate complete blr for a dsqlScratch.
void GEN_request(DsqlCompilerScratch* scratch, DmlNode* node)
2001-05-23 15:26:42 +02:00
{
DsqlCompiledStatement* statement = scratch->getStatement();
if (statement->getBlrVersion() == 4)
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);
2012-02-20 03:02:04 +01:00
node->genBlr(scratch);
}
else
{
2012-01-19 05:54:39 +01:00
const bool block = statement->getType() == DsqlCompiledStatement::TYPE_EXEC_BLOCK ||
statement->getType() == DsqlCompiledStatement::TYPE_SELECT_BLOCK;
// To parse sub-routines messages, they must not have that begin...end pair.
// And since it appears to be unnecessary for execute block too, do not generate them.
if (!block)
scratch->appendUChar(blr_begin);
2001-05-23 15:26:42 +02:00
GEN_hidden_variables(scratch);
switch (statement->getType())
2004-10-26 09:21:47 +02:00
{
case DsqlCompiledStatement::TYPE_SELECT:
case DsqlCompiledStatement::TYPE_SELECT_UPD:
case DsqlCompiledStatement::TYPE_EXEC_BLOCK:
case DsqlCompiledStatement::TYPE_SELECT_BLOCK:
2012-02-20 03:02:04 +01:00
node->genBlr(scratch);
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);
2012-02-20 03:02:04 +01:00
node->genBlr(scratch);
2001-05-23 15:26:42 +02:00
}
}
if (!block)
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_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
}
}
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
}
2012-04-25 03:42:47 +02:00
// Generate blr for an access plan expression.
static void gen_plan(DsqlCompilerScratch* dsqlScratch, const PlanNode* planNode)
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
2012-04-25 03:42:47 +02:00
const Array<NestConst<PlanNode> >& list = planNode->subNodes;
if (list.hasData())
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_join);
2012-04-25 03:42:47 +02:00
dsqlScratch->appendUChar(list.getCount());
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
2012-04-25 03:42:47 +02:00
for (const NestConst<PlanNode>* ptr = list.begin(); ptr != list.end(); ++ptr)
{
2012-04-25 03:42:47 +02:00
const PlanNode* node = *ptr;
if (node->subNodes.hasData())
2009-11-16 10:18:24 +01:00
{
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
2012-04-25 03:42:47 +02:00
// stuff the relation -- the relation id itself is redundant except
2009-04-18 16:13:26 +02:00
// when there is a need to differentiate the base tables of views
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
// ASF: node->dsqlRecordSourceNode may be NULL, and then a BLR error will happen.
// Example command: select * from (select * from t1) a plan (a natural);
if (node->dsqlRecordSourceNode)
node->dsqlRecordSourceNode->genBlr(dsqlScratch);
2001-05-23 15:26:42 +02:00
// now stuff the access method for this stream
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
unsigned delta = 0;
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
switch (node->accessType->type)
{
case PlanNode::AccessType::TYPE_SEQUENTIAL:
dsqlScratch->appendUChar(blr_sequential);
break;
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
case PlanNode::AccessType::TYPE_NAVIGATIONAL:
dsqlScratch->appendUChar(blr_navigational);
dsqlScratch->appendNullString(node->accessType->items[0].indexName.c_str());
if (node->accessType->items.getCount() == 1)
break;
// dimitr: FALL INTO, if the plan item is ORDER ... INDEX (...)
// ASF: The first item of a TYPE_NAVIGATIONAL is not for blr_indices.
++delta;
case PlanNode::AccessType::TYPE_INDICES:
{
dsqlScratch->appendUChar(blr_indices);
2012-04-25 03:42:47 +02:00
dsqlScratch->appendUChar(node->accessType->items.getCount() - delta);
const ObjectsArray<PlanNode::AccessItem>& items = node->accessType->items;
for (ObjectsArray<PlanNode::AccessItem>::const_iterator ptr2 = items.begin();
ptr2 != items.end();
++ptr2)
{
2012-04-25 03:42:47 +02:00
if (delta > 0)
--delta;
else
dsqlScratch->appendNullString(ptr2->indexName.c_str());
}
2012-04-25 03:42:47 +02:00
break;
2001-05-23 15:26:42 +02:00
}
2012-04-25 03:42:47 +02:00
default:
fb_assert(false);
break;
2001-05-23 15:26:42 +02:00
}
}
}
/**
GEN_rse
@brief Generate a record selection expression.
@param dsqlScratch
@param rse
**/
2012-04-25 03:42:47 +02:00
void GEN_rse(DsqlCompilerScratch* dsqlScratch, const RseNode* rse)
2001-05-23 15:26:42 +02:00
{
if (rse->dsqlFlags & RecordSourceNode::DFLAG_SINGLETON)
dsqlScratch->appendUChar(blr_singular);
2001-05-23 15:26:42 +02:00
if (rse->dsqlExplicitJoin)
{
dsqlScratch->appendUChar(blr_rs_stream);
2012-04-25 03:42:47 +02:00
fb_assert(rse->dsqlStreams->dsqlArgs.getCount() == 2);
}
else
dsqlScratch->appendUChar(blr_rse);
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
2012-04-25 03:42:47 +02:00
dsqlScratch->appendUChar(rse->dsqlStreams->dsqlArgs.getCount());
RecordSourceNode* const* ptr = rse->dsqlStreams->dsqlArgs.begin();
for (const RecordSourceNode* const* const end = rse->dsqlStreams->dsqlArgs.end(); ptr != end; ++ptr)
GEN_expr(dsqlScratch, *ptr);
2001-05-23 15:26:42 +02:00
if (rse->flags & RseNode::FLAG_WRITELOCK)
dsqlScratch->appendUChar(blr_writelock);
2012-04-25 03:42:47 +02:00
if (rse->dsqlFirst)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_first);
2012-04-25 03:42:47 +02:00
GEN_expr(dsqlScratch, rse->dsqlFirst);
2001-05-23 15:26:42 +02:00
}
2012-04-25 03:42:47 +02:00
if (rse->dsqlSkip)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_skip);
2012-04-25 03:42:47 +02:00
GEN_expr(dsqlScratch, rse->dsqlSkip);
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);
}
2012-04-25 03:42:47 +02:00
if (rse->dsqlWhere)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_boolean);
2012-04-25 03:42:47 +02:00
GEN_expr(dsqlScratch, rse->dsqlWhere);
2001-05-23 15:26:42 +02:00
}
2012-04-25 03:42:47 +02:00
if (rse->dsqlOrder)
GEN_sort(dsqlScratch, rse->dsqlOrder);
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
if (rse->dsqlDistinct)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_project);
2012-04-25 03:42:47 +02:00
dsqlScratch->appendUChar(rse->dsqlDistinct->dsqlArgs.getCount());
2012-04-25 03:42:47 +02:00
ValueExprNode** ptr = rse->dsqlDistinct->dsqlArgs.begin();
2012-04-25 03:42:47 +02:00
for (const ValueExprNode* const* const end = rse->dsqlDistinct->dsqlArgs.end(); 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
2012-04-25 03:42:47 +02:00
if (rse->rse_plan)
2009-11-16 10:18:24 +01:00
{
dsqlScratch->appendUChar(blr_plan);
2012-04-25 03:42:47 +02:00
gen_plan(dsqlScratch, rse->rse_plan);
2001-05-23 15:26:42 +02:00
}
dsqlScratch->appendUChar(blr_end);
2001-05-23 15:26:42 +02:00
}
// Generate a sort clause.
2012-04-25 03:42:47 +02:00
void GEN_sort(DsqlCompilerScratch* dsqlScratch, ValueListNode* list)
2001-05-23 15:26:42 +02:00
{
dsqlScratch->appendUChar(blr_sort);
2012-04-25 03:42:47 +02:00
dsqlScratch->appendUChar(list->dsqlArgs.getCount());
2001-05-23 15:26:42 +02:00
2012-04-25 03:42:47 +02:00
ValueExprNode* const* ptr = list->dsqlArgs.begin();
for (const ValueExprNode* const* const end = list->dsqlArgs.end(); ptr != end; ++ptr)
{
2012-04-25 03:42:47 +02:00
const OrderNode* orderNode = (*ptr)->as<OrderNode>();
2010-10-24 02:26:00 +02:00
2012-04-08 06:15:09 +02:00
switch (orderNode->nullsPlacement)
2009-01-06 06:53:34 +01:00
{
2012-04-08 06:15:09 +02:00
case OrderNode::NULLS_FIRST:
dsqlScratch->appendUChar(blr_nullsfirst);
break;
case OrderNode::NULLS_LAST:
dsqlScratch->appendUChar(blr_nullslast);
break;
}
2012-04-08 06:15:09 +02:00
dsqlScratch->appendUChar((orderNode->descending ? blr_descending : blr_ascending));
GEN_expr(dsqlScratch, orderNode->dsqlValue);
2001-05-23 15:26:42 +02:00
}
}
// 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);
}
}