/* * 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 * and all contributors signed below. * * All Rights Reserved. * Contributor(s): ______________________________________. */ #include "../jrd/PreparedStatement.h" #include "../jrd/ResultSet.h" #include "../jrd/align.h" #include "../jrd/jrd.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::const_iterator i = outputSlots.begin(); i != outputSlots.end(); ++i) { switch (i->type) { case TYPE_SSHORT: *(SSHORT*) i->address = rs->getSmallInt(tdbb, i->number); break; case TYPE_SLONG: *(SLONG*) i->address = rs->getInt(tdbb, i->number); break; case TYPE_SINT64: *(SINT64*) i->address = rs->getBigInt(tdbb, i->number); break; case TYPE_DOUBLE: *(double*) i->address = rs->getDouble(tdbb, i->number); break; case TYPE_STRING: { AbstractString* str = (AbstractString*) i->address; str->replace(0, str->length(), rs->getString(tdbb, i->number)); break; } case TYPE_METANAME: *(MetaName*) i->address = rs->getMetaName(tdbb, i->number); 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::const_iterator i = inputSlots.begin(); i != inputSlots.end(); ++i) { if (i->specifiedAddress && !*i->specifiedAddress) { stmt->setNull(i->number); continue; } switch (i->type) { case TYPE_SSHORT: stmt->setSmallInt(tdbb, i->number, *(SSHORT*) i->address); break; case TYPE_SLONG: stmt->setInt(tdbb, i->number, *(SLONG*) i->address); break; case TYPE_SINT64: stmt->setBigInt(tdbb, i->number, *(SINT64*) i->address); break; case TYPE_DOUBLE: stmt->setDouble(tdbb, i->number, *(double*) i->address); break; case TYPE_STRING: stmt->setString(tdbb, i->number, *(AbstractString*) i->address); break; case TYPE_METANAME: stmt->setMetaName(tdbb, i->number, *(MetaName*) i->address); break; default: fb_assert(false); } } } PreparedStatement::PreparedStatement(thread_db* tdbb, MemoryPool& pool, Attachment* attachment, jrd_tra* transaction, const string& text, bool isInternalRequest) : 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 autoAttCharset(&attachment->att_charset, (isInternalRequest ? CS_METADATA : attachment->att_charset)); request = DSQL_allocate_statement(tdbb, attachment); try { 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); const DsqlCompiledStatement* statement = request->getStatement(); if (statement->getSendMsg()) parseDsqlMessage(statement->getSendMsg(), inValues, inBlr, inMessage); if (statement->getReceiveMsg()) parseDsqlMessage(statement->getReceiveMsg(), outValues, outBlr, outMessage); } catch (const Exception&) { DSQL_free_statement(tdbb, request, DSQL_drop); throw; } } void PreparedStatement::setDesc(thread_db* tdbb, unsigned param, const dsc& value) { fb_assert(param > 0); // Setup tdbb info necessary for blobs. AutoSetRestore2 autoRequest( tdbb, &thread_db::getRequest, &thread_db::setRequest, getRequest()->req_request); AutoSetRestore autoRequestTrans(&getRequest()->req_request->req_transaction, tdbb->getTransaction()); MOV_move(tdbb, const_cast(&value), &inValues[(param - 1) * 2]); const dsc* desc = &inValues[(param - 1) * 2 + 1]; fb_assert(desc->dsc_dtype == dtype_short); *reinterpret_cast(desc->dsc_address) = 0; } void PreparedStatement::execute(thread_db* tdbb, jrd_tra* transaction) { 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); } unsigned PreparedStatement::executeUpdate(thread_db* tdbb, jrd_tra* transaction) { execute(tdbb, transaction); return request->req_request->req_records_updated; } int PreparedStatement::getResultCount() const { return outValues.getCount() / 2; } void PreparedStatement::parseDsqlMessage(const dsql_msg* dsqlMsg, Array& values, UCharBuffer& blr, UCharBuffer& msg) { // hvlad: Parameters in dsqlMsg->msg_parameters almost always linked in descending // 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. // ASF: Input parameters don't come necessarily in ascending or descending order, // so I changed the code to use a SortedArray. SortedArray, const dsql_par*, DefaultKeyValue, ParamCmp> params; for (size_t i = 0; i < dsqlMsg->msg_parameters.getCount(); ++i) { dsql_par* par = dsqlMsg->msg_parameters[i]; if (par->par_index) params.add(par); } size_t msgLength = 0; size_t paramCount = params.getCount(); for (size_t i = 0; i < paramCount; ++i) { const dsql_par* par = params[i]; #ifdef DEV_BUILD // Verify that par_index is in ascending order if (i < paramCount - 1) { fb_assert(par->par_index < params[i + 1]->par_index); } #endif 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; // 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; dsc* value = values.begin(); 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); ++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 *((SSHORT*) value->dsc_address) = -1; msgLength += sizeof(SSHORT); // Generate BLR for the NULL indicator generateBlr(value, blr); ++value; } blr.add(blr_end); } 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