mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 20:03:03 +01:00
452 lines
11 KiB
C++
452 lines
11 KiB
C++
/*
|
|
* PROGRAM: InterBase Access Method
|
|
* MODULE: functions.cpp
|
|
* DESCRIPTION: External entrypoint definitions
|
|
*
|
|
* 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): ______________________________________.
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
#include "../jrd/common.h"
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include "../jrd/jrd.h" /* For MAXPATHLEN Bug #126614 */
|
|
#include "../jrd/license.h"
|
|
#include "../jrd/tra.h"
|
|
#include "../jrd/dsc_proto.h"
|
|
#include "../jrd/pag_proto.h"
|
|
#include "../jrd/thread_proto.h"
|
|
|
|
using namespace Jrd;
|
|
|
|
struct FN {
|
|
const char* fn_module;
|
|
const char* fn_entrypoint;
|
|
FPTR_INT fn_function;
|
|
};
|
|
|
|
|
|
static int test(const long*, char*);
|
|
static DSC* ni(DSC*, DSC*);
|
|
static SLONG* byteLen(const dsc*);
|
|
static SLONG set_context(const vary* ns_vary, const vary* name_vary, const vary* value_vary);
|
|
static vary* get_context(const vary* ns_vary, const vary* name_vary);
|
|
|
|
// Context variable function names. Do not forget to change functions.h too if changing
|
|
static const char
|
|
RDB_GET_CONTEXT[] = "RDB$GET_CONTEXT",
|
|
RDB_SET_CONTEXT[] = "RDB$SET_CONTEXT";
|
|
|
|
// Namespace names
|
|
static const char
|
|
SYSTEM_NAMESPACE[] = "SYSTEM",
|
|
USER_SESSION_NAMESPACE[] = "USER_SESSION",
|
|
USER_TRANSACTION_NAMESPACE[] = "USER_TRANSACTION";
|
|
|
|
// System variables names
|
|
static const char
|
|
ENGINE_VERSION[] = "ENGINE_VERSION",
|
|
NETWORK_PROTOCOL_NAME[] = "NETWORK_PROTOCOL",
|
|
CLIENT_ADDRESS_NAME[] = "CLIENT_ADDRESS",
|
|
DATABASE_NAME[] = "DB_NAME",
|
|
ISOLATION_LEVEL_NAME[] = "ISOLATION_LEVEL",
|
|
TRANSACTION_ID_NAME[] = "TRANSACTION_ID",
|
|
SESSION_ID_NAME[] = "SESSION_ID",
|
|
CURRENT_USER_NAME[] = "CURRENT_USER",
|
|
CURRENT_ROLE_NAME[] = "CURRENT_ROLE";
|
|
|
|
// Isolation values modes
|
|
static const char
|
|
READ_COMMITTED_VALUE[] = "READ COMMITTED",
|
|
CONSISTENCY_VALUE[] = "CONSISTENCY",
|
|
SNAPSHOT_VALUE[] = "SNAPSHOT";
|
|
|
|
|
|
#define FUNCTION(ROUTINE, FUNCTION_NAME, MODULE_NAME, ENTRYPOINT, RET_ARG) \
|
|
{MODULE_NAME, ENTRYPOINT, (FPTR_INT) ROUTINE},
|
|
#define END_FUNCTION
|
|
#define FUNCTION_ARGUMENT(MECHANISM, TYPE, SCALE, LENGTH, SUB_TYPE, CHARSET, PRECISION, CHAR_LENGTH)
|
|
|
|
static const FN isc_functions[] = {
|
|
#ifndef DECLARE_EXAMPLE_IUDF_AUTOMATICALLY
|
|
{"test_module", "test_function", (FPTR_INT) test}, // see functions.h
|
|
#endif
|
|
{"test_module", "ni", (FPTR_INT) ni},
|
|
{"test_module", "ns", (FPTR_INT) ni},
|
|
{"test_module", "nn", (FPTR_INT) ni},
|
|
#ifndef DECLARE_EXAMPLE_IUDF_AUTOMATICALLY
|
|
{"test_module", "byte_len", (FPTR_INT) byteLen}, // see functions.h
|
|
#endif
|
|
|
|
#include "../jrd/functions.h"
|
|
|
|
{0, 0, 0}
|
|
};
|
|
|
|
|
|
FPTR_INT FUNCTIONS_entrypoint(const char* module, const char* entrypoint)
|
|
{
|
|
/**************************************
|
|
*
|
|
* F U N C T I O N S _ e n t r y p o i n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lookup function in hardcoded table. The module and
|
|
* entrypoint names are null terminated, but may contain
|
|
* insignificant trailing blanks.
|
|
*
|
|
**************************************/
|
|
char temp[MAXPATHLEN + 128]; /* Bug #126614 Fix */
|
|
|
|
char* p = temp;
|
|
|
|
while (*module && *module != ' ')
|
|
*p++ = *module++;
|
|
|
|
*p++ = 0;
|
|
const char* const ep = p;
|
|
|
|
while (*entrypoint && *entrypoint != ' ')
|
|
*p++ = *entrypoint++;
|
|
|
|
*p = 0;
|
|
|
|
for (const FN* function = isc_functions; function->fn_module; ++function) {
|
|
if (!strcmp(temp, function->fn_module)
|
|
&& !strcmp(ep, function->fn_entrypoint))
|
|
{
|
|
return function->fn_function;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static vary* make_result_str(const char* str, size_t str_len) {
|
|
vary *result_vary = (vary*) malloc(str_len + 2);
|
|
result_vary->vary_length = str_len;
|
|
memcpy(result_vary->vary_string, str, result_vary->vary_length);
|
|
return result_vary;
|
|
}
|
|
|
|
static vary* make_result_str(const Firebird::string& str) {
|
|
return make_result_str(str.c_str(), str.length());
|
|
}
|
|
|
|
vary* get_context(const vary* ns_vary, const vary* name_vary)
|
|
{
|
|
// Complain if namespace or variable name is null
|
|
if (!ns_vary || !name_vary) {
|
|
ERR_post(isc_ctx_bad_argument, isc_arg_string, RDB_GET_CONTEXT, 0);
|
|
}
|
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
|
|
Database* dbb;
|
|
Attachment *att;
|
|
jrd_tra* transaction;
|
|
|
|
// See if JRD thread data structure looks sane for occasion
|
|
if (!tdbb ||
|
|
!(dbb = tdbb->tdbb_database) ||
|
|
!(transaction = tdbb->tdbb_transaction) ||
|
|
!(att = tdbb->tdbb_attachment))
|
|
{
|
|
fb_assert(false);
|
|
return NULL;
|
|
}
|
|
|
|
Firebird::string ns_str(ns_vary->vary_string, ns_vary->vary_length),
|
|
name_str(name_vary->vary_string, name_vary->vary_length);
|
|
|
|
// Handle system variables
|
|
if (ns_str == SYSTEM_NAMESPACE)
|
|
{
|
|
if (name_str == ENGINE_VERSION)
|
|
{
|
|
Firebird::string version;
|
|
version.printf("%s.%s.%s", FB_MAJOR_VER, FB_MINOR_VER, FB_REV_NO);
|
|
|
|
return make_result_str(version);
|
|
}
|
|
|
|
if (name_str == NETWORK_PROTOCOL_NAME)
|
|
{
|
|
if (att->att_network_protocol.isEmpty())
|
|
return NULL;
|
|
|
|
return make_result_str(att->att_network_protocol);
|
|
}
|
|
|
|
if (name_str == CLIENT_ADDRESS_NAME)
|
|
{
|
|
if (att->att_remote_address.isEmpty())
|
|
return NULL;
|
|
|
|
return make_result_str(att->att_remote_address);
|
|
}
|
|
|
|
if (name_str == DATABASE_NAME)
|
|
{
|
|
return make_result_str(dbb->dbb_database_name.ToString());
|
|
}
|
|
|
|
if (name_str == CURRENT_USER_NAME)
|
|
{
|
|
if (!att->att_user || att->att_user->usr_user_name.isEmpty())
|
|
return NULL;
|
|
|
|
return make_result_str(att->att_user->usr_user_name);
|
|
}
|
|
|
|
if (name_str == CURRENT_ROLE_NAME)
|
|
{
|
|
if (!att->att_user || att->att_user->usr_sql_role_name.isEmpty())
|
|
return NULL;
|
|
|
|
return make_result_str(att->att_user->usr_sql_role_name);
|
|
}
|
|
|
|
if (name_str == SESSION_ID_NAME)
|
|
{
|
|
Firebird::string session_id;
|
|
|
|
// This synchronization is necessary to keep SuperServer happy.
|
|
// PAG_attachment_id() may modify database header, call lock manager, etc
|
|
THREAD_ENTER();
|
|
SLONG att_id = PAG_attachment_id(tdbb);
|
|
THREAD_EXIT();
|
|
|
|
session_id.printf("%d", att_id);
|
|
return make_result_str(session_id);
|
|
}
|
|
|
|
if (name_str == TRANSACTION_ID_NAME)
|
|
{
|
|
Firebird::string transaction_id;
|
|
transaction_id.printf("%d", transaction->tra_number);
|
|
return make_result_str(transaction_id);
|
|
}
|
|
|
|
if (name_str == ISOLATION_LEVEL_NAME)
|
|
{
|
|
const char* isolation;
|
|
|
|
if (transaction->tra_flags & TRA_read_committed)
|
|
isolation = READ_COMMITTED_VALUE;
|
|
else if (transaction->tra_flags & TRA_degree3)
|
|
isolation = CONSISTENCY_VALUE;
|
|
else
|
|
isolation = SNAPSHOT_VALUE;
|
|
|
|
return make_result_str(isolation, strlen(isolation));
|
|
}
|
|
|
|
// "Context variable %s is not found in namespace %s"
|
|
ERR_post(isc_ctx_var_not_found,
|
|
isc_arg_string, ERR_cstring(name_str.c_str()),
|
|
isc_arg_string, ERR_cstring(ns_str.c_str()), 0);
|
|
}
|
|
|
|
// Handle user-defined variables
|
|
if (ns_str == USER_SESSION_NAMESPACE)
|
|
{
|
|
Firebird::string result_str;
|
|
|
|
if (!att->att_context_vars.get(name_str, result_str))
|
|
return NULL;
|
|
|
|
return make_result_str(result_str);
|
|
}
|
|
|
|
if (ns_str == USER_TRANSACTION_NAMESPACE) {
|
|
Firebird::string result_str;
|
|
|
|
if (!transaction->tra_context_vars.get(name_str, result_str))
|
|
return NULL;
|
|
|
|
return make_result_str(result_str);
|
|
}
|
|
|
|
// "Invalid namespace name %s passed to %s"
|
|
ERR_post(isc_ctx_namespace_invalid,
|
|
isc_arg_string, ERR_cstring(ns_str.c_str()),
|
|
isc_arg_string, RDB_GET_CONTEXT, 0);
|
|
return NULL;
|
|
}
|
|
|
|
static SLONG set_context(const vary* ns_vary, const vary* name_vary, const vary* value_vary)
|
|
{
|
|
// Complain if namespace or variable name is null
|
|
if (!ns_vary || !name_vary)
|
|
{
|
|
ERR_post(isc_ctx_bad_argument, isc_arg_string, RDB_SET_CONTEXT, 0);
|
|
}
|
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
|
|
if (!tdbb) {
|
|
// Something is seriously wrong
|
|
fb_assert(false);
|
|
return 0;
|
|
}
|
|
|
|
Firebird::string ns_str(ns_vary->vary_string, ns_vary->vary_length),
|
|
name_str(name_vary->vary_string, name_vary->vary_length);
|
|
|
|
if (ns_str == USER_SESSION_NAMESPACE)
|
|
{
|
|
Attachment* att = tdbb->tdbb_attachment;
|
|
|
|
if (!att) {
|
|
fb_assert(false);
|
|
return 0;
|
|
}
|
|
|
|
if (!value_vary)
|
|
return att->att_context_vars.remove(name_str);
|
|
|
|
if (att->att_context_vars.count() >= MAX_CONTEXT_VARS) {
|
|
// "Too many context variables"
|
|
ERR_post(isc_ctx_too_big, 0);
|
|
}
|
|
|
|
return att->att_context_vars.put(name_str,
|
|
Firebird::string(value_vary->vary_string, value_vary->vary_length));
|
|
}
|
|
else if (ns_str == USER_TRANSACTION_NAMESPACE) {
|
|
jrd_tra* tra = tdbb->tdbb_transaction;
|
|
|
|
if (!tra) {
|
|
fb_assert(false);
|
|
return 0;
|
|
}
|
|
|
|
if (!value_vary)
|
|
return tra->tra_context_vars.remove(name_str);
|
|
|
|
if (tra->tra_context_vars.count() >= MAX_CONTEXT_VARS) {
|
|
// "Too many context variables"
|
|
ERR_post(isc_ctx_too_big, 0);
|
|
}
|
|
|
|
return tra->tra_context_vars.put(name_str,
|
|
Firebird::string(value_vary->vary_string, value_vary->vary_length));
|
|
}
|
|
else {
|
|
// "Invalid namespace name %s passed to %s"
|
|
ERR_post(isc_ctx_namespace_invalid,
|
|
isc_arg_string, ERR_cstring(ns_str.c_str()),
|
|
isc_arg_string, RDB_SET_CONTEXT, 0);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static int test(const long* n, char *result)
|
|
{
|
|
/**************************************
|
|
*
|
|
* t e s t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Sample extern function. Defined in database by:
|
|
*
|
|
* QLI:
|
|
* define function test module_name "test_module" entry_point "test_function"
|
|
* long by reference //by value, CVC: BY VALUE is deprecated for input params
|
|
* char [20] by reference return_argument;
|
|
* ISQL:
|
|
* declare external function test
|
|
* int null, // CVC: with NULL signaling
|
|
* char(20) returns parameter 2
|
|
* entry_point 'test_function' module_name 'test_module';
|
|
*
|
|
**************************************/
|
|
|
|
if (n)
|
|
sprintf(result, "%ld is a number", *n);
|
|
else
|
|
sprintf(result, "is NULL");
|
|
const char* const end = result + 20;
|
|
|
|
while (*result)
|
|
result++;
|
|
|
|
while (result < end)
|
|
*result++ = ' ';
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static dsc* ni(dsc* v, dsc* v2)
|
|
{
|
|
if (v)
|
|
return v;
|
|
else
|
|
return v2;
|
|
}
|
|
|
|
|
|
// byteLen: return the length in bytes of a given argument. For NULL, return NULL, too.
|
|
// v = input descriptor
|
|
// rc = return value, allocated dynamically. To be freed by the engine.
|
|
// The declaration through SQL is:
|
|
// declare external function sys_byte_len
|
|
// int by descriptor
|
|
// returns int free_it
|
|
// entry_point 'byte_len' module_name 'test_module';
|
|
static SLONG* byteLen(const dsc* v)
|
|
{
|
|
if (!v || !v->dsc_address || (v->dsc_flags & DSC_null))
|
|
return 0;
|
|
else
|
|
{
|
|
SLONG& rc = *(SLONG*) malloc(sizeof(SLONG));
|
|
switch (v->dsc_dtype)
|
|
{
|
|
case dtype_text:
|
|
{
|
|
const UCHAR* const ini = v->dsc_address;
|
|
const UCHAR* end = ini + v->dsc_length;
|
|
while (ini < end && *--end == ' '); // empty loop body
|
|
rc = end - ini + 1;
|
|
break;
|
|
}
|
|
case dtype_cstring:
|
|
{
|
|
rc = 0;
|
|
for (const UCHAR* p = v->dsc_address; *p; ++p, ++rc); // empty loop body
|
|
break;
|
|
}
|
|
case dtype_varying:
|
|
rc = reinterpret_cast<const vary*>(v->dsc_address)->vary_length;
|
|
break;
|
|
default:
|
|
rc = DSC_string_length(v);
|
|
break;
|
|
}
|
|
|
|
return &rc;
|
|
}
|
|
}
|