8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 18:03:04 +01:00
firebird-mirror/src/jrd/ExtEngineManager.cpp

1501 lines
40 KiB
C++

/*
* 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 "firebird.h"
#include "consts_pub.h"
#include "iberror.h"
#include "inf_pub.h"
#include "../jrd/ExtEngineManager.h"
#include "../dsql/sqlda_pub.h"
#include "../common/dsc.h"
#include "../jrd/align.h"
#include "../jrd/jrd.h"
#include "../jrd/exe.h"
#include "../jrd/req.h"
#include "../jrd/status.h"
#include "../jrd/tra.h"
#include "../dsql/StmtNodes.h"
#include "../common/os/path_utils.h"
#include "../jrd/cmp_proto.h"
#include "../jrd/cvt_proto.h"
#include "../jrd/evl_proto.h"
#include "../jrd/intl_proto.h"
#include "../jrd/met_proto.h"
#include "../jrd/mov_proto.h"
#include "../jrd/par_proto.h"
#include "../jrd/Function.h"
#include "../common/isc_proto.h"
#include "../common/classes/auto.h"
#include "../common/classes/fb_string.h"
#include "../common/classes/init.h"
#include "../common/classes/objects_array.h"
#include "../common/config/config.h"
#include "../common/ScanDir.h"
#include "../common/utils_proto.h"
#include "../common/classes/GetPlugins.h"
using namespace Firebird;
using namespace Jrd;
namespace
{
// External message node.
class ExtMessageNode : public MessageNode
{
public:
ExtMessageNode(thread_db* tdbb, MemoryPool& pool, CompilerScratch* csb, USHORT message,
Array<NestConst<Parameter> >& aParameters, const Format* aFormat)
: MessageNode(pool),
parameters(aParameters),
format(aFormat),
isSpecial(pool)
{
setup(tdbb, csb, message, format->fmt_count);
}
virtual USHORT setupDesc(thread_db* tdbb, CompilerScratch* csb, USHORT index,
dsc* desc, ItemInfo* itemInfo)
{
*desc = format->fmt_desc[index];
if (index % 2 == 0 && index / 2u < parameters.getCount())
{
const Parameter* param = parameters[index / 2];
if (param->prm_mechanism != prm_mech_type_of &&
!fb_utils::implicit_domain(param->prm_field_source.c_str()))
{
MetaNamePair namePair(param->prm_field_source, "");
FieldInfo fieldInfo;
bool exist = csb->csb_map_field_info.get(namePair, fieldInfo);
MET_get_domain(tdbb, csb->csb_pool, param->prm_field_source, desc,
(exist ? NULL : &fieldInfo));
if (!exist)
csb->csb_map_field_info.put(namePair, fieldInfo);
itemInfo->field = namePair;
itemInfo->nullable = fieldInfo.nullable;
itemInfo->fullDomain = true;
}
itemInfo->name = param->prm_name;
if (!param->prm_nullable)
itemInfo->nullable = false;
isSpecial.getBuffer(index / 2 + 1)[index / 2] = itemInfo->isSpecial();
}
return type_alignments[desc->dsc_dtype];
}
virtual const StmtNode* execute(thread_db* tdbb, jrd_req* request, ExeState* exeState) const
{
if (request->req_operation == jrd_req::req_evaluate)
{
// Clear the message. This is important for external routines.
UCHAR* msg = request->getImpure<UCHAR>(impureOffset);
memset(msg, 0, format->fmt_length);
}
return MessageNode::execute(tdbb, request, exeState);
}
public:
Array<NestConst<Parameter> >& parameters;
const Format* format;
Array<bool> isSpecial;
};
// Initialize output parameters with their domains default value or NULL.
// Kind of blr_init_variable, but for parameters.
class ExtInitParameterNode : public TypedNode<StmtNode, StmtNode::TYPE_EXT_INIT_PARAMETER>
{
public:
ExtInitParameterNode(thread_db* tdbb, MemoryPool& pool, CompilerScratch* csb,
ExtMessageNode* aMessage, USHORT aArgNumber)
: TypedNode<StmtNode, StmtNode::TYPE_EXT_INIT_PARAMETER>(pool),
message(aMessage),
argNumber(aArgNumber)
{
Parameter* parameter = message->parameters[argNumber / 2];
if (parameter->prm_mechanism == prm_mech_type_of ||
fb_utils::implicit_domain(parameter->prm_field_source.c_str()) ||
!parameter->prm_default_value)
{
defaultValueNode = NULL;
}
else
defaultValueNode = CMP_clone_node(tdbb, csb, parameter->prm_default_value);
}
string internalPrint(NodePrinter& printer) const
{
StmtNode::internalPrint(printer);
NODE_PRINT(printer, message);
NODE_PRINT(printer, argNumber);
NODE_PRINT(printer, defaultValueNode);
return "ExtInitParameterNode";
}
void genBlr(DsqlCompilerScratch* /*dsqlScratch*/)
{
}
ExtInitParameterNode* pass1(thread_db* tdbb, CompilerScratch* csb)
{
doPass1(tdbb, csb, &defaultValueNode);
return this;
}
ExtInitParameterNode* pass2(thread_db* tdbb, CompilerScratch* csb)
{
ExprNode::doPass2(tdbb, csb, &defaultValueNode);
return this;
}
const StmtNode* execute(thread_db* tdbb, jrd_req* request, ExeState* /*exeState*/) const
{
if (request->req_operation == jrd_req::req_evaluate)
{
dsc* defaultDesc = NULL;
if (defaultValueNode)
{
defaultDesc = EVL_expr(tdbb, request, defaultValueNode);
if (request->req_flags & req_null)
defaultDesc = NULL;
}
if (defaultDesc)
{
// Initialize the value. The null flag is already initialized to not-null
// by the ExtMessageNode.
dsc desc = message->format->fmt_desc[argNumber];
desc.dsc_address = request->getImpure<UCHAR>(
message->impureOffset + (IPTR) desc.dsc_address);
MOV_move(tdbb, defaultDesc, &desc);
}
else
{
SSHORT tempValue = -1;
dsc temp;
temp.makeShort(0, &tempValue);
dsc desc = message->format->fmt_desc[argNumber + 1];
desc.dsc_address = request->getImpure<UCHAR>(
message->impureOffset + (IPTR) desc.dsc_address);
MOV_move(tdbb, &temp, &desc);
}
request->req_operation = jrd_req::req_return;
}
return parentStmt;
}
private:
ExtMessageNode* message;
USHORT argNumber;
ValueExprNode* defaultValueNode;
};
// External output parameters initialization.
class ExtInitOutputNode : public CompoundStmtNode
{
public:
ExtInitOutputNode(thread_db* tdbb, MemoryPool& pool, CompilerScratch* csb,
ExtMessageNode* message)
: CompoundStmtNode(pool)
{
// Iterate over the format items, except the EOF item.
for (USHORT i = 0; i < (message->format->fmt_count / 2) * 2; i += 2)
{
ExtInitParameterNode* init = FB_NEW_POOL(pool) ExtInitParameterNode(
tdbb, pool, csb, message, i);
statements.add(init);
}
}
};
// External parameters validation.
class ExtValidationNode : public CompoundStmtNode
{
public:
ExtValidationNode(MemoryPool& pool, ExtMessageNode* message, bool procedure, bool input)
: CompoundStmtNode(pool)
{
// Iterate over the format items, except the EOF item.
for (USHORT i = 0; i < (message->format->fmt_count / 2) * 2; i += 2)
{
if (!message->isSpecial[i / 2])
continue;
ParameterNode* flag = FB_NEW_POOL(pool) ParameterNode(pool);
flag->message = message;
flag->argNumber = i + 1;
ParameterNode* param = FB_NEW_POOL(pool) ParameterNode(pool);
param->message = message;
param->argNumber = i;
param->argFlag = flag;
AssignmentNode* assign = FB_NEW_POOL(pool) AssignmentNode(pool);
assign->asgnFrom = param;
statements.add(assign);
// It's sufficient to assign input parameters to NULL, but output parameters
// need to be assigned to themselves to validate correctly.
if (input)
assign->asgnTo = FB_NEW_POOL(pool) NullNode(pool);
else
{
flag = FB_NEW_POOL(pool) ParameterNode(pool);
flag->message = message;
flag->argNumber = i + 1;
param = FB_NEW_POOL(pool) ParameterNode(pool);
param->message = message;
param->argNumber = i;
param->argFlag = flag;
assign->asgnTo = param;
}
}
// A stall is required to read req_proc_fetch state correctly.
if (procedure)
statements.add(FB_NEW_POOL(pool) StallNode(pool));
}
};
// External function node.
class ExtFunctionNode : public SuspendNode
{
public:
ExtFunctionNode(MemoryPool& pool, const ReceiveNode* aReceiveNode,
const ExtEngineManager::Function* aFunction)
: SuspendNode(pool),
receiveNode(aReceiveNode),
function(aFunction)
{
}
virtual const StmtNode* execute(thread_db* tdbb, jrd_req* request, ExeState* exeState) const
{
if (request->req_operation == jrd_req::req_evaluate)
{
UCHAR* inMsg = receiveNode ?
request->getImpure<UCHAR>(receiveNode->message->impureOffset) : NULL;
UCHAR* outMsg = request->getImpure<UCHAR>(message->impureOffset);
function->execute(tdbb, inMsg, outMsg);
}
return SuspendNode::execute(tdbb, request, exeState);
}
private:
const ReceiveNode* receiveNode;
const ExtEngineManager::Function* function;
};
// External procedure node.
class ExtProcedureNode : public SuspendNode
{
public:
ExtProcedureNode(MemoryPool& pool, const ReceiveNode* aReceiveNode,
const ExtEngineManager::Procedure* aProcedure)
: SuspendNode(pool),
receiveNode(aReceiveNode),
procedure(aProcedure)
{
}
virtual const StmtNode* execute(thread_db* tdbb, jrd_req* request, ExeState* exeState) const
{
ExtEngineManager::ResultSet*& resultSet = request->req_ext_resultset;
UCHAR* inMsg = receiveNode ?
request->getImpure<UCHAR>(receiveNode->message->impureOffset) : NULL;
UCHAR* outMsg = request->getImpure<UCHAR>(message->impureOffset);
USHORT* eof = (USHORT*) (outMsg + (IPTR) message->format->fmt_desc.back().dsc_address);
switch (request->req_operation)
{
case jrd_req::req_evaluate:
fb_assert(!resultSet && *eof == 0);
resultSet = procedure->open(tdbb, inMsg, outMsg);
if (resultSet)
*eof = -1;
else
{
if (!(request->req_flags & req_proc_fetch))
{
request->req_operation = jrd_req::req_evaluate;
return statement;
}
}
request->req_operation = jrd_req::req_return;
// fall into
case jrd_req::req_return:
if (*eof == 0)
{
fb_assert(!resultSet);
return parentStmt;
}
fb_assert(resultSet);
if (!resultSet->fetch(tdbb) || !(request->req_flags & req_proc_fetch))
{
*eof = 0;
delete resultSet;
resultSet = NULL;
}
break;
case jrd_req::req_proceed:
request->req_operation = jrd_req::req_evaluate;
return statement;
case jrd_req::req_unwind:
delete resultSet;
resultSet = NULL;
break;
}
return SuspendNode::execute(tdbb, request, exeState);
}
private:
const ReceiveNode* receiveNode;
const ExtEngineManager::Procedure* procedure;
};
// External trigger node.
class ExtTriggerNode : public TypedNode<StmtNode, StmtNode::TYPE_EXT_TRIGGER>
{
public:
ExtTriggerNode(MemoryPool& pool, const ExtEngineManager::Trigger* aTrigger)
: TypedNode<StmtNode, StmtNode::TYPE_EXT_TRIGGER>(pool),
trigger(aTrigger)
{
}
string internalPrint(NodePrinter& printer) const
{
StmtNode::internalPrint(printer);
return "ExtTriggerNode";
}
void genBlr(DsqlCompilerScratch* /*dsqlScratch*/)
{
}
ExtTriggerNode* pass1(thread_db* /*tdbb*/, CompilerScratch* /*csb*/)
{
return this;
}
ExtTriggerNode* pass2(thread_db* /*tdbb*/, CompilerScratch* /*csb*/)
{
return this;
}
const StmtNode* execute(thread_db* tdbb, jrd_req* request, ExeState* /*exeState*/) const
{
if (request->req_operation == jrd_req::req_evaluate)
{
trigger->execute(tdbb, request->req_trigger_action,
getRpb(request, 0), getRpb(request, 1));
request->req_operation = jrd_req::req_return;
}
return parentStmt;
}
private:
static record_param* getRpb(jrd_req* request, USHORT n)
{
return request->req_rpb.getCount() > n && request->req_rpb[n].rpb_number.isValid() ?
&request->req_rpb[n] : NULL;
}
private:
const ExtEngineManager::Trigger* trigger;
};
}
namespace Jrd {
template <typename T> class ExtEngineManager::ContextManager
{
public:
ContextManager(thread_db* tdbb, EngineAttachmentInfo* aAttInfo, T* obj,
CallerName aCallerName = CallerName())
: attInfo(aAttInfo),
attachment(tdbb->getAttachment()),
transaction(tdbb->getTransaction()),
charSet(attachment->att_charset),
attInUse(attachment->att_in_use),
traInUse(transaction ? transaction->tra_in_use : false)
{
// !!!!! needs async lock to be safe
attachment->att_in_use = true;
if (transaction)
{
callerName = transaction->tra_caller_name;
transaction->tra_caller_name = aCallerName;
++transaction->tra_callback_count;
transaction->tra_in_use = true;
}
attInfo->context->setTransaction(tdbb);
setCharSet(tdbb, attInfo, obj);
}
ContextManager(thread_db* tdbb, EngineAttachmentInfo* aAttInfo, USHORT aCharSet,
CallerName aCallerName = CallerName())
: attInfo(aAttInfo),
attachment(tdbb->getAttachment()),
transaction(tdbb->getTransaction()),
charSet(attachment->att_charset),
attInUse(attachment->att_in_use),
traInUse(transaction ? transaction->tra_in_use : false)
{
attachment->att_charset = aCharSet;
// !!!!! needs async lock to be safe
attachment->att_in_use = true;
if (transaction)
{
callerName = transaction->tra_caller_name;
transaction->tra_caller_name = aCallerName;
++transaction->tra_callback_count;
transaction->tra_in_use = true;
}
attInfo->context->setTransaction(tdbb);
}
~ContextManager()
{
if (transaction)
{
--transaction->tra_callback_count;
transaction->tra_in_use = traInUse;
transaction->tra_caller_name = callerName;
}
// !!!!! needs async lock to be safe
attachment->att_in_use = attInUse;
attachment->att_charset = charSet;
}
private:
void setCharSet(thread_db* tdbb, EngineAttachmentInfo* attInfo, T* obj)
{
attachment->att_charset = attInfo->adminCharSet;
if (!obj)
return;
char charSetName[MAX_SQL_IDENTIFIER_SIZE];
{ // scope
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
obj->getCharSet(&status, attInfo->context, charSetName, MAX_SQL_IDENTIFIER_LEN);
status.check();
charSetName[MAX_SQL_IDENTIFIER_LEN] = '\0';
}
USHORT charSetId;
if (!MET_get_char_coll_subtype(tdbb, &charSetId,
reinterpret_cast<const UCHAR*>(charSetName), static_cast<USHORT>(strlen(charSetName))))
{
status_exception::raise(Arg::Gds(isc_charset_not_found) << Arg::Str(charSetName));
}
attachment->att_charset = charSetId;
}
private:
EngineAttachmentInfo* attInfo;
Jrd::Attachment* attachment;
jrd_tra* transaction;
// These data members are to restore the original information.
const USHORT charSet;
const bool attInUse;
const bool traInUse;
CallerName callerName;
};
//---------------------
ExtEngineManager::ExternalContextImpl::ExternalContextImpl(thread_db* tdbb,
IExternalEngine* aEngine)
: engine(aEngine),
internalAttachment(tdbb->getAttachment()),
internalTransaction(NULL),
externalAttachment(NULL),
externalTransaction(NULL),
miscInfo(*internalAttachment->att_pool)
{
//// TODO: admin rights
clientCharSet = INTL_charset_lookup(tdbb, internalAttachment->att_client_charset)->getName();
internalAttachment->getStable()->addRef();
externalAttachment = MasterInterfacePtr()->registerAttachment(JProvider::getInstance(),
internalAttachment->getInterface());
}
ExtEngineManager::ExternalContextImpl::~ExternalContextImpl()
{
releaseTransaction();
if (externalAttachment)
{
externalAttachment->release();
externalAttachment = NULL;
}
}
void ExtEngineManager::ExternalContextImpl::releaseTransaction()
{
if (externalTransaction)
{
externalTransaction->release();
externalTransaction = NULL;
}
internalTransaction = NULL;
}
void ExtEngineManager::ExternalContextImpl::setTransaction(thread_db* tdbb)
{
jrd_tra* newTransaction = tdbb->getTransaction();
if (newTransaction == internalTransaction)
return;
releaseTransaction();
fb_assert(!externalTransaction);
if ((internalTransaction = newTransaction))
{
internalTransaction->getInterface()->addRef();
externalTransaction = MasterInterfacePtr()->registerTransaction(externalAttachment,
internalTransaction->getInterface());
}
}
IMaster* ExtEngineManager::ExternalContextImpl::getMaster()
{
MasterInterfacePtr master;
return master;
}
IExternalEngine* ExtEngineManager::ExternalContextImpl::getEngine(CheckStatusWrapper* /*status*/)
{
return engine;
}
Firebird::IAttachment* ExtEngineManager::ExternalContextImpl::getAttachment(
CheckStatusWrapper* /*status*/)
{
externalAttachment->addRef();
return externalAttachment;
}
Firebird::ITransaction* ExtEngineManager::ExternalContextImpl::getTransaction(
CheckStatusWrapper* /*status*/)
{
externalTransaction->addRef();
return externalTransaction;
}
const char* ExtEngineManager::ExternalContextImpl::getUserName()
{
return internalAttachment->att_user->getUserName().c_str();
}
const char* ExtEngineManager::ExternalContextImpl::getDatabaseName()
{
return internalAttachment->att_database->dbb_database_name.c_str();
}
const char* ExtEngineManager::ExternalContextImpl::getClientCharSet()
{
return clientCharSet.c_str();
}
int ExtEngineManager::ExternalContextImpl::obtainInfoCode()
{
static AtomicCounter counter;
return ++counter;
}
void* ExtEngineManager::ExternalContextImpl::getInfo(int code)
{
void* value = NULL;
miscInfo.get(code, value);
return value;
}
void* ExtEngineManager::ExternalContextImpl::setInfo(int code, void* value)
{
void* oldValue = getInfo(code);
miscInfo.put(code, value);
return oldValue;
}
//---------------------
ExtEngineManager::Function::Function(thread_db* tdbb, ExtEngineManager* aExtManager,
IExternalEngine* aEngine, RoutineMetadata* aMetadata, IExternalFunction* aFunction,
const Jrd::Function* aUdf)
: extManager(aExtManager),
engine(aEngine),
metadata(aMetadata),
function(aFunction),
udf(aUdf),
database(tdbb->getDatabase())
{
}
ExtEngineManager::Function::~Function()
{
//Database::Checkout dcoHolder(database);
function->dispose();
}
void ExtEngineManager::Function::execute(thread_db* tdbb, UCHAR* inMsg, UCHAR* outMsg) const
{
EngineAttachmentInfo* attInfo = extManager->getEngineAttachment(tdbb, engine);
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, function,
(udf->getName().package.isEmpty() ?
CallerName(obj_udf, udf->getName().identifier) :
CallerName(obj_package_header, udf->getName().package)));
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
function->execute(&status, attInfo->context, inMsg, outMsg);
status.check();
}
//---------------------
ExtEngineManager::Procedure::Procedure(thread_db* tdbb, ExtEngineManager* aExtManager,
IExternalEngine* aEngine, RoutineMetadata* aMetadata, IExternalProcedure* aProcedure,
const jrd_prc* aPrc)
: extManager(aExtManager),
engine(aEngine),
metadata(aMetadata),
procedure(aProcedure),
prc(aPrc),
database(tdbb->getDatabase())
{
}
ExtEngineManager::Procedure::~Procedure()
{
//Database::Checkout dcoHolder(database);
procedure->dispose();
}
ExtEngineManager::ResultSet* ExtEngineManager::Procedure::open(thread_db* tdbb,
UCHAR* inMsg, UCHAR* outMsg) const
{
return FB_NEW_POOL(*tdbb->getDefaultPool()) ResultSet(tdbb, inMsg, outMsg, this);
}
//---------------------
ExtEngineManager::ResultSet::ResultSet(thread_db* tdbb, UCHAR* inMsg, UCHAR* outMsg,
const ExtEngineManager::Procedure* aProcedure)
: procedure(aProcedure),
attachment(tdbb->getAttachment()),
firstFetch(true)
{
attInfo = procedure->extManager->getEngineAttachment(tdbb, procedure->engine);
ContextManager<IExternalProcedure> ctxManager(tdbb, attInfo, procedure->procedure,
(procedure->prc->getName().package.isEmpty() ?
CallerName(obj_procedure, procedure->prc->getName().identifier) :
CallerName(obj_package_header, procedure->prc->getName().package)));
charSet = attachment->att_charset;
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
resultSet = procedure->procedure->open(&status, attInfo->context, inMsg, outMsg);
status.check();
}
ExtEngineManager::ResultSet::~ResultSet()
{
if (resultSet)
{
EngineCheckout cout(JRD_get_thread_data(), FB_FUNCTION);
resultSet->dispose();
}
}
bool ExtEngineManager::ResultSet::fetch(thread_db* tdbb)
{
bool wasFirstFetch = firstFetch;
firstFetch = false;
if (!resultSet)
return wasFirstFetch;
ContextManager<IExternalProcedure> ctxManager(tdbb, attInfo, charSet,
(procedure->prc->getName().package.isEmpty() ?
CallerName(obj_procedure, procedure->prc->getName().identifier) :
CallerName(obj_package_header, procedure->prc->getName().package)));
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
bool ret = resultSet->fetch(&status);
status.check();
return ret;
}
//---------------------
ExtEngineManager::Trigger::Trigger(thread_db* tdbb, MemoryPool& pool, ExtEngineManager* aExtManager,
IExternalEngine* aEngine, RoutineMetadata* aMetadata,
IExternalTrigger* aTrigger, const Jrd::Trigger* aTrg)
: extManager(aExtManager),
engine(aEngine),
metadata(aMetadata),
trigger(aTrigger),
trg(aTrg),
fieldsPos(pool),
database(tdbb->getDatabase())
{
dsc shortDesc;
shortDesc.makeShort(0);
jrd_rel* relation = trg->relation;
if (relation)
{
GenericMap<Left<MetaName, USHORT> > fieldsMap;
for (FB_SIZE_T i = 0; i < relation->rel_fields->count(); ++i)
{
jrd_fld* field = (*relation->rel_fields)[i];
if (field)
fieldsMap.put(field->fld_name, (USHORT) i);
}
format = Routine::createFormat(pool, metadata->triggerFields, false);
FbLocalStatus status;
for (unsigned i = 0; i < format->fmt_count / 2u; ++i)
{
const char* fieldName = metadata->triggerFields->getField(&status, i);
status.check();
USHORT pos;
if (!fieldsMap.get(fieldName, pos))
fb_assert(false);
else
fieldsPos.add(pos);
}
}
}
ExtEngineManager::Trigger::~Trigger()
{
// hvlad: shouldn't we call trigger->dispose() here ?
}
void ExtEngineManager::Trigger::execute(thread_db* tdbb, unsigned action,
record_param* oldRpb, record_param* newRpb) const
{
EngineAttachmentInfo* attInfo = extManager->getEngineAttachment(tdbb, engine);
ContextManager<IExternalTrigger> ctxManager(tdbb, attInfo, trigger,
CallerName(obj_trigger, trg->name));
// ASF: Using Array instead of HalfStaticArray to not need to do alignment hacks here.
Array<UCHAR> oldMsg;
Array<UCHAR> newMsg;
if (oldRpb)
setValues(tdbb, oldMsg, oldRpb);
if (newRpb)
setValues(tdbb, newMsg, newRpb);
{ // scope
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
trigger->execute(&status, attInfo->context, action,
(oldMsg.hasData() ? oldMsg.begin() : NULL), (newMsg.hasData() ? newMsg.begin() : NULL));
status.check();
}
if (newRpb)
{
// Move data back from the message to the record.
Record* record = newRpb->rpb_record;
UCHAR* p = newMsg.begin();
for (unsigned i = 0; i < format->fmt_count / 2u; ++i)
{
USHORT fieldPos = fieldsPos[i];
dsc target;
bool readonly = !EVL_field(newRpb->rpb_relation, record, fieldPos, &target) &&
target.dsc_address && !(target.dsc_flags & DSC_null);
if (!readonly)
{
SSHORT* nullSource = (SSHORT*) (p + (IPTR) format->fmt_desc[i * 2 + 1].dsc_address);
if (*nullSource == 0)
{
dsc source = format->fmt_desc[i * 2];
source.dsc_address += (IPTR) p;
MOV_move(tdbb, &source, &target);
record->clearNull(fieldPos);
}
else
record->setNull(fieldPos);
}
}
}
}
void ExtEngineManager::Trigger::setValues(thread_db* tdbb, Array<UCHAR>& msgBuffer,
record_param* rpb) const
{
if (!rpb || !rpb->rpb_record)
return;
UCHAR* p = msgBuffer.getBuffer(format->fmt_length);
memset(p, 0, format->fmt_length);
for (unsigned i = 0; i < format->fmt_count / 2u; ++i)
{
USHORT fieldPos = fieldsPos[i];
dsc source;
EVL_field(rpb->rpb_relation, rpb->rpb_record, fieldPos, &source);
// CVC: I'm not sure why it's not important to check EVL_field's result.
SSHORT* nullTarget = (SSHORT*) (p + (IPTR) format->fmt_desc[i * 2 + 1].dsc_address);
*nullTarget = (source.dsc_flags & DSC_null) != 0 ? -1 : 0;
if (*nullTarget == 0)
{
dsc target = format->fmt_desc[i * 2];
target.dsc_address += (IPTR) p;
MOV_move(tdbb, &source, &target);
}
}
}
//---------------------
ExtEngineManager::~ExtEngineManager()
{
fb_assert(enginesAttachments.count() == 0);
/*
AP: Commented out this code due to later AV.
When engine is released, it does dlclose() plugin module (libudr_engine.so),
but that module is not actually unloaded - because UDR module (libudrcpp_example.so) is using
symbols from plugin module, therefore raising plugin module's reference count.
UDR module can be unloaded only from plugin module's global variable (ModuleMap modules) dtor,
which is not called as long as plugin module is not unloaded. As the result all this will be
unloaded only on program exit, causing at that moment AV if this code is active: it happens that
~ModuleMap dlcloses itself.
PluginManagerInterfacePtr pi;
EnginesMap::Accessor accessor(&engines);
for (bool found = accessor.getFirst(); found; found = accessor.getNext())
{
IExternalEngine* engine = accessor.current()->second;
pi->releasePlugin(engine);
}
*/
}
//---------------------
void ExtEngineManager::initialize()
{
}
void ExtEngineManager::closeAttachment(thread_db* tdbb, Attachment* attachment)
{
Array<IExternalEngine*> enginesCopy;
{ // scope
ReadLockGuard readGuard(enginesLock, FB_FUNCTION);
EnginesMap::Accessor accessor(&engines);
for (bool found = accessor.getFirst(); found; found = accessor.getNext())
enginesCopy.add(accessor.current()->second);
}
RefDeb(DEB_RLS_JATT, "ExtEngineManager::closeAttachment");
EngineCheckout cout(tdbb, FB_FUNCTION, true);
for (Array<IExternalEngine*>::iterator i = enginesCopy.begin(); i != enginesCopy.end(); ++i)
{
IExternalEngine* engine = *i;
EngineAttachmentInfo* attInfo = getEngineAttachment(tdbb, engine, true);
if (attInfo)
{
{ // scope
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, attInfo->adminCharSet);
FbLocalStatus status;
engine->closeAttachment(&status, attInfo->context); //// FIXME: log status
}
delete attInfo;
}
}
}
void ExtEngineManager::makeFunction(thread_db* tdbb, CompilerScratch* csb, Jrd::Function* udf,
const MetaName& engine, const string& entryPoint, const string& body)
{
string entryPointTrimmed = entryPoint;
entryPointTrimmed.trim();
EngineAttachmentInfo* attInfo = getEngineAttachment(tdbb, engine);
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, attInfo->adminCharSet,
(udf->getName().package.isEmpty() ?
CallerName(obj_udf, udf->getName().identifier) :
CallerName(obj_package_header, udf->getName().package)));
///MemoryPool& pool = *tdbb->getDefaultPool();
MemoryPool& pool = *getDefaultMemoryPool();
AutoPtr<RoutineMetadata> metadata(FB_NEW_POOL(pool) RoutineMetadata(pool));
metadata->package = udf->getName().package;
metadata->name = udf->getName().identifier;
metadata->entryPoint = entryPointTrimmed;
metadata->body = body;
metadata->inputParameters = Routine::createMetadata(udf->getInputFields());
metadata->outputParameters = Routine::createMetadata(udf->getOutputFields());
FbLocalStatus status;
RefPtr<IMetadataBuilder> inBuilder(REF_NO_INCR, metadata->inputParameters->getBuilder(&status));
status.check();
RefPtr<IMetadataBuilder> outBuilder(REF_NO_INCR, metadata->outputParameters->getBuilder(&status));
status.check();
IExternalFunction* externalFunction;
{ // scope
EngineCheckout cout(tdbb, FB_FUNCTION);
externalFunction = attInfo->engine->makeFunction(&status, attInfo->context, metadata,
inBuilder, outBuilder);
status.check();
if (!externalFunction)
{
status_exception::raise(
Arg::Gds(isc_eem_func_not_returned) << udf->getName().toString() << engine);
}
metadata->inputParameters = inBuilder->getMetadata(&status);
status.check();
metadata->outputParameters = outBuilder->getMetadata(&status);
status.check();
}
udf->setInputFormat(Routine::createFormat(pool, metadata->inputParameters, false));
udf->setOutputFormat(Routine::createFormat(pool, metadata->outputParameters, true));
try
{
udf->fun_external = FB_NEW_POOL(getPool()) Function(tdbb, this, attInfo->engine,
metadata.release(), externalFunction, udf);
CompoundStmtNode* mainNode = FB_NEW_POOL(getPool()) CompoundStmtNode(getPool());
ExtMessageNode* inMessageNode = udf->getInputFields().hasData() ?
FB_NEW_POOL(getPool()) ExtMessageNode(tdbb, getPool(), csb, 0,
udf->getInputFields(), udf->getInputFormat()) :
NULL;
if (inMessageNode)
mainNode->statements.add(inMessageNode);
ExtMessageNode* outMessageNode = FB_NEW_POOL(getPool()) ExtMessageNode(tdbb, getPool(), csb, 1,
udf->getOutputFields(), udf->getOutputFormat());
mainNode->statements.add(outMessageNode);
ExtInitOutputNode* initOutputNode = FB_NEW_POOL(getPool()) ExtInitOutputNode(
tdbb, getPool(), csb, outMessageNode);
mainNode->statements.add(initOutputNode);
ReceiveNode* receiveNode = inMessageNode ?
FB_NEW_POOL(getPool()) ReceiveNode(getPool()) : NULL;
if (inMessageNode)
{
receiveNode->message = inMessageNode;
receiveNode->statement = FB_NEW_POOL(getPool()) ExtValidationNode(
getPool(), inMessageNode, false, true);
mainNode->statements.add(receiveNode);
}
ExtFunctionNode* extFunctionNode = FB_NEW_POOL(getPool()) ExtFunctionNode(getPool(),
receiveNode, udf->fun_external);
mainNode->statements.add(extFunctionNode);
extFunctionNode->message = outMessageNode;
extFunctionNode->statement = FB_NEW_POOL(getPool()) ExtValidationNode(
getPool(), outMessageNode, false, false);
JrdStatement* statement = udf->getStatement();
PAR_preparsed_node(tdbb, NULL, mainNode, NULL, &csb, &statement, false, 0);
udf->setStatement(statement);
}
catch (...)
{
EngineCheckout cout(tdbb, FB_FUNCTION);
externalFunction->dispose();
throw;
}
}
void ExtEngineManager::makeProcedure(thread_db* tdbb, CompilerScratch* csb, jrd_prc* prc,
const MetaName& engine, const string& entryPoint, const string& body)
{
string entryPointTrimmed = entryPoint;
entryPointTrimmed.trim();
EngineAttachmentInfo* attInfo = getEngineAttachment(tdbb, engine);
ContextManager<IExternalProcedure> ctxManager(tdbb, attInfo, attInfo->adminCharSet,
(prc->getName().package.isEmpty() ?
CallerName(obj_procedure, prc->getName().identifier) :
CallerName(obj_package_header, prc->getName().package)));
///MemoryPool& pool = *tdbb->getDefaultPool();
MemoryPool& pool = *getDefaultMemoryPool();
AutoPtr<RoutineMetadata> metadata(FB_NEW_POOL(pool) RoutineMetadata(pool));
metadata->package = prc->getName().package;
metadata->name = prc->getName().identifier;
metadata->entryPoint = entryPointTrimmed;
metadata->body = body;
metadata->inputParameters = Routine::createMetadata(prc->getInputFields());
metadata->outputParameters = Routine::createMetadata(prc->getOutputFields());
FbLocalStatus status;
RefPtr<IMetadataBuilder> inBuilder(REF_NO_INCR, metadata->inputParameters->getBuilder(&status));
status.check();
RefPtr<IMetadataBuilder> outBuilder(REF_NO_INCR, metadata->outputParameters->getBuilder(&status));
status.check();
IExternalProcedure* externalProcedure;
{ // scope
EngineCheckout cout(tdbb, FB_FUNCTION);
externalProcedure = attInfo->engine->makeProcedure(&status, attInfo->context, metadata,
inBuilder, outBuilder);
status.check();
if (!externalProcedure)
{
status_exception::raise(
Arg::Gds(isc_eem_proc_not_returned) <<
prc->getName().toString() << engine);
}
metadata->inputParameters = inBuilder->getMetadata(&status);
status.check();
metadata->outputParameters = outBuilder->getMetadata(&status);
status.check();
}
prc->setInputFormat(Routine::createFormat(pool, metadata->inputParameters, false));
prc->setOutputFormat(Routine::createFormat(pool, metadata->outputParameters, true));
try
{
prc->setExternal(FB_NEW_POOL(getPool()) Procedure(tdbb, this, attInfo->engine,
metadata.release(), externalProcedure, prc));
CompoundStmtNode* mainNode = FB_NEW_POOL(getPool()) CompoundStmtNode(getPool());
ExtMessageNode* inMessageNode = prc->getInputFields().hasData() ?
FB_NEW_POOL(getPool()) ExtMessageNode(tdbb, getPool(), csb, 0,
prc->getInputFields(), prc->getInputFormat()) :
NULL;
if (inMessageNode)
mainNode->statements.add(inMessageNode);
ExtMessageNode* outMessageNode = FB_NEW_POOL(getPool()) ExtMessageNode(tdbb, getPool(), csb, 1,
prc->getOutputFields(), prc->getOutputFormat());
mainNode->statements.add(outMessageNode);
ExtInitOutputNode* initOutputNode = FB_NEW_POOL(getPool()) ExtInitOutputNode(
tdbb, getPool(), csb, outMessageNode);
mainNode->statements.add(initOutputNode);
ReceiveNode* receiveNode = inMessageNode ?
FB_NEW_POOL(getPool()) ReceiveNode(getPool()) : NULL;
if (inMessageNode)
{
receiveNode->message = inMessageNode;
receiveNode->statement = FB_NEW_POOL(getPool()) ExtValidationNode(
getPool(), inMessageNode, true, true);
mainNode->statements.add(receiveNode);
}
ExtProcedureNode* extProcedureNode = FB_NEW_POOL(getPool()) ExtProcedureNode(getPool(),
receiveNode, prc->getExternal());
mainNode->statements.add(extProcedureNode);
extProcedureNode->message = outMessageNode;
extProcedureNode->statement = FB_NEW_POOL(getPool()) ExtValidationNode(
getPool(), outMessageNode, true, false);
JrdStatement* statement = prc->getStatement();
PAR_preparsed_node(tdbb, NULL, mainNode, NULL, &csb, &statement, false, 0);
prc->setStatement(statement);
}
catch (...)
{
EngineCheckout cout(tdbb, FB_FUNCTION);
externalProcedure->dispose();
throw;
}
}
void ExtEngineManager::makeTrigger(thread_db* tdbb, CompilerScratch* csb, Jrd::Trigger* trg,
const MetaName& engine, const string& entryPoint, const string& body,
unsigned type)
{
string entryPointTrimmed = entryPoint;
entryPointTrimmed.trim();
EngineAttachmentInfo* attInfo = getEngineAttachment(tdbb, engine);
ContextManager<IExternalTrigger> ctxManager(tdbb, attInfo, attInfo->adminCharSet,
CallerName(obj_trigger, trg->name));
///MemoryPool& pool = *tdbb->getDefaultPool();
MemoryPool& pool = *getDefaultMemoryPool();
AutoPtr<RoutineMetadata> metadata(FB_NEW_POOL(pool) RoutineMetadata(pool));
metadata->name = trg->name;
metadata->entryPoint = entryPointTrimmed;
metadata->body = body;
metadata->triggerType = type;
jrd_rel* relation = trg->relation;
if (relation)
{
metadata->triggerTable = relation->rel_name;
MsgMetadata* fieldsMsg = FB_NEW MsgMetadata;
metadata->triggerFields = fieldsMsg;
Format* relFormat = relation->rel_current_format;
for (FB_SIZE_T i = 0; i < relation->rel_fields->count(); ++i)
{
jrd_fld* field = (*relation->rel_fields)[i];
if (field)
fieldsMsg->addItem(field->fld_name, !field->fld_not_null, relFormat->fmt_desc[i]);
}
}
FbLocalStatus status;
RefPtr<IMetadataBuilder> fieldsBuilder(REF_NO_INCR, relation ?
metadata->triggerFields->getBuilder(&status) : NULL);
if (relation)
{
status.check();
}
IExternalTrigger* externalTrigger;
{ // scope
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
externalTrigger = attInfo->engine->makeTrigger(&status, attInfo->context, metadata,
fieldsBuilder);
status.check();
if (!externalTrigger)
{
status_exception::raise(
Arg::Gds(isc_eem_trig_not_returned) << trg->name << engine);
}
if (relation)
{
metadata->triggerFields = fieldsBuilder->getMetadata(&status);
status.check();
}
}
try
{
trg->extTrigger = FB_NEW_POOL(getPool()) Trigger(tdbb, pool, this, attInfo->engine,
metadata.release(), externalTrigger, trg);
CompoundStmtNode* mainNode = FB_NEW_POOL(getPool()) CompoundStmtNode(getPool());
ExtTriggerNode* extTriggerNode = FB_NEW_POOL(getPool()) ExtTriggerNode(getPool(),
trg->extTrigger);
mainNode->statements.add(extTriggerNode);
PAR_preparsed_node(tdbb, trg->relation, mainNode, NULL, &csb, &trg->statement, true, 0);
}
catch (...)
{
EngineCheckout cout(tdbb, FB_FUNCTION);
externalTrigger->dispose();
throw;
}
}
IExternalEngine* ExtEngineManager::getEngine(thread_db* tdbb, const MetaName& name)
{
ReadLockGuard readGuard(enginesLock, FB_FUNCTION);
IExternalEngine* engine = NULL;
if (!engines.get(name, engine))
{
readGuard.release();
WriteLockGuard writeGuard(enginesLock, FB_FUNCTION);
if (!engines.get(name, engine))
{
GetPlugins<IExternalEngine> engineControl(IPluginManager::TYPE_EXTERNAL_ENGINE, name.c_str());
if (engineControl.hasData())
{
EngineAttachment key(NULL, NULL);
AutoPtr<EngineAttachmentInfo> attInfo;
try
{
EngineCheckout cout(tdbb, FB_FUNCTION);
engine = engineControl.plugin();
if (engine)
{
Attachment::SyncGuard attGuard(tdbb->getAttachment(), FB_FUNCTION);
key = EngineAttachment(engine, tdbb->getAttachment());
attInfo = FB_NEW_POOL(getPool()) EngineAttachmentInfo();
attInfo->engine = engine;
attInfo->context = FB_NEW_POOL(getPool()) ExternalContextImpl(tdbb, engine);
setupAdminCharSet(tdbb, engine, attInfo);
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, attInfo->adminCharSet);
FbLocalStatus status;
engine->openAttachment(&status, attInfo->context); //// FIXME: log status
}
}
catch (...)
{
if (engine)
{
PluginManagerInterfacePtr()->releasePlugin(engine);
}
throw;
}
if (engine)
{
engine->addRef();
engines.put(name, engine);
enginesAttachments.put(key, attInfo);
attInfo.release();
}
}
}
}
if (!engine)
{
status_exception::raise(Arg::Gds(isc_eem_engine_notfound) << name);
}
return engine;
}
ExtEngineManager::EngineAttachmentInfo* ExtEngineManager::getEngineAttachment(
thread_db* tdbb, const MetaName& name)
{
IExternalEngine* engine = getEngine(tdbb, name);
return getEngineAttachment(tdbb, engine);
}
ExtEngineManager::EngineAttachmentInfo* ExtEngineManager::getEngineAttachment(
thread_db* tdbb, IExternalEngine* engine, bool closing)
{
EngineAttachment key(engine, tdbb->getAttachment());
EngineAttachmentInfo* attInfo = NULL;
ReadLockGuard readGuard(&enginesLock, FB_FUNCTION);
if (!enginesAttachments.get(key, attInfo) && !closing)
{
readGuard.release();
WriteLockGuard writeGuard(enginesLock, FB_FUNCTION);
if (!enginesAttachments.get(key, attInfo))
{
attInfo = FB_NEW_POOL(getPool()) EngineAttachmentInfo();
attInfo->engine = engine;
attInfo->context = FB_NEW_POOL(getPool()) ExternalContextImpl(tdbb, engine);
setupAdminCharSet(tdbb, engine, attInfo);
enginesAttachments.put(key, attInfo);
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, attInfo->adminCharSet);
EngineCheckout cout(tdbb, FB_FUNCTION);
FbLocalStatus status;
engine->openAttachment(&status, attInfo->context); //// FIXME: log status
}
return attInfo;
}
if (closing && attInfo)
{
readGuard.release();
WriteLockGuard writeGuard(enginesLock, FB_FUNCTION);
enginesAttachments.remove(key);
}
return attInfo;
}
void ExtEngineManager::setupAdminCharSet(thread_db* tdbb, IExternalEngine* engine,
EngineAttachmentInfo* attInfo)
{
ContextManager<IExternalFunction> ctxManager(tdbb, attInfo, CS_UTF8);
char charSetName[MAX_SQL_IDENTIFIER_SIZE] = "NONE";
FbLocalStatus status;
engine->open(&status, attInfo->context, charSetName, MAX_SQL_IDENTIFIER_LEN);
status.check();
charSetName[MAX_SQL_IDENTIFIER_LEN] = '\0';
if (!MET_get_char_coll_subtype(tdbb, &attInfo->adminCharSet,
reinterpret_cast<const UCHAR*>(charSetName),
static_cast<USHORT>(strlen(charSetName))))
{
status_exception::raise(
Arg::Gds(isc_charset_not_found) <<
Arg::Str(charSetName));
}
}
} // namespace Jrd