8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 05:23:02 +01:00
firebird-mirror/src/jrd/PreparedStatement.cpp

442 lines
10 KiB
C++
Raw Normal View History

/*
* The contents of this file are subject to the Initial
* Developer's 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.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* 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 Adriano dos Santos Fernandes
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2008 Adriano dos Santos Fernandes <adrianosf@uol.com.br>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#include "../jrd/PreparedStatement.h"
#include "../jrd/ResultSet.h"
#include "../jrd/align.h"
#include "../jrd/intl.h"
#include "../jrd/jrd.h"
#include "../jrd/dsc.h"
#include "../jrd/req.h"
#include "../dsql/dsql.h"
#include "../common/classes/auto.h"
#include "../dsql/sqlda_pub.h"
#include "../dsql/dsql_proto.h"
#include "../jrd/mov_proto.h"
#include "../jrd/Attachment.h"
using namespace Firebird;
namespace
{
class ParamCmp
{
public:
static int greaterThan(const Jrd::dsql_par* p1, const Jrd::dsql_par* p2)
{
return p1->par_index > p2->par_index;
}
};
}
namespace Jrd {
// Move data from result set message to user variables.
void PreparedStatement::Builder::moveFromResultSet(thread_db* tdbb, ResultSet* rs) const
{
for (Array<Slot>::const_iterator i = outputSlots.begin(); i != outputSlots.end(); ++i)
{
dsc* desc = &rs->getDesc(i->number);
switch (i->type)
{
case TYPE_SLONG:
*(SLONG*) i->address = MOV_get_long(desc, 0);
break;
case TYPE_STRING:
{
string* str = (string*) i->address;
*str = MOV_make_string2(tdbb, desc, CS_NONE);
break;
}
case TYPE_METANAME:
{
MetaName* str = (MetaName*) i->address;
*str = MOV_make_string2(tdbb, desc, CS_METADATA);
break;
}
default:
fb_assert(false);
}
if (i->specifiedAddress && rs->isNull(i->number))
*i->specifiedAddress = false;
}
}
// Move data from user variables to the request input message.
void PreparedStatement::Builder::moveToStatement(thread_db* tdbb, PreparedStatement* stmt) const
{
for (Array<Slot>::const_iterator i = inputSlots.begin(); i != inputSlots.end(); ++i)
{
if (i->specifiedAddress && !*i->specifiedAddress)
{
stmt->setNull(i->number);
continue;
}
dsc desc;
switch (i->type)
{
case TYPE_SLONG:
desc.makeLong(0, (SLONG*) i->address);
break;
case TYPE_STRING:
{
string* str = (string*) i->address;
desc.makeText(str->length(), CS_NONE, (UCHAR*) str->c_str());
break;
}
case TYPE_METANAME:
{
MetaName* str = (MetaName*) i->address;
desc.makeText(str->length(), CS_METADATA, (UCHAR*) str->c_str());
break;
}
default:
fb_assert(false);
}
stmt->setDesc(tdbb, i->number, desc);
}
}
2009-10-31 23:35:48 +01:00
PreparedStatement::PreparedStatement(thread_db* tdbb, MemoryPool& pool,
Attachment* attachment, jrd_tra* transaction, const string& text,
bool isInternalRequest)
2009-10-31 23:35:48 +01:00
: PermanentStorage(pool),
builder(NULL),
inValues(pool),
outValues(pool),
inBlr(pool),
outBlr(pool),
inMessage(pool),
outMessage(pool),
resultSet(NULL)
{
init(tdbb, attachment, transaction, text, isInternalRequest);
}
PreparedStatement::PreparedStatement(thread_db* tdbb, MemoryPool& pool,
Attachment* attachment, jrd_tra* transaction, const Builder& aBuilder,
bool isInternalRequest)
: PermanentStorage(pool),
builder(&aBuilder),
inValues(pool),
outValues(pool),
inBlr(pool),
outBlr(pool),
inMessage(pool),
outMessage(pool),
resultSet(NULL)
{
init(tdbb, attachment, transaction, builder->getText(), isInternalRequest);
}
PreparedStatement::~PreparedStatement()
{
thread_db* tdbb = JRD_get_thread_data();
DSQL_free_statement(tdbb, request, DSQL_drop);
if (resultSet)
resultSet->stmt = NULL;
}
void PreparedStatement::init(thread_db* tdbb, Attachment* attachment, jrd_tra* transaction,
const Firebird::string& text, bool isInternalRequest)
{
AutoSetRestore<SSHORT> autoAttCharset(&attachment->att_charset,
(isInternalRequest ? CS_METADATA : attachment->att_charset));
request = DSQL_allocate_statement(tdbb, attachment);
try
{
2008-04-10 15:06:34 +02:00
const Database& dbb = *tdbb->getDatabase();
const int dialect = isInternalRequest || (dbb.dbb_flags & DBB_DB_SQL_dialect_3) ?
SQL_DIALECT_V6 : SQL_DIALECT_V5;
DSQL_prepare(tdbb, transaction, &request, text.length(), text.c_str(), dialect,
0, NULL, 0, NULL, isInternalRequest);
2009-12-21 00:41:48 +01:00
const DsqlCompiledStatement* statement = request->getStatement();
if (statement->getSendMsg())
parseDsqlMessage(statement->getSendMsg(), inValues, inBlr, inMessage);
if (statement->getReceiveMsg())
parseDsqlMessage(statement->getReceiveMsg(), outValues, outBlr, outMessage);
}
2009-10-31 23:35:48 +01:00
catch (const Exception&)
{
DSQL_free_statement(tdbb, request, DSQL_drop);
throw;
}
}
2009-12-19 16:00:18 +01:00
void PreparedStatement::setDesc(thread_db* tdbb, unsigned param, const dsc& value)
{
2009-12-19 16:00:18 +01:00
fb_assert(param > 0);
// Setup tdbb info necessary for blobs.
AutoSetRestore2<jrd_req*, thread_db> autoRequest(
tdbb, &thread_db::getRequest, &thread_db::setRequest, getRequest()->req_request);
AutoSetRestore<jrd_tra*> autoRequestTrans(&getRequest()->req_request->req_transaction,
tdbb->getTransaction());
MOV_move(tdbb, const_cast<dsc*>(&value), &inValues[(param - 1) * 2]);
const dsc* desc = &inValues[(param - 1) * 2 + 1];
fb_assert(desc->dsc_dtype == dtype_short);
*reinterpret_cast<SSHORT*>(desc->dsc_address) = 0;
}
void PreparedStatement::execute(thread_db* tdbb, jrd_tra* transaction)
{
2008-03-15 18:55:27 +01:00
fb_assert(resultSet == NULL);
if (builder)
builder->moveToStatement(tdbb, this);
DSQL_execute(tdbb, &transaction, request, inBlr.getCount(), inBlr.begin(), 0,
inMessage.getCount(), inMessage.begin(), 0, NULL, /*0,*/ 0, NULL);
}
ResultSet* PreparedStatement::executeQuery(thread_db* tdbb, jrd_tra* transaction)
{
fb_assert(resultSet == NULL && request->getStatement()->getReceiveMsg());
if (builder)
builder->moveToStatement(tdbb, this);
return FB_NEW(getPool()) ResultSet(tdbb, this, transaction);
}
2009-12-19 16:00:18 +01:00
unsigned PreparedStatement::executeUpdate(thread_db* tdbb, jrd_tra* transaction)
{
2009-12-19 16:00:18 +01:00
execute(tdbb, transaction);
return request->req_request->req_records_updated;
}
2009-12-19 16:00:18 +01:00
int PreparedStatement::getResultCount() const
{
2009-12-19 16:00:18 +01:00
return outValues.getCount() / 2;
}
void PreparedStatement::parseDsqlMessage(const dsql_msg* dsqlMsg, Array<dsc>& values,
2009-10-31 23:35:48 +01:00
UCharBuffer& blr, UCharBuffer& msg)
{
// hvlad: Parameters in dsqlMsg->msg_parameters almost always linked in descending
2008-12-05 02:20:14 +01:00
// order by par_index. The only known exception is EXECUTE BLOCK statement.
// To generate correct BLR we must walk params in ascending par_index order.
// So store all params in array in an ascending par_index order despite of
// order in linked list.
2009-10-30 11:50:59 +01:00
// ASF: Input parameters don't come necessarily in ascending or descending order,
// so I changed the code to use a SortedArray.
2009-12-19 16:00:18 +01:00
SortedArray<const dsql_par*, EmptyStorage<const dsql_par*>, const dsql_par*,
DefaultKeyValue<const dsql_par*>, ParamCmp> params;
for (size_t i = 0; i < dsqlMsg->msg_parameters.getCount(); ++i)
2008-04-10 15:06:34 +02:00
{
dsql_par* par = dsqlMsg->msg_parameters[i];
if (par->par_index)
params.add(par);
}
2008-04-10 15:06:34 +02:00
size_t msgLength = 0;
2009-12-19 16:00:18 +01:00
size_t paramCount = params.getCount();
2008-12-05 02:20:14 +01:00
2009-12-20 10:19:15 +01:00
for (size_t i = 0; i < paramCount; ++i)
{
const dsql_par* par = params[i];
2009-12-19 16:00:18 +01:00
2008-04-10 15:06:34 +02:00
#ifdef DEV_BUILD
// Verify that par_index is in ascending order
2008-04-10 15:06:34 +02:00
if (i < paramCount - 1)
{
fb_assert(par->par_index < params[i + 1]->par_index);
2008-04-10 15:06:34 +02:00
}
#endif
2009-12-19 16:00:18 +01:00
if (type_alignments[par->par_desc.dsc_dtype])
msgLength = FB_ALIGN(msgLength, type_alignments[par->par_desc.dsc_dtype]);
msgLength += par->par_desc.dsc_length;
2009-12-19 16:00:18 +01:00
// NULL flag
msgLength = FB_ALIGN(msgLength, type_alignments[dtype_short]);
msgLength += sizeof(SSHORT);
}
paramCount *= 2;
blr.add(blr_version5);
blr.add(blr_begin);
blr.add(blr_message);
blr.add(0);
blr.add(paramCount);
blr.add(paramCount >> 8);
values.resize(paramCount);
msg.resize(msgLength);
msgLength = 0;
2008-04-10 04:35:30 +02:00
dsc* value = values.begin();
2009-12-20 10:19:15 +01:00
for (size_t i = 0; i < paramCount / 2; ++i)
{
const dsql_par* par = params[i];
if (type_alignments[par->par_desc.dsc_dtype])
msgLength = FB_ALIGN(msgLength, type_alignments[par->par_desc.dsc_dtype]);
*value = par->par_desc;
value->dsc_address = msg.begin() + msgLength;
msgLength += par->par_desc.dsc_length;
generateBlr(value, blr);
2009-12-19 16:00:18 +01:00
++value;
// Calculate the NULL indicator offset
if (type_alignments[dtype_short])
msgLength = FB_ALIGN(msgLength, type_alignments[dtype_short]);
value->makeShort(0);
value->dsc_address = msg.begin() + msgLength;
// set NULL indicator value
2008-04-10 04:35:30 +02:00
*((SSHORT*) value->dsc_address) = -1;
msgLength += sizeof(SSHORT);
// Generate BLR for the NULL indicator
generateBlr(value, blr);
2009-12-19 16:00:18 +01:00
++value;
}
blr.add(blr_end);
}
2009-10-31 23:35:48 +01:00
void PreparedStatement::generateBlr(const dsc* desc, UCharBuffer& blr)
{
USHORT length = 0;
switch (desc->dsc_dtype)
{
case dtype_text:
blr.add(blr_text2);
blr.add(desc->dsc_ttype());
blr.add(desc->dsc_ttype() >> 8);
length = desc->dsc_length;
blr.add(length);
blr.add(length >> 8);
break;
case dtype_varying:
blr.add(blr_varying2);
blr.add(desc->dsc_ttype());
blr.add(desc->dsc_ttype() >> 8);
fb_assert(desc->dsc_length >= sizeof(USHORT));
length = desc->dsc_length - sizeof(USHORT);
blr.add(length);
blr.add(length >> 8);
break;
case dtype_short:
blr.add(blr_short);
blr.add(desc->dsc_scale);
break;
case dtype_long:
blr.add(blr_long);
blr.add(desc->dsc_scale);
break;
case dtype_quad:
blr.add(blr_quad);
blr.add(desc->dsc_scale);
break;
case dtype_int64:
blr.add(blr_int64);
blr.add(desc->dsc_scale);
break;
case dtype_real:
blr.add(blr_float);
break;
case dtype_double:
blr.add(blr_double);
break;
case dtype_sql_date:
blr.add(blr_sql_date);
break;
case dtype_sql_time:
blr.add(blr_sql_time);
break;
case dtype_timestamp:
blr.add(blr_timestamp);
break;
case dtype_array:
blr.add(blr_quad);
blr.add(0);
break;
case dtype_blob:
blr.add(blr_blob2);
blr.add(desc->dsc_sub_type);
blr.add(desc->dsc_sub_type >> 8);
blr.add(desc->getTextType());
blr.add(desc->getTextType() >> 8);
break;
default:
fb_assert(false);
}
}
} // namespace