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

755 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: make.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Routines to make various blocks.
*
* 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 08:56:51 +02:00
* 2001.11.21 Claudio Valderrama: Finally solved the mystery of DSQL
* not recognizing when a UDF returns NULL. This fixes SF bug #484399.
* See case nod_udf in MAKE_desc().
* 2001.02.23 Claudio Valderrama: Fix SF bug #518350 with substring()
* and text blobs containing charsets other than ASCII/NONE/BINARY.
* 2002.07.30 Arno Brinkman:
* COALESCE, CASE support added
* procedure MAKE_desc_from_list added
* 2003.01.25 Dmitry Yemanov: Fixed problem with concatenation which
* trashed RDB$FIELD_LENGTH in the system tables. This change may
* potentially interfere with the one made by Claudio one year ago.
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 <ctype.h>
#include <string.h>
#include "../dsql/dsql.h"
2008-02-28 14:48:16 +01:00
#include "../dsql/node.h"
#include "../dsql/Nodes.h"
#include "../dsql/ExprNodes.h"
2003-11-08 00:27:24 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/intl.h"
#include "../jrd/constants.h"
#include "../jrd/align.h"
#include "../dsql/errd_proto.h"
#include "../dsql/make_proto.h"
2005-05-28 00:45:31 +02:00
#include "../dsql/metd_proto.h"
#include "../dsql/utld_proto.h"
#include "../dsql/DSqlDataTypeUtil.h"
#include "../jrd/DataTypeUtil.h"
#include "../jrd/RecordSourceNodes.h"
2008-02-28 14:48:16 +01:00
#include "../jrd/jrd.h"
#include "../jrd/ods.h"
#include "../jrd/ini.h"
2010-10-12 10:02:57 +02:00
#include "../common/dsc_proto.h"
#include "../common/cvt.h"
2005-05-28 00:45:31 +02:00
#include "../jrd/thread_proto.h"
2010-10-12 10:02:57 +02:00
#include "../yvalve/why_proto.h"
#include "../common/config/config.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;
2001-05-23 15:26:42 +02:00
dsql_nod* MAKE_const_slong(SLONG value)
{
2008-02-28 14:48:16 +01:00
thread_db* tdbb = JRD_get_thread_data();
2010-10-24 02:26:00 +02:00
SLONG* valuePtr = FB_NEW(*tdbb->getDefaultPool()) SLONG(value);
LiteralNode* literal = FB_NEW(*tdbb->getDefaultPool()) LiteralNode(*tdbb->getDefaultPool());
literal->litDesc.dsc_dtype = dtype_long;
literal->litDesc.dsc_length = sizeof(SLONG);
literal->litDesc.dsc_scale = 0;
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_address = reinterpret_cast<UCHAR*>(valuePtr);
2010-10-24 02:26:00 +02:00
dsql_nod* node = MAKE_node(nod_class_exprnode, 1);
node->nod_flags = 0;
node->nod_arg[0] = reinterpret_cast<dsql_nod*>(literal);
return node;
}
2001-12-24 03:51:06 +01:00
/**
MAKE_constant
@brief Make a constant node.
@param constant
@param numeric_flag
**/
2003-11-10 10:16:38 +01:00
dsql_nod* MAKE_constant(dsql_str* constant, dsql_constant_type numeric_flag)
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
2010-10-24 02:26:00 +02:00
LiteralNode* literal = FB_NEW(*tdbb->getDefaultPool()) LiteralNode(*tdbb->getDefaultPool());
dsql_nod* node = MAKE_node(nod_class_exprnode, 1);
node->nod_flags = 0;
node->nod_arg[0] = reinterpret_cast<dsql_nod*>(literal);
2001-05-23 15:26:42 +02:00
switch (numeric_flag)
{
case CONSTANT_DOUBLE:
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(constant, dsql_type_str);
2001-05-23 15:26:42 +02:00
2009-04-18 16:13:26 +02:00
// This is a numeric value which is transported to the engine as
// a string. The engine will convert it. Use dtype_double so that
// the engine can distinguish it from an actual string.
// Note: Due to the size of dsc_scale we are limited to numeric
// constants of less than 256 bytes.
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_dtype = dtype_double;
// Scale has no use for double
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_scale = static_cast<signed char>(constant->str_length);
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_length = sizeof(double);
literal->litDesc.dsc_address = (UCHAR*) constant->str_data;
literal->litDesc.dsc_ttype() = ttype_ascii;
literal->dsqlStr = constant;
break;
2001-05-23 15:26:42 +02:00
case CONSTANT_SINT64:
{
2010-10-24 02:26:00 +02:00
// We convert the string to an int64.
literal->litDesc.dsc_dtype = dtype_int64;
literal->litDesc.dsc_length = sizeof(SINT64);
literal->litDesc.dsc_scale = 0;
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_address = (UCHAR*) FB_NEW(*tdbb->getDefaultPool()) SINT64;
// Now convert the string to an int64. We can omit testing for
// overflow, because we would never have gotten here if yylex
// hadn't recognized the string as a valid 64-bit integer value.
// We *might* have "9223372936854775808", which works an an int64
// only if preceded by a '-', but that issue is handled in GEN_expr,
// and need not be addressed here.
// Recent change to support hex numeric constants means the input
// string now can be X8000000000000000, for example.
// Hex constants coming through this code are guaranteed to be
// valid - they start with X and contains only 0-9, A-F.
// And, they will fit in a SINT64 without overflow.
SINT64 value = 0;
const UCHAR* p = reinterpret_cast<const UCHAR*>(constant->str_data);
2003-12-22 11:00:59 +01:00
if (*p == 'X')
{
// oh no, a hex string!
2010-11-09 16:03:33 +01:00
++p; // skip the 'X' part.
UCHAR byte = 0;
2009-01-03 20:04:32 +01:00
bool nibble = ((strlen(constant->str_data) - 1) & 1);
SSHORT c;
// hex string is already upper-cased
2008-06-05 13:02:42 +02:00
while (isdigit(*p) || ((*p >= 'A') && (*p <= 'F')))
{
// Now convert the character to a nibble
if (*p >= 'A')
c = (*p - 'A') + 10;
else
c = (*p - '0');
if (nibble)
{
byte = (byte << 4) + (UCHAR) c;
2008-06-05 13:02:42 +02:00
nibble = false;
value = (value << 8) + byte;
}
else
{
byte = c;
2008-06-05 13:02:42 +02:00
nibble = true;
}
2010-11-09 16:03:33 +01:00
++p;
}
// if value is negative, then GEN_constant (from dsql/gen.cpp)
// is going to want 2 nodes: NegateNode (to hold the minus)
2010-10-24 02:26:00 +02:00
// and LiteralNode as a child to hold the value.
if (value < 0)
{
value = -value;
2010-10-24 02:26:00 +02:00
*(SINT64*) literal->litDesc.dsc_address = value;
NegateNode* negateNode = FB_NEW(*tdbb->getDefaultPool()) NegateNode(
*tdbb->getDefaultPool(), node);
node = MAKE_node(nod_class_exprnode, 1);
node->nod_arg[0] = reinterpret_cast<dsql_nod*>(negateNode);
}
else
2010-10-24 02:26:00 +02:00
*(SINT64*) literal->litDesc.dsc_address = value;
} // hex constant
else
{
2010-10-24 02:26:00 +02:00
// Good old-fashioned base-10 number from the lexer.
// We convert the string to an int64.
while (isdigit(*p))
value = 10 * value + (*(p++) - '0');
if (*p++ == '.')
{
while (isdigit(*p))
{
value = 10 * value + (*p++ - '0');
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_scale--;
}
}
2010-10-24 02:26:00 +02:00
*(FB_UINT64*) (literal->litDesc.dsc_address) = value;
}
2001-05-23 15:26:42 +02:00
}
break;
2001-05-23 15:26:42 +02:00
case CONSTANT_DATE:
case CONSTANT_TIME:
case CONSTANT_TIMESTAMP:
{
// Setup the constant's descriptor
2001-05-23 15:26:42 +02:00
2009-01-06 06:53:34 +01:00
switch (numeric_flag)
{
case CONSTANT_DATE:
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_dtype = dtype_sql_date;
break;
case CONSTANT_TIME:
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_dtype = dtype_sql_time;
break;
case CONSTANT_TIMESTAMP:
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_dtype = dtype_timestamp;
break;
}
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_scale = 0;
literal->litDesc.dsc_length = type_lengths[literal->litDesc.dsc_dtype];
literal->litDesc.dsc_address =
FB_NEW(*tdbb->getDefaultPool()) UCHAR[literal->litDesc.dsc_length];
2001-05-23 15:26:42 +02:00
// Set up a descriptor to point to the string
2001-05-23 15:26:42 +02:00
dsc tmp;
tmp.dsc_dtype = dtype_text;
tmp.dsc_scale = 0;
tmp.dsc_flags = 0;
tmp.dsc_ttype() = ttype_ascii;
tmp.dsc_length = static_cast<USHORT>(constant->str_length);
tmp.dsc_address = (UCHAR*) constant->str_data;
2001-05-23 15:26:42 +02:00
// Now invoke the string_to_date/time/timestamp routines
2001-05-23 15:26:42 +02:00
2010-10-24 02:26:00 +02:00
CVT_move(&tmp, &literal->litDesc, ERRD_post);
break;
}
2001-05-23 15:26:42 +02:00
2010-12-18 03:17:06 +01:00
case CONSTANT_BOOLEAN:
DEV_BLKCHK(constant, dsql_type_str);
literal->litDesc.makeBoolean((UCHAR*) constant->str_data);
literal->dsqlStr = constant;
break;
default:
2003-11-04 00:59:24 +01:00
fb_assert(numeric_flag == CONSTANT_STRING);
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(constant, dsql_type_str);
2001-05-23 15:26:42 +02:00
2010-10-24 02:26:00 +02:00
literal->litDesc.dsc_dtype = dtype_text;
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_scale = 0;
literal->litDesc.dsc_length = static_cast<USHORT>(constant->str_length);
literal->litDesc.dsc_address = (UCHAR*) constant->str_data;
literal->litDesc.dsc_ttype() = ttype_dynamic;
// carry a pointer to the constant to resolve character set in pass1
2010-10-24 02:26:00 +02:00
literal->dsqlStr = constant;
break;
2001-05-23 15:26:42 +02:00
}
return node;
}
/**
MAKE_str_constant
@brief Make a constant node when the
character set ID is already known.
@param constant
@param character_set
**/
2003-11-10 10:16:38 +01:00
dsql_nod* MAKE_str_constant(dsql_str* constant, SSHORT character_set)
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
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(constant, dsql_type_str);
2001-05-23 15:26:42 +02:00
2010-10-24 02:26:00 +02:00
LiteralNode* literal = FB_NEW(*tdbb->getDefaultPool()) LiteralNode(*tdbb->getDefaultPool());
literal->litDesc.dsc_dtype = dtype_text;
literal->litDesc.dsc_sub_type = 0;
literal->litDesc.dsc_scale = 0;
literal->litDesc.dsc_length = static_cast<USHORT>(constant->str_length);
literal->litDesc.dsc_address = (UCHAR*) constant->str_data;
literal->litDesc.dsc_ttype() = character_set;
literal->dsqlStr = constant;
dsql_nod* node = MAKE_node(nod_class_exprnode, 1);
node->nod_flags = 0;
node->nod_arg[0] = reinterpret_cast<dsql_nod*>(literal);
2001-05-23 15:26:42 +02:00
return node;
}
/**
MAKE_cstring
@brief Make a string node for a string whose
length is not known, but is null-terminated.
@param str
**/
2003-11-10 10:16:38 +01:00
dsql_str* MAKE_cstring(const char* str)
2001-05-23 15:26:42 +02:00
{
return MAKE_string(str, strlen(str));
2001-05-23 15:26:42 +02:00
}
/**
MAKE_desc
@brief Make a descriptor from input node.
2003-10-05 08:27:16 +02:00
This function can modify node->nod_flags to add NOD_COMP_DIALECT
@param desc
@param node
**/
void MAKE_desc(DsqlCompilerScratch* dsqlScratch, dsc* desc, dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
dsc desc1;
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(node, dsql_type_nod);
2001-05-23 15:26:42 +02:00
// If we already know the datatype, don't worry about anything.
if (node->nod_desc.dsc_dtype)
{
2001-05-23 15:26:42 +02:00
*desc = node->nod_desc;
return;
}
2009-01-06 06:53:34 +01:00
switch (node->nod_type)
{
case nod_class_exprnode:
{
ValueExprNode* exprNode = reinterpret_cast<ValueExprNode*>(node->nod_arg[0]);
2010-12-18 03:17:06 +01:00
if (exprNode->kind == DmlNode::KIND_VALUE)
exprNode->make(dsqlScratch, desc);
}
return;
2001-05-23 15:26:42 +02:00
#ifdef DEV_BUILD
case nod_collate:
ERRD_bugcheck("Not expecting nod_collate in dsql/MAKE_desc");
return;
#endif
case nod_select_expr: // this should come from pass1_any call to set_parameter_type
{
node = node->nod_arg[e_sel_query_spec];
RseNode* rseNode = ExprNode::as<RseNode>(node);
fb_assert(rseNode);
node = rseNode->dsqlSelectList;
fb_assert(node->nod_type == nod_list && node->nod_count > 0);
MAKE_desc(dsqlScratch, desc, node->nod_arg[0]);
}
return;
2001-05-23 15:26:42 +02:00
default:
fb_assert(false); // unexpected dsql_nod type
2001-05-23 15:26:42 +02:00
}
}
/**
MAKE_desc_from_field
@brief Compute a DSC from a field's description information.
@param desc
@param field
**/
2003-10-05 08:27:16 +02:00
void MAKE_desc_from_field(dsc* desc, const dsql_fld* field)
2001-05-23 15:26:42 +02:00
{
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(field, dsql_type_fld);
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
desc->dsc_dtype = static_cast<UCHAR>(field->fld_dtype);
desc->dsc_scale = static_cast<SCHAR>(field->fld_scale);
2001-05-23 15:26:42 +02:00
desc->dsc_sub_type = field->fld_sub_type;
desc->dsc_length = field->fld_length;
desc->dsc_flags = (field->fld_flags & FLD_nullable) ? DSC_nullable : 0;
2009-11-28 20:39:23 +01:00
if (desc->isText() || desc->isBlob())
2005-05-28 00:45:31 +02:00
{
2009-11-28 20:39:23 +01:00
desc->setTextType(INTL_CS_COLL_TO_TTYPE(
field->fld_character_set_id, field->fld_collation_id));
2005-05-28 00:45:31 +02:00
}
2001-05-23 15:26:42 +02:00
}
/**
MAKE_desc_from_list
@brief Make a descriptor from a list of values
according to the sql-standard.
@param desc
@param node
@param expression_name
**/
void MAKE_desc_from_list(DsqlCompilerScratch* dsqlScratch, dsc* desc, dsql_nod* node,
const TEXT* expression_name)
{
Array<const dsc*> args;
2007-04-12 17:56:34 +02:00
fb_assert(node->nod_type == nod_list);
2007-04-12 17:56:34 +02:00
for (dsql_nod** p = node->nod_arg; p < node->nod_arg + node->nod_count; ++p)
{
MAKE_desc(dsqlScratch, &(*p)->nod_desc, *p);
2007-04-12 17:56:34 +02:00
args.add(&(*p)->nod_desc);
}
DSqlDataTypeUtil(dsqlScratch).makeFromList(desc, expression_name, args.getCount(), args.begin());
}
/**
MAKE_field
@brief Make up a field node.
@param context
@param field
@param indices
**/
2003-11-10 10:16:38 +01:00
dsql_nod* MAKE_field(dsql_ctx* context, dsql_fld* field, dsql_nod* indices)
2001-05-23 15:26:42 +02:00
{
2001-12-24 03:51:06 +01:00
DEV_BLKCHK(context, dsql_type_ctx);
DEV_BLKCHK(field, dsql_type_fld);
DEV_BLKCHK(indices, dsql_type_nod);
2001-05-23 15:26:42 +02:00
thread_db* const tdbb = JRD_get_thread_data();
2011-02-18 01:52:10 +01:00
FieldNode* const node = FB_NEW(*tdbb->getDefaultPool()) FieldNode(
*tdbb->getDefaultPool(), context, field, indices);
2010-12-19 22:42:32 +01:00
2009-01-06 06:53:34 +01:00
if (field->fld_dimensions)
{
2009-01-08 10:26:06 +01:00
if (indices)
{
2010-12-19 22:42:32 +01:00
MAKE_desc_from_field(&node->dsqlDesc, field);
node->dsqlDesc.dsc_dtype = static_cast<UCHAR>(field->fld_element_dtype);
node->dsqlDesc.dsc_length = field->fld_element_length;
2009-04-18 16:13:26 +02:00
2010-12-19 22:42:32 +01:00
// node->dsqlDesc.dsc_scale = field->fld_scale;
// node->dsqlDesc.dsc_sub_type = field->fld_sub_type;
2001-05-23 15:26:42 +02:00
}
2009-01-08 10:26:06 +01:00
else
{
2010-12-19 22:42:32 +01:00
node->dsqlDesc.dsc_dtype = dtype_array;
node->dsqlDesc.dsc_length = sizeof(ISC_QUAD);
node->dsqlDesc.dsc_scale = static_cast<SCHAR>(field->fld_scale);
node->dsqlDesc.dsc_sub_type = field->fld_sub_type;
2001-05-23 15:26:42 +02:00
}
}
2009-01-08 10:26:06 +01:00
else
{
if (indices)
{
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-607) <<
Arg::Gds(isc_dsql_only_can_subscript_array) << Arg::Str(field->fld_name));
}
2010-12-19 22:42:32 +01:00
MAKE_desc_from_field(&node->dsqlDesc, field);
2001-05-23 15:26:42 +02:00
}
2008-06-05 13:02:42 +02:00
if ((field->fld_flags & FLD_nullable) || (context->ctx_flags & CTX_outer_join))
2010-12-19 22:42:32 +01:00
node->dsqlDesc.dsc_flags |= DSC_nullable;
2001-05-23 15:26:42 +02:00
// UNICODE_FSS_HACK
2006-10-07 19:06:59 +02:00
// check if the field is a system domain and the type is CHAR/VARCHAR CHARACTER SET UNICODE_FSS
2010-12-19 22:42:32 +01:00
if ((field->fld_flags & FLD_system) && node->dsqlDesc.dsc_dtype <= dtype_varying &&
INTL_GET_CHARSET(&node->dsqlDesc) == CS_METADATA)
2005-05-28 00:45:31 +02:00
{
USHORT adjust = 0;
2005-05-28 00:45:31 +02:00
2010-12-19 22:42:32 +01:00
if (node->dsqlDesc.dsc_dtype == dtype_varying)
adjust = sizeof(USHORT);
2010-12-19 22:42:32 +01:00
else if (node->dsqlDesc.dsc_dtype == dtype_cstring)
adjust = 1;
2005-05-28 00:45:31 +02:00
2010-12-19 22:42:32 +01:00
node->dsqlDesc.dsc_length -= adjust;
node->dsqlDesc.dsc_length *= 3;
node->dsqlDesc.dsc_length += adjust;
2005-05-28 00:45:31 +02:00
}
2010-12-19 22:42:32 +01:00
dsql_nod* nod = MAKE_node(nod_class_exprnode, 1);
nod->nod_arg[0] = reinterpret_cast<dsql_nod*>(node);
return nod;
2001-05-23 15:26:42 +02:00
}
2008-02-28 14:48:16 +01:00
/**
2008-02-28 14:48:16 +01:00
MAKE_field_name
2008-02-28 14:48:16 +01:00
@brief Make up a field name node.
2008-02-28 14:48:16 +01:00
@param field_name
**/
dsql_nod* MAKE_field_name(const char* field_name)
{
dsql_nod* const field_node = MAKE_node(nod_field_name, (int) e_fln_count);
field_node->nod_arg[e_fln_name] = (dsql_nod*) MAKE_cstring(field_name);
return field_node;
}
/**
MAKE_list
@brief Make a list node from a linked list stack of things.
@param stack
**/
dsql_nod* MAKE_list(DsqlNodStack& stack)
2001-05-23 15:26:42 +02:00
{
USHORT count = stack.getCount();
dsql_nod* node = MAKE_node(nod_list, count);
dsql_nod** ptr = node->nod_arg + count;
2001-05-23 15:26:42 +02:00
while (stack.hasData())
{
*--ptr = stack.pop();
}
2001-05-23 15:26:42 +02:00
return node;
}
/**
MAKE_node
@brief Make a node of given type.
@param type
@param count
**/
dsql_nod* MAKE_node(NOD_TYPE type, int count)
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
2008-02-28 14:48:16 +01:00
dsql_nod* node = FB_NEW_RPT(*tdbb->getDefaultPool(), count) dsql_nod;
2001-05-23 15:26:42 +02:00
node->nod_type = type;
node->nod_count = count;
return node;
}
/**
MAKE_parameter
@brief Generate a parameter block for a message. If requested,
set up for a null flag as well.
@param message
@param sqlda_flag
@param null_flag
@param sqlda_index
@param node
**/
2004-02-02 12:02:12 +01:00
dsql_par* MAKE_parameter(dsql_msg* message, bool sqlda_flag, bool null_flag,
USHORT sqlda_index, const dsql_nod* node)
2001-05-23 15:26:42 +02:00
{
2009-11-16 10:18:24 +01:00
if (!message)
{
ERRD_post(Arg::Gds(isc_sqlerr) << Arg::Num(-901) <<
Arg::Gds(isc_badmsgnum));
}
if (sqlda_flag && sqlda_index && sqlda_index <= message->msg_index)
{
// This parameter is possibly already here. Look for it.
for (size_t i = 0; i < message->msg_parameters.getCount(); ++i)
2009-11-16 10:18:24 +01:00
{
dsql_par* temp = message->msg_parameters[i];
if (temp->par_index == sqlda_index)
return temp;
}
}
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
dsql_par* parameter = FB_NEW(*tdbb->getDefaultPool()) dsql_par(*tdbb->getDefaultPool());
2001-05-23 15:26:42 +02:00
parameter->par_message = message;
message->msg_parameters.insert(0, parameter);
2001-05-23 15:26:42 +02:00
parameter->par_parameter = message->msg_parameter++;
2001-05-23 15:26:42 +02:00
parameter->par_rel_name = NULL;
parameter->par_owner_name = NULL;
parameter->par_rel_alias = NULL;
2001-05-23 15:26:42 +02:00
if (node)
MAKE_parameter_names(parameter, node);
2009-04-18 16:13:26 +02:00
// If the parameter is used declared, set SQLDA index
2009-01-08 10:26:06 +01:00
if (sqlda_flag)
{
2009-11-16 10:18:24 +01:00
if (sqlda_index)
{
parameter->par_index = sqlda_index;
2003-12-22 11:00:59 +01:00
if (message->msg_index < sqlda_index)
message->msg_index = sqlda_index;
}
else {
parameter->par_index = ++message->msg_index;
}
}
2009-04-18 16:13:26 +02:00
// If a null handing has been requested, set up a null flag
2001-05-23 15:26:42 +02:00
2009-11-16 10:18:24 +01:00
if (null_flag)
{
dsql_par* null = MAKE_parameter(message, false, false, 0, NULL);
2003-10-05 08:27:16 +02:00
parameter->par_null = null;
2001-05-23 15:26:42 +02:00
null->par_desc.dsc_dtype = dtype_short;
null->par_desc.dsc_scale = 0;
null->par_desc.dsc_length = sizeof(SSHORT);
}
return parameter;
}
/**
MAKE_string
@brief Generalized routine for making a string block.
@param str
@param length
**/
2003-11-10 10:16:38 +01:00
dsql_str* MAKE_string(const char* str, int length)
2001-05-23 15:26:42 +02:00
{
fb_assert(length >= 0);
2001-05-23 15:26:42 +02:00
return MAKE_tagged_string(str, length, NULL);
}
/**
MAKE_tagged_string
@brief Generalized routine for making a string block.
Which is tagged with a character set descriptor.
@param str_
@param length
@param charset
**/
dsql_str* MAKE_tagged_string(const char* strvar, size_t length, const char* charset)
2001-05-23 15:26:42 +02:00
{
2008-02-28 14:48:16 +01:00
thread_db* tdbb = JRD_get_thread_data();
2008-02-28 14:48:16 +01:00
dsql_str* string = FB_NEW_RPT(*tdbb->getDefaultPool(), length) dsql_str;
string->str_charset = charset;
string->str_length = length;
memcpy(string->str_data, strvar, length);
2001-05-23 15:26:42 +02:00
return string;
}
/**
MAKE_parameter_names
@brief Determine relation/column/alias names (if appropriate)
and store them in the given parameter.
@param parameter
@param item
**/
void MAKE_parameter_names(dsql_par* parameter, const dsql_nod* item)
{
fb_assert(parameter && item);
2011-02-17 15:25:56 +01:00
if (item->nod_type == nod_class_exprnode)
2009-01-06 06:53:34 +01:00
{
2011-02-17 15:25:56 +01:00
ValueExprNode* exprNode = reinterpret_cast<ValueExprNode*>(item->nod_arg[0]);
if (exprNode->kind == DmlNode::KIND_VALUE)
2010-12-19 22:42:32 +01:00
exprNode->setParameterName(parameter);
}
}