mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-26 07:23:08 +01:00
1023 lines
25 KiB
Plaintext
1023 lines
25 KiB
Plaintext
/*
|
|
* PROGRAM: JRD Access Method
|
|
* MODULE: fun.e
|
|
* DESCRIPTION: External Function handling code.
|
|
*
|
|
* 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): ______________________________________.
|
|
*
|
|
* 2001.9.18 Claudio Valderrama: Allow return parameter by descriptor
|
|
* to signal NULL by testing the flags of the parameter's descriptor.
|
|
*
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
*
|
|
*/
|
|
/*
|
|
$Id: fun.epp,v 1.18 2003-02-17 08:41:58 eku Exp $
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "../jrd/jrd.h"
|
|
#include "../jrd/val.h"
|
|
#include "../jrd/exe.h"
|
|
#include "../jrd/gds.h"
|
|
#include "../jrd/req.h"
|
|
#include "../jrd/lls.h"
|
|
#include "../jrd/blb.h"
|
|
#include "../jrd/flu.h"
|
|
#include "../jrd/common.h"
|
|
#include "../jrd/constants.h"
|
|
#include "../jrd/ibsetjmp.h"
|
|
#include "../jrd/irq.h"
|
|
#include "../jrd/all_proto.h"
|
|
#include "../jrd/blb_proto.h"
|
|
#include "../jrd/cmp_proto.h"
|
|
#include "../jrd/dsc_proto.h"
|
|
#include "../jrd/err_proto.h"
|
|
#include "../jrd/evl_proto.h"
|
|
#include "../jrd/exe_proto.h"
|
|
#include "../jrd/flu_proto.h"
|
|
#include "../jrd/fun_proto.h"
|
|
#include "../jrd/gds_proto.h"
|
|
#include "../jrd/mov_proto.h"
|
|
#include "../jrd/sym_proto.h"
|
|
#include "../jrd/thd_proto.h"
|
|
#include "../jrd/sch_proto.h"
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
#ifdef VMS
|
|
#define CALL_UDF(ptr, udftype) (*(udftype (*)(SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG))(ptr)) (args [0], args[1], args[2], args[3], args [4], args [5], args [6], args [7], args [8], args [9])
|
|
#else
|
|
#define CALL_UDF(ptr, udftype) ((udftype (*)(SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG, SLONG))(ptr)) (args [0], args[1], args[2], args[3], args [4], args [5], args [6], args [7], args [8], args [9])
|
|
#endif
|
|
|
|
DATABASE DB = FILENAME "ODS.RDB";
|
|
|
|
#ifdef VMS
|
|
extern double MTH$CVT_D_G(), MTH$CVT_G_D();
|
|
#endif
|
|
|
|
#define EXCEPTION_MESSAGE "The user defined function: \t%s\n\t referencing entrypoint: \t%s\n\t in module: \t%s\n\tcaused the fatal exception:"
|
|
|
|
static SSHORT blob_get_segment(BLB, UCHAR *, USHORT, USHORT *);
|
|
static void blob_put_segment(BLB, UCHAR *, USHORT);
|
|
static SLONG blob_lseek(BLB, USHORT, SLONG);
|
|
static SLONG get_scalar_array(fun_repeat *, DSC *, SAD, LLS *);
|
|
|
|
|
|
void DLL_EXPORT FUN_evaluate(FUN function, JRD_NOD node, VLU value)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N _ e v a l u a t e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Evaluate a function.
|
|
*
|
|
**************************************/
|
|
|
|
SLONG args[MAX_UDF_ARGUMENTS + 1];
|
|
USHORT length;
|
|
UCHAR temp[800];
|
|
STR temp_string;
|
|
SSHORT unsup_datatype = 0;
|
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
|
|
|
// Start by constructing argument list
|
|
UCHAR* temp_ptr;
|
|
|
|
if (function->fun_temp_length < sizeof(temp))
|
|
{
|
|
temp_string = NULL;
|
|
temp_ptr = temp;
|
|
MOVE_CLEAR(temp, sizeof(temp));
|
|
}
|
|
else
|
|
{
|
|
/* Need to align the starting address because it consists of
|
|
a number of data blocks with aligned length */
|
|
|
|
temp_string =
|
|
FB_NEW_RPT(*tdbb->tdbb_default,
|
|
function->fun_temp_length + DOUBLE_ALIGN - 1) str;
|
|
MOVE_CLEAR(temp_string->str_data, temp_string->str_length);
|
|
temp_ptr =
|
|
(UCHAR *) FB_ALIGN((U_IPTR) temp_string->str_data, DOUBLE_ALIGN);
|
|
}
|
|
|
|
MOVE_CLEAR(args, sizeof(args));
|
|
SLONG* arg_ptr = args;
|
|
LLS blob_stack = 0;
|
|
LLS array_stack = 0;
|
|
JRD_REQ request = tdbb->tdbb_request;
|
|
USHORT null_flag = (USHORT)(request->req_flags & req_null);
|
|
fun_repeat* return_ptr = function->fun_rpt + function->fun_return_arg;
|
|
JRD_NOD* ptr = node->nod_arg;
|
|
|
|
value->vlu_desc = return_ptr->fun_desc;
|
|
value->vlu_desc.dsc_address = (UCHAR *) & value->vlu_misc;
|
|
|
|
/* Trap any potential errors */
|
|
|
|
#pragma FB_COMPILER_MESSAGE("Note that we cannot use C++ EH here for Win32 SS!")
|
|
#if !defined(SUPERSERVER) && !defined(WIN_NT)
|
|
try {
|
|
#endif
|
|
|
|
START_CHECK_FOR_EXCEPTIONS( (TEXT*) function->fun_exception_message);
|
|
|
|
// If the return data type is any of the string types,
|
|
// allocate space to hold value
|
|
|
|
|
|
if (value->vlu_desc.dsc_dtype <= dtype_varying)
|
|
{
|
|
length = value->vlu_desc.dsc_length;
|
|
STR string = value->vlu_string;
|
|
if (string && string->str_length < length) {
|
|
delete string;
|
|
string = NULL;
|
|
}
|
|
if (!string) {
|
|
string = FB_NEW_RPT(*tdbb->tdbb_default, length) str;
|
|
string->str_length = length;
|
|
value->vlu_string = string;
|
|
}
|
|
value->vlu_desc.dsc_address = string->str_data;
|
|
}
|
|
|
|
|
|
DSC temp_desc;
|
|
double d;
|
|
double* dp;
|
|
|
|
// Process arguments
|
|
const fun_repeat* end = function->fun_rpt + 1 + function->fun_count;
|
|
for (fun_repeat* tail = function->fun_rpt + 1; tail < end; ++tail)
|
|
{
|
|
DSC* input;
|
|
if (tail == return_ptr) {
|
|
input = &value->vlu_desc; /*(DSC*) value;*/
|
|
} else {
|
|
input = EVL_expr(tdbb, *ptr++);
|
|
}
|
|
SLONG** ap = (SLONG **) arg_ptr;
|
|
|
|
// If we're passing data type ISC descriptor, there's
|
|
// nothing left to be done
|
|
|
|
if (tail->fun_mechanism == FUN_descriptor) {
|
|
*ap++ = (SLONG *) input;
|
|
arg_ptr = (SLONG *) ap;
|
|
continue;
|
|
}
|
|
|
|
temp_desc = tail->fun_desc;
|
|
temp_desc.dsc_address = temp_ptr;
|
|
length = FB_ALIGN(temp_desc.dsc_length, DOUBLE_ALIGN);
|
|
|
|
/* If we've got a null argument, just pass zeros (got any better ideas?) */
|
|
|
|
if (!input || (request->req_flags & req_null))
|
|
{
|
|
if (tail->fun_mechanism == FUN_value)
|
|
{
|
|
UCHAR* p = (UCHAR *) arg_ptr;
|
|
MOVE_CLEAR(p, (SLONG) length);
|
|
p += length;
|
|
arg_ptr = (SLONG *) p;
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
MOVE_CLEAR(temp_ptr, (SLONG) length);
|
|
}
|
|
}
|
|
else if ((SSHORT) abs(tail->fun_mechanism) == FUN_scalar_array)
|
|
{
|
|
length = get_scalar_array(tail, input, (SAD)temp_ptr, &array_stack);
|
|
}
|
|
else
|
|
{
|
|
switch (tail->fun_desc.dsc_dtype)
|
|
{
|
|
case dtype_short:
|
|
{
|
|
SSHORT s = MOV_get_long(input,
|
|
(SSHORT) tail->fun_desc.dsc_scale);
|
|
if (tail->fun_mechanism == FUN_value) {
|
|
/* For (apparent) portability reasons, SHORT by value
|
|
* is always passed as a LONG. See v3.2 release notes
|
|
* Passing by value is not supported in SQL due to
|
|
* these problems, but can still occur in GDML.
|
|
* 1994-September-28 David Schnepper
|
|
*/
|
|
*arg_ptr++ = (SLONG) s;
|
|
continue;
|
|
}
|
|
{
|
|
SSHORT* sp = (SSHORT *) temp_ptr;
|
|
*sp = s;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case dtype_sql_time:
|
|
case dtype_sql_date:
|
|
case dtype_long:
|
|
{
|
|
SLONG l;
|
|
/* this check added for bug 74193 */
|
|
if (tail->fun_desc.dsc_dtype == dtype_long)
|
|
{
|
|
l =
|
|
MOV_get_long(input,
|
|
(SSHORT) tail->fun_desc.dsc_scale);
|
|
}
|
|
else
|
|
{
|
|
/* for sql_date and sql_time just move the value to a long,
|
|
scale does not mean anything for it */
|
|
l = *((SLONG *) (input->dsc_address));
|
|
}
|
|
|
|
if (tail->fun_mechanism == FUN_value)
|
|
{
|
|
*arg_ptr++ = l;
|
|
continue;
|
|
}
|
|
{
|
|
SLONG* lp = (SLONG *) temp_ptr;
|
|
*lp = l;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case dtype_int64:
|
|
{
|
|
SINT64* pi64;
|
|
SINT64 i64 =
|
|
MOV_get_int64(input, (SSHORT)tail->fun_desc.dsc_scale);
|
|
|
|
if (tail->fun_mechanism == FUN_value)
|
|
{
|
|
pi64 = (SINT64 *) arg_ptr;
|
|
*pi64++ = i64;
|
|
arg_ptr = (SLONG *) pi64;
|
|
continue;
|
|
}
|
|
pi64 = (SINT64 *) temp_ptr;
|
|
*pi64 = i64;
|
|
}
|
|
break;
|
|
|
|
case dtype_real:
|
|
{
|
|
float f = (float) MOV_get_double(input);
|
|
if (tail->fun_mechanism == FUN_value)
|
|
{
|
|
/* For (apparent) portability reasons, FLOAT by value
|
|
* is always passed as a DOUBLE. See v3.2 release notes
|
|
* Passing by value is not supported in SQL due to
|
|
* these problems, but can still occur in GDML.
|
|
* 1994-September-28 David Schnepper
|
|
*/
|
|
double* dp = (double *) arg_ptr;
|
|
*dp++ = (double) f;
|
|
arg_ptr = (SLONG *) dp;
|
|
continue;
|
|
}
|
|
float* fp = (float *) temp_ptr;
|
|
*fp = f;
|
|
}
|
|
break;
|
|
|
|
case dtype_double:
|
|
#ifdef VMS
|
|
case dtype_d_float:
|
|
d = MOV_get_double(input);
|
|
if (tail->fun_desc.dsc_dtype == SPECIAL_DOUBLE)
|
|
{
|
|
d = CNVT_FROM_DFLT(&d);
|
|
}
|
|
#else
|
|
d = MOV_get_double(input);
|
|
#endif
|
|
if (tail->fun_mechanism == FUN_value)
|
|
{
|
|
dp = (double *) arg_ptr;
|
|
*dp++ = d;
|
|
arg_ptr = (SLONG *) dp;
|
|
continue;
|
|
}
|
|
dp = (double *) temp_ptr;
|
|
*dp = d;
|
|
break;
|
|
|
|
case dtype_text:
|
|
case dtype_cstring:
|
|
case dtype_varying:
|
|
if (tail == return_ptr) {
|
|
temp_ptr = value->vlu_desc.dsc_address;
|
|
length = 0;
|
|
}
|
|
else
|
|
MOV_move(input, &temp_desc);
|
|
break;
|
|
|
|
/* CVC: There's no other solution for now: timestamp can't be returned
|
|
by value and the other way is to force the user to pass a dummy value as
|
|
an argument to keep the engine happy. So, here's the hack. */
|
|
case dtype_timestamp:
|
|
if (tail == return_ptr)
|
|
{
|
|
temp_ptr = value->vlu_desc.dsc_address;
|
|
length = sizeof(GDS_TIMESTAMP);
|
|
}
|
|
else
|
|
{
|
|
MOV_move(input, &temp_desc);
|
|
}
|
|
break;
|
|
|
|
case dtype_quad:
|
|
case dtype_array:
|
|
MOV_move(input, &temp_desc);
|
|
break;
|
|
|
|
case dtype_blob:
|
|
{
|
|
BLOB blob_desc = (BLOB) temp_ptr;
|
|
BLB blob;
|
|
length = sizeof(struct blob);
|
|
if (tail == return_ptr)
|
|
{
|
|
blob =
|
|
BLB_create(tdbb, tdbb->tdbb_request->req_transaction,
|
|
(BID) & value->vlu_misc);
|
|
}
|
|
else
|
|
{
|
|
SLONG* blob_id;
|
|
if (request->req_flags & req_null)
|
|
{
|
|
SLONG null_quad[2];
|
|
null_quad[0] = null_quad[1] = 0;
|
|
blob_id = null_quad;
|
|
}
|
|
else
|
|
{
|
|
if (input->dsc_dtype != dtype_quad &&
|
|
input->dsc_dtype != dtype_blob)
|
|
{
|
|
ERR_post(gds_wish_list,
|
|
gds_arg_gds, gds_blobnotsup,
|
|
gds_arg_string, "conversion", 0);
|
|
}
|
|
blob_id = (SLONG *) input->dsc_address;
|
|
}
|
|
blob = BLB_open(tdbb,
|
|
tdbb->tdbb_request->req_transaction,
|
|
(BID)blob_id);
|
|
}
|
|
LLS_PUSH(blob, &blob_stack);
|
|
blob_desc->blob_get_segment = (SSHORT (*)())blob_get_segment;
|
|
blob_desc->blob_put_segment = (void (*)())blob_put_segment;
|
|
blob_desc->blob_seek = (SLONG (*)())blob_lseek;
|
|
blob_desc->blob_handle = (int *) blob;
|
|
blob_desc->blob_number_segments = blob->blb_count;
|
|
blob_desc->blob_max_segment = blob->blb_max_segment;
|
|
blob_desc->blob_total_length = blob->blb_length;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
assert(FALSE);
|
|
MOV_move(input, &temp_desc);
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
*ap++ = (SLONG *) temp_ptr;
|
|
arg_ptr = (SLONG *) ap;
|
|
temp_ptr += length;
|
|
}
|
|
|
|
if (function->fun_return_arg)
|
|
{
|
|
THREAD_EXIT;
|
|
CALL_UDF(function->fun_entrypoint, void);
|
|
THREAD_ENTER;
|
|
}
|
|
else if ((SLONG) return_ptr->fun_mechanism == FUN_value)
|
|
{
|
|
switch (value->vlu_desc.dsc_dtype)
|
|
{
|
|
case dtype_sql_time:
|
|
case dtype_sql_date:
|
|
case dtype_long:
|
|
THREAD_EXIT;
|
|
value->vlu_misc.vlu_long =
|
|
CALL_UDF(function->fun_entrypoint, SLONG);
|
|
THREAD_ENTER;
|
|
break;
|
|
|
|
case dtype_short:
|
|
/* For (apparent) portability reasons, SHORT by value
|
|
* must always be returned as a LONG. See v3.2 release notes
|
|
* 1994-September-28 David Schnepper
|
|
*/
|
|
THREAD_EXIT;
|
|
value->vlu_misc.vlu_short = (SSHORT)
|
|
CALL_UDF(function->fun_entrypoint, SLONG);
|
|
THREAD_ENTER;
|
|
break;
|
|
|
|
case dtype_real:
|
|
/* For (apparent) portability reasons, FLOAT by value
|
|
* must always be returned as a DOUBLE. See v3.2 release notes
|
|
* 1994-September-28 David Schnepper
|
|
*/
|
|
THREAD_EXIT;
|
|
value->vlu_misc.vlu_float = (float)
|
|
CALL_UDF(function->fun_entrypoint, double);
|
|
THREAD_ENTER;
|
|
break;
|
|
|
|
case dtype_int64:
|
|
THREAD_EXIT;
|
|
value->vlu_misc.vlu_int64 =
|
|
CALL_UDF(function->fun_entrypoint, SINT64);
|
|
THREAD_ENTER;
|
|
break;
|
|
|
|
case dtype_double:
|
|
#ifdef VMS
|
|
case dtype_d_float:
|
|
#endif
|
|
THREAD_EXIT;
|
|
value->vlu_misc.vlu_double =
|
|
CALL_UDF(function->fun_entrypoint, double);
|
|
THREAD_ENTER;
|
|
break;
|
|
|
|
case dtype_timestamp:
|
|
default:
|
|
unsup_datatype = 1;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
THREAD_EXIT;
|
|
typedef UCHAR* pUCHAR;
|
|
temp_ptr = CALL_UDF(function->fun_entrypoint, pUCHAR);
|
|
THREAD_ENTER;
|
|
|
|
if (temp_ptr != NULL)
|
|
{
|
|
/* CVC: Allow processing of return by descriptor.
|
|
The assumption is that the user didn't modify the return type,
|
|
that has no sense after all, because it is carved in stone. */
|
|
if (return_ptr->fun_mechanism == FUN_descriptor)
|
|
{
|
|
temp_ptr = ((DSC*) temp_ptr)->dsc_address;
|
|
}
|
|
if (temp_ptr)
|
|
{
|
|
switch (value->vlu_desc.dsc_dtype)
|
|
{
|
|
case dtype_sql_date:
|
|
case dtype_sql_time:
|
|
case dtype_long:
|
|
value->vlu_misc.vlu_long = *(SLONG *) temp_ptr;
|
|
break;
|
|
|
|
case dtype_short:
|
|
value->vlu_misc.vlu_short = *(SSHORT *) temp_ptr;
|
|
break;
|
|
|
|
case dtype_real:
|
|
value->vlu_misc.vlu_float = *(float *) temp_ptr;
|
|
break;
|
|
|
|
case dtype_int64:
|
|
value->vlu_misc.vlu_int64 = *(SINT64 *) temp_ptr;
|
|
break;
|
|
|
|
case dtype_double:
|
|
#ifdef VMS
|
|
case dtype_d_float:
|
|
#endif
|
|
value->vlu_misc.vlu_double = *(double *) temp_ptr;
|
|
break;
|
|
|
|
case dtype_text:
|
|
case dtype_cstring:
|
|
case dtype_varying:
|
|
temp_desc = value->vlu_desc;
|
|
temp_desc.dsc_address = temp_ptr;
|
|
temp_desc.dsc_length = strlen((char*)temp_ptr) + 1;
|
|
MOV_move(&temp_desc, &value->vlu_desc);
|
|
break;
|
|
|
|
case dtype_timestamp:
|
|
{
|
|
SLONG* ip = (SLONG *) temp_ptr;
|
|
value->vlu_misc.vlu_dbkey[0] = *ip++;
|
|
value->vlu_misc.vlu_dbkey[1] = *ip;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
unsup_datatype = 1;
|
|
break;
|
|
}
|
|
/* check if this is function has the FREE_IT set, if set and
|
|
return_value is not null, then free the return value */
|
|
if (((SLONG) return_ptr->fun_mechanism < 0) && temp_ptr)
|
|
free(temp_ptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
END_CHECK_FOR_EXCEPTIONS((TEXT*)function->fun_exception_message);
|
|
|
|
if (unsup_datatype) {
|
|
IBERROR(169); /* msg 169 return data type not supported */
|
|
}
|
|
|
|
#if !defined(SUPERSERVER) && !defined(WIN_NT)
|
|
}
|
|
catch (const std::exception&) {
|
|
delete temp_string;
|
|
while (array_stack) {
|
|
delete[] (UCHAR*)LLS_POP(&array_stack);
|
|
}
|
|
ERR_punt();
|
|
}
|
|
#endif
|
|
|
|
delete temp_string;
|
|
|
|
while (blob_stack)
|
|
{
|
|
BLB_close(tdbb, (BLB)LLS_POP(&blob_stack));
|
|
}
|
|
|
|
while (array_stack)
|
|
{
|
|
delete[] (UCHAR*)LLS_POP(&array_stack);
|
|
}
|
|
|
|
if (temp_ptr == NULL || function->fun_return_arg &&
|
|
return_ptr->fun_mechanism == FUN_descriptor &&
|
|
(value->vlu_desc.dsc_flags & DSC_null))
|
|
{
|
|
request->req_flags |= req_null;
|
|
}
|
|
else
|
|
{
|
|
request->req_flags &= ~req_null;
|
|
}
|
|
request->req_flags |= null_flag;
|
|
}
|
|
|
|
|
|
void DLL_EXPORT FUN_fini( TDBB tdbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N _ f i n i
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Unregister interest in external function modules.
|
|
*
|
|
**************************************/
|
|
DBB dbb;
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
while (dbb->dbb_modules)
|
|
FLU_unregister_module((MOD) LLS_POP(&dbb->dbb_modules));
|
|
}
|
|
|
|
|
|
void DLL_EXPORT FUN_init(void)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N _ i n i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Initialize external function mechanism.
|
|
*
|
|
**************************************/
|
|
}
|
|
|
|
|
|
FUN DLL_EXPORT FUN_lookup_function(TEXT * name)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N _ l o o k u p _ f u n c t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lookup function by name.
|
|
*
|
|
**************************************/
|
|
TDBB tdbb;
|
|
DBB dbb;
|
|
BLK request_fun, request_arg;
|
|
FUN function, prior;
|
|
SYM symbol;
|
|
STR string;
|
|
STR exception_msg;
|
|
TEXT *p;
|
|
MOD module;
|
|
LLS stack;
|
|
USHORT count, args, l;
|
|
ULONG length;
|
|
fun_repeat *tail, temp[MAX_UDF_ARGUMENTS + 1];
|
|
|
|
tdbb = GET_THREAD_DATA;
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
/* Start by looking for already defined symbol */
|
|
|
|
V4_JRD_MUTEX_LOCK(dbb->dbb_mutexes + DBB_MUTX_udf);
|
|
|
|
for (symbol = SYM_lookup(name); symbol; symbol = symbol->sym_homonym)
|
|
{
|
|
if (symbol->sym_type == SYM_fun) {
|
|
V4_JRD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_udf);
|
|
return (FUN) symbol->sym_object;
|
|
}
|
|
}
|
|
|
|
prior = NULL;
|
|
|
|
request_fun = (BLK) CMP_find_request(tdbb, irq_l_functions, IRQ_REQUESTS);
|
|
request_arg = (BLK) CMP_find_request(tdbb, irq_l_args, IRQ_REQUESTS);
|
|
|
|
FOR(REQUEST_HANDLE request_fun) X IN RDB$FUNCTIONS
|
|
WITH X.RDB$FUNCTION_NAME EQ name
|
|
if (!REQUEST(irq_l_functions))
|
|
REQUEST(irq_l_functions) = request_fun;
|
|
count = args = 0;
|
|
MOVE_CLEAR(temp, (SLONG) sizeof(temp));
|
|
length = 0;
|
|
FOR(REQUEST_HANDLE request_arg) Y IN RDB$FUNCTION_ARGUMENTS
|
|
WITH Y.RDB$FUNCTION_NAME EQ X.RDB$FUNCTION_NAME
|
|
SORTED BY Y.RDB$ARGUMENT_POSITION
|
|
|
|
if (!REQUEST(irq_l_args))
|
|
REQUEST(irq_l_args) = request_arg;
|
|
tail = temp + Y.RDB$ARGUMENT_POSITION;
|
|
tail->fun_mechanism = (FUN_T) Y.RDB$MECHANISM;
|
|
count = MAX(count, Y.RDB$ARGUMENT_POSITION);
|
|
DSC_make_descriptor(&tail->fun_desc, Y.RDB$FIELD_TYPE,
|
|
Y.RDB$FIELD_SCALE, Y.RDB$FIELD_LENGTH,
|
|
Y.RDB$FIELD_SUB_TYPE, Y.RDB$CHARACTER_SET_ID,
|
|
0);
|
|
|
|
if (tail->fun_desc.dsc_dtype == dtype_cstring)
|
|
tail->fun_desc.dsc_length++;
|
|
|
|
if (Y.RDB$ARGUMENT_POSITION != X.RDB$RETURN_ARGUMENT)
|
|
++args;
|
|
l = FB_ALIGN(tail->fun_desc.dsc_length, DOUBLE_ALIGN);
|
|
if (tail->fun_desc.dsc_dtype == dtype_blob)
|
|
l = sizeof(struct blob);
|
|
length += l;
|
|
END_FOR;
|
|
function = FB_NEW_RPT(*dbb->dbb_permanent, count + 1) fun;
|
|
function->fun_count = count;
|
|
function->fun_args = args;
|
|
function->fun_return_arg = X.RDB$RETURN_ARGUMENT;
|
|
function->fun_type = X.RDB$FUNCTION_TYPE;
|
|
function->fun_temp_length = length;
|
|
MOVE_FAST(temp, function->fun_rpt,
|
|
(count + 1) * sizeof(fun_repeat));
|
|
#ifndef WIN_NT /* NT allows blanks in file paths */
|
|
for (p = X.RDB$MODULE_NAME; *p && *p != ' '; p++);
|
|
*p = 0;
|
|
#endif
|
|
for (p = X.RDB$ENTRYPOINT; *p && *p != ' '; p++);
|
|
*p = 0;
|
|
|
|
/* Prepare the exception message to be used in case this function ever
|
|
causes an exception. This is done at this time to save us from preparing
|
|
(thus allocating) this message every time the function is called. */
|
|
exception_msg =
|
|
FB_NEW_RPT(*dbb->dbb_permanent,
|
|
strlen(EXCEPTION_MESSAGE) + strlen(name) +
|
|
strlen(X.RDB$ENTRYPOINT) +
|
|
strlen(X.RDB$MODULE_NAME) + 1) str;
|
|
sprintf((char*)exception_msg->str_data, EXCEPTION_MESSAGE, name,
|
|
X.RDB$ENTRYPOINT, X.RDB$MODULE_NAME);
|
|
function->fun_exception_message = exception_msg;
|
|
|
|
function->fun_entrypoint =
|
|
ISC_lookup_entrypoint(X.RDB$MODULE_NAME, X.RDB$ENTRYPOINT,
|
|
ISC_EXT_LIB_PATH_ENV);
|
|
|
|
if ( (module = FLU_lookup_module(X.RDB$MODULE_NAME)) ) {
|
|
/* Register interest in the module by database. */
|
|
|
|
for (stack = dbb->dbb_modules; stack; stack = stack->lls_next)
|
|
if (module == (MOD) stack->lls_object)
|
|
break;
|
|
|
|
/* If the module was already registered with this database
|
|
then decrement the use count that was incremented in
|
|
ISC_lookup_entrypoint() above. Otherwise push it onto
|
|
the stack of registered modules. */
|
|
|
|
if (stack)
|
|
{
|
|
FLU_unregister_module(module);
|
|
}
|
|
else
|
|
{
|
|
JrdMemoryPool *old_pool;
|
|
|
|
old_pool = tdbb->tdbb_default;
|
|
tdbb->tdbb_default = dbb->dbb_permanent;
|
|
LLS_PUSH(module, &dbb->dbb_modules);
|
|
tdbb->tdbb_default = old_pool;
|
|
}
|
|
}
|
|
|
|
/* Could not find a function with given MODULE, ENTRYPOINT,
|
|
* Try the list of internally implemented functions.
|
|
*/
|
|
if (!function->fun_entrypoint)
|
|
function->fun_entrypoint =
|
|
BUILTIN_entrypoint(X.RDB$MODULE_NAME, X.RDB$ENTRYPOINT);
|
|
|
|
if (prior) {
|
|
function->fun_homonym = prior->fun_homonym;
|
|
prior->fun_homonym = function;
|
|
}
|
|
else
|
|
{
|
|
prior = function;
|
|
function->fun_symbol = symbol = FB_NEW(*dbb->dbb_permanent) sym;
|
|
symbol->sym_object = (BLK) function;
|
|
string = FB_NEW_RPT(*dbb->dbb_permanent, strlen(name)) str;
|
|
strcpy((char*)string->str_data, name);
|
|
symbol->sym_string = (TEXT *) string->str_data;
|
|
symbol->sym_type = SYM_fun;
|
|
SYM_insert(symbol);
|
|
}
|
|
END_FOR;
|
|
|
|
if (!REQUEST(irq_l_functions))
|
|
REQUEST(irq_l_functions) = request_fun;
|
|
if (!REQUEST(irq_l_args))
|
|
REQUEST(irq_l_args) = request_arg;
|
|
|
|
V4_JRD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_udf);
|
|
|
|
return prior;
|
|
}
|
|
|
|
|
|
FUN DLL_EXPORT FUN_resolve(CSB csb, FUN function, JRD_NOD args)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N _ r e s o l v e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Resolve instance of potentially overloaded function.
|
|
*
|
|
**************************************/
|
|
FUN best;
|
|
JRD_NOD *ptr, *end;
|
|
DSC arg;
|
|
int best_score, score;
|
|
fun_repeat *tail;
|
|
TDBB tdbb;
|
|
|
|
tdbb = GET_THREAD_DATA;
|
|
|
|
best = NULL;
|
|
best_score = 0;
|
|
end = args->nod_arg + args->nod_count;
|
|
|
|
for (; function; function = function->fun_homonym)
|
|
if (function->fun_entrypoint && function->fun_args == args->nod_count) {
|
|
score = 0;
|
|
for (ptr = args->nod_arg, tail = function->fun_rpt + 1; ptr < end;
|
|
ptr++, tail++) {
|
|
CMP_get_desc(tdbb, csb, *ptr, &arg);
|
|
if ((SSHORT) abs(tail->fun_mechanism) == FUN_descriptor)
|
|
score += 10;
|
|
else if (tail->fun_desc.dsc_dtype == dtype_blob ||
|
|
arg.dsc_dtype == dtype_blob) {
|
|
score = 0;
|
|
break;
|
|
}
|
|
else if (tail->fun_desc.dsc_dtype >= arg.dsc_dtype)
|
|
score += 10 - (arg.dsc_dtype - tail->fun_desc.dsc_dtype);
|
|
else
|
|
score += 1;
|
|
}
|
|
if (!best || score > best_score) {
|
|
best_score = score;
|
|
best = function;
|
|
}
|
|
}
|
|
|
|
return best;
|
|
}
|
|
|
|
|
|
static SLONG blob_lseek( BLB blob, USHORT mode, SLONG offset)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b l o b _ l s e e k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* lseek a a blob segement. Return the offset
|
|
*
|
|
**************************************/
|
|
|
|
SLONG return_offset;
|
|
|
|
/* add thread enter and thread_exit wrappers */
|
|
THREAD_ENTER;
|
|
return_offset = BLB_lseek(blob, mode, offset);
|
|
THREAD_EXIT;
|
|
return (return_offset);
|
|
}
|
|
|
|
|
|
static void blob_put_segment( BLB blob, UCHAR * buffer, USHORT length)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b l o b _ p u t _ s e g m e n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Put segment into a blob. Return nothing
|
|
*
|
|
**************************************/
|
|
TDBB tdbb;
|
|
|
|
/* As this is a call-back from a UDF, must reacquire the
|
|
engine mutex */
|
|
|
|
THREAD_ENTER;
|
|
tdbb = GET_THREAD_DATA;
|
|
BLB_put_segment(tdbb, blob, buffer, length);
|
|
THREAD_EXIT;
|
|
}
|
|
|
|
|
|
static SSHORT blob_get_segment(
|
|
BLB blob,
|
|
UCHAR * buffer,
|
|
USHORT length, USHORT * return_length)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b l o b _ g e t _ s e g m e n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Get next segment of a blob. Return the following:
|
|
*
|
|
* 1 -- Complete segment has been returned.
|
|
* 0 -- End of blob (no data returned).
|
|
* -1 -- Current segment is incomplete.
|
|
*
|
|
**************************************/
|
|
TDBB tdbb;
|
|
|
|
/* add thread enter and thread_exit wrappers */
|
|
THREAD_ENTER;
|
|
tdbb = GET_THREAD_DATA;
|
|
*return_length = BLB_get_segment(tdbb, blob, buffer, length);
|
|
THREAD_EXIT;
|
|
|
|
if (blob->blb_flags & BLB_eof)
|
|
return 0;
|
|
|
|
if (blob->blb_fragment_size)
|
|
return -1;
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
static SLONG get_scalar_array(fun_repeat* arg,
|
|
DSC* value,
|
|
SAD scalar_desc,
|
|
LLS* stack)
|
|
{
|
|
/**************************************
|
|
*
|
|
* g e t _ s c a l a r _ a r r a y
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Get and format a scalar array descriptor, then allocate space
|
|
* and fetch the array. If conversion is required, convert.
|
|
* Return length of array desc.
|
|
*
|
|
**************************************/
|
|
TDBB tdbb;
|
|
BLB blob;
|
|
ADS array_desc;
|
|
SLONG stuff[ADS_LEN(16) / 4], n;
|
|
UCHAR *data, *temp;
|
|
DSC from, to;
|
|
USHORT dimensions;
|
|
ads::ads_repeat *tail1;
|
|
sad::sad_repeat *tail2, *end;
|
|
|
|
tdbb = GET_THREAD_DATA;
|
|
|
|
/* Get first the array descriptor, then the array */
|
|
|
|
array_desc = (ADS) stuff;
|
|
blob = BLB_get_array(tdbb,
|
|
tdbb->tdbb_request->req_transaction,
|
|
(BID)value->dsc_address,
|
|
array_desc);
|
|
data = FB_NEW(*getDefaultMemoryPool()) UCHAR[array_desc->ads_total_length];
|
|
BLB_get_data(tdbb, blob, data, array_desc->ads_total_length);
|
|
dimensions = array_desc->ads_dimensions;
|
|
|
|
/* Convert array, if necessary */
|
|
|
|
to = arg->fun_desc;
|
|
from = array_desc->ads_rpt[0].ads_desc;
|
|
|
|
if (to.dsc_dtype != from.dsc_dtype ||
|
|
to.dsc_scale != from.dsc_scale || to.dsc_length != from.dsc_length) {
|
|
n = array_desc->ads_count;
|
|
to.dsc_address = temp = FB_NEW(*getDefaultMemoryPool()) UCHAR[(SLONG) to.dsc_length * n];
|
|
from.dsc_address = data;
|
|
for (; n; --n, to.dsc_address += to.dsc_length,
|
|
from.dsc_address += array_desc->ads_element_length)
|
|
MOV_move(&from, &to);
|
|
delete[] data;
|
|
data = temp;
|
|
}
|
|
|
|
/* Fill out the scalar array descriptor */
|
|
|
|
LLS_PUSH(data, stack);
|
|
scalar_desc->sad_desc = arg->fun_desc;
|
|
scalar_desc->sad_desc.dsc_address = data;
|
|
scalar_desc->sad_dimensions = dimensions;
|
|
|
|
for (tail2 = scalar_desc->sad_rpt, end = tail2 + dimensions, tail1 =
|
|
array_desc->ads_rpt; tail2 < end; ++tail1, ++tail2) {
|
|
tail2->sad_upper = tail1->ads_upper;
|
|
tail2->sad_lower = tail1->ads_lower;
|
|
}
|
|
|
|
return sizeof(struct sad) + (dimensions - 1) * sizeof(sad::sad_repeat);
|
|
}
|