/* * 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 #include #include "../jrd/jrd.h" /* For MAXPATHLEN Bug #126614 */ #include "../jrd/license.h" #include "../jrd/tra.h" #include "../jrd/dsc_proto.h" #include "../jrd/fun_proto.h" #include "../jrd/pag_proto.h" #include "../jrd/thread_proto.h" using namespace Jrd; using namespace Firebird; 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*) IbUtil::alloc(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) { 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->getDatabase()) || !(transaction = tdbb->getTransaction()) || !(att = tdbb->getAttachment())) { fb_assert(false); return NULL; } Database::SyncGuard dsGuard(dbb); // Complain if namespace or variable name is null if (!ns_vary || !name_vary) { ERR_post(Arg::Gds(isc_ctx_bad_argument) << Arg::Str(RDB_GET_CONTEXT)); } const Firebird::string ns_str(ns_vary->vary_string, ns_vary->vary_length); const Firebird::string 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; const SLONG att_id = PAG_attachment_id(tdbb); 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(Arg::Gds(isc_ctx_var_not_found) << Arg::Str(name_str) << Arg::Str(ns_str)); } // 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(Arg::Gds(isc_ctx_namespace_invalid) << Arg::Str(ns_str) << Arg::Str(RDB_GET_CONTEXT)); 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(Arg::Gds(isc_ctx_bad_argument) << Arg::Str(RDB_SET_CONTEXT)); } thread_db* tdbb = JRD_get_thread_data(); if (!tdbb) { // Something is seriously wrong fb_assert(false); return 0; } const Firebird::string ns_str(ns_vary->vary_string, ns_vary->vary_length); const Firebird::string name_str(name_vary->vary_string, name_vary->vary_length); if (ns_str == USER_SESSION_NAMESPACE) { Attachment* att = tdbb->getAttachment(); 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(Arg::Gds(isc_ctx_too_big)); } return att->att_context_vars.put(name_str, Firebird::string(value_vary->vary_string, value_vary->vary_length)); } if (ns_str == USER_TRANSACTION_NAMESPACE) { jrd_tra* tra = tdbb->getTransaction(); 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(Arg::Gds(isc_ctx_too_big)); } return tra->tra_context_vars.put(name_str, Firebird::string(value_vary->vary_string, value_vary->vary_length)); } // "Invalid namespace name %s passed to %s" ERR_post(Arg::Gds(isc_ctx_namespace_invalid) << Arg::Str(ns_str) << Arg::Str(RDB_SET_CONTEXT)); 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) { return v ? v : 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; SLONG& rc = *(SLONG*) IbUtil::alloc(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(v->dsc_address)->vary_length; break; default: rc = DSC_string_length(v); break; } return &rc; }