8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-30 20:03:03 +01:00
firebird-mirror/src/jrd/met.epp

5064 lines
133 KiB
Plaintext
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
* MODULE: met.epp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Meta data handler
*
* 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): ______________________________________.
*
2002-07-01 18:59:09 +02:00
* 2001.6.25 Claudio Valderrama: Finish MET_lookup_generator_id() by
* assigning it a number in the compiled requests table.
*
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
* conditionals, as the engine now fully supports
* readonly databases.
* 2001.10.03 Claudio Valderrama: MET_relation_owns_trigger() determines if
* there's a row in rdb$triggers with the given relation's and trigger's names.
* 2001.10.04 Claudio Valderrama: MET_relation_default_class() determines if the
* given security class name is the default security class for a relation.
* Modify MET_lookup_field() so it can verify the field's security class, too.
* 2002-02-24 Sean Leyne - Code Cleanup of old Win 3.1 port (WINDOWS_ONLY)
* 2002-09-16 Nickolay Samofatov - Deferred trigger compilation changes
2002-10-30 07:40:58 +01:00
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
2005-05-28 00:45:31 +02:00
* 2004.01.16 Vlad Horsun: added support for default parameters
2001-05-23 15:26:42 +02:00
*/
#include "firebird.h"
2004-04-29 00:43:34 +02:00
#include <stdio.h>
2001-05-23 15:26:42 +02:00
#include <string.h>
#include <stdarg.h>
2003-11-08 17:40:17 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/jrd.h"
#include "../jrd/val.h"
#include "../jrd/irq.h"
#include "../jrd/tra.h"
#include "../jrd/lck.h"
#include "../jrd/ods.h"
#include "../jrd/btr.h"
#include "../jrd/req.h"
#include "../jrd/exe.h"
#include "../jrd/scl.h"
#include "../jrd/blb.h"
#include "../jrd/met.h"
2003-07-14 12:35:49 +02:00
#include "../jrd/os/pio.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/sdw.h"
#include "../jrd/flags.h"
#include "../jrd/lls.h"
#include "../jrd/intl.h"
#include "../jrd/align.h"
#include "../jrd/flu.h"
#include "../jrd/blob_filter.h"
#include "../dsql/StmtNodes.h"
2001-05-23 15:26:42 +02:00
#include "../intl/charsets.h"
#include "../common/gdsassert.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/blb_proto.h"
#include "../jrd/cmp_proto.h"
#include "../jrd/dfw_proto.h"
2010-10-12 10:02:57 +02:00
#include "../common/dsc_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/err_proto.h"
#include "../jrd/evl_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/exe_proto.h"
#include "../jrd/ext_proto.h"
#include "../jrd/flu_proto.h"
2010-10-12 10:02:57 +02:00
#include "../yvalve/gds_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/idx_proto.h"
#include "../jrd/ini_proto.h"
#include "../jrd/lck_proto.h"
#include "../jrd/met_proto.h"
#include "../jrd/mov_proto.h"
#include "../jrd/par_proto.h"
#include "../jrd/pcmet_proto.h"
2003-07-14 12:35:49 +02:00
#include "../jrd/os/pio_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/scl_proto.h"
#include "../jrd/sdw_proto.h"
#include "../jrd/thread_proto.h"
2003-12-31 06:36:12 +01:00
#include "../common/utils_proto.h"
#include "../jrd/PreparedStatement.h"
#include "../jrd/RecordSourceNodes.h"
#include "../jrd/DebugInterface.h"
#include "../common/classes/MsgPrint.h"
#include "../jrd/Function.h"
2001-05-23 15:26:42 +02:00
#ifdef HAVE_CTYPE_H
#include <ctype.h>
#endif
2009-11-21 04:23:47 +01:00
// Pick up relation ids
#include "../jrd/ini.h"
2001-05-23 15:26:42 +02:00
DATABASE DB = FILENAME "ODS.RDB";
using namespace Jrd;
using namespace Firebird;
static int blocking_ast_dsql_cache(void* ast_object);
static DSqlCacheItem* get_dsql_cache_item(thread_db* tdbb, int type, const QualifiedName& name);
static int blocking_ast_procedure(void*);
static int blocking_ast_relation(void*);
static int partners_ast_relation(void*);
static int rescan_ast_relation(void*);
static ULONG get_rel_flags_from_FLAGS(USHORT);
static void get_trigger(thread_db*, jrd_rel*, bid*, bid*, trig_vec**, const TEXT*, FB_UINT64, bool,
2010-06-12 18:32:46 +02:00
USHORT, const MetaName&, const string&, const bid*);
2006-04-05 18:34:18 +02:00
static bool get_type(thread_db*, USHORT*, const UCHAR*, const TEXT*);
static void lookup_view_contexts(thread_db*, jrd_rel*);
2010-06-12 18:32:46 +02:00
static void make_relation_scope_name(const TEXT*, const USHORT, string& str);
static ValueExprNode* parse_field_default_blr(thread_db* tdbb, bid* blob_id);
2013-03-31 20:23:54 +02:00
static BoolExprNode* parse_field_validation_blr(thread_db* tdbb, bid* blob_id, const MetaName name);
2008-12-18 11:57:12 +01:00
static bool resolve_charset_and_collation(thread_db*, USHORT*, const UCHAR*, const UCHAR*);
static void save_trigger_data(thread_db*, trig_vec**, jrd_rel*, JrdStatement*, blb*, bid*,
2010-06-12 18:32:46 +02:00
const TEXT*, FB_UINT64, bool, USHORT, const MetaName&, const string&,
const bid*);
static void scan_partners(thread_db*, jrd_rel*);
static void store_dependencies(thread_db*, CompilerScratch*, const jrd_rel*,
2010-06-12 18:32:46 +02:00
const MetaName&, int, jrd_tra*);
static bool verify_TRG_ignore_perm(thread_db*, const MetaName&);
2001-05-23 15:26:42 +02:00
// Decompile all triggers from vector
static void release_cached_triggers(thread_db* tdbb, trig_vec* vector)
{
2005-05-28 00:45:31 +02:00
if (!vector)
2004-04-19 17:29:29 +02:00
{
2003-12-22 11:00:59 +01:00
return;
}
2005-05-28 00:45:31 +02:00
for (size_t i = 0; i < vector->getCount(); i++)
2004-04-19 17:29:29 +02:00
{
(*vector)[i].release(tdbb);
}
}
2004-05-09 07:48:33 +02:00
static void inc_int_use_count(JrdStatement* statement)
2004-05-09 07:48:33 +02:00
{
// Increment int_use_count for all procedures in resource list of request
ResourceList& list = statement->resources;
size_t i;
for (list.find(Resource(Resource::rsc_procedure, 0, NULL, NULL, NULL), i);
i < list.getCount(); i++)
{
2004-05-09 07:48:33 +02:00
Resource& resource = list[i];
2005-05-28 00:45:31 +02:00
if (resource.rsc_type != Resource::rsc_procedure)
break;
fb_assert(resource.rsc_prc->prc_int_use_count >= 0);
++resource.rsc_prc->prc_int_use_count;
}
}
2004-05-09 07:48:33 +02:00
// Increment int_use_count for all procedures used by triggers
2004-04-19 17:29:29 +02:00
static void post_used_procedures(trig_vec* vector)
{
2005-05-28 00:45:31 +02:00
if (!vector)
2004-04-19 17:29:29 +02:00
{
2003-12-22 11:00:59 +01:00
return;
}
2005-05-28 00:45:31 +02:00
for (size_t i = 0; i < vector->getCount(); i++)
2004-04-19 17:29:29 +02:00
{
JrdStatement* stmt = (*vector)[i].statement;
if (stmt && !stmt->isActive())
inc_int_use_count(stmt);
2004-04-19 17:29:29 +02:00
}
}
2004-05-09 07:48:33 +02:00
void MET_get_domain(thread_db* tdbb, MemoryPool& csbPool, const MetaName& name, dsc* desc,
FieldInfo* fieldInfo)
{
/**************************************
*
* M E T _ g e t _ d o m a i n
*
**************************************
*
* Functional description
* Get domain descriptor and informations.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
bool found = false;
AutoCacheRequest handle(tdbb, irq_l_domain, IRQ_REQUESTS);
FOR(REQUEST_HANDLE handle)
FLD IN RDB$FIELDS WITH FLD.RDB$FIELD_NAME EQ name.c_str()
{
if (DSC_make_descriptor(desc,
FLD.RDB$FIELD_TYPE,
FLD.RDB$FIELD_SCALE,
FLD.RDB$FIELD_LENGTH,
FLD.RDB$FIELD_SUB_TYPE,
FLD.RDB$CHARACTER_SET_ID,
FLD.RDB$COLLATION_ID))
{
2008-01-16 10:29:37 +01:00
found = true;
if (fieldInfo)
{
fieldInfo->nullable = FLD.RDB$NULL_FLAG.NULL || FLD.RDB$NULL_FLAG == 0;
Jrd::ContextPoolHolder context(tdbb, &csbPool);
if (FLD.RDB$DEFAULT_VALUE.NULL)
fieldInfo->defaultValue = NULL;
else
fieldInfo->defaultValue = parse_field_default_blr(tdbb, &FLD.RDB$DEFAULT_VALUE);
if (FLD.RDB$VALIDATION_BLR.NULL)
fieldInfo->validationExpr = NULL;
else
2008-01-16 10:29:37 +01:00
{
fieldInfo->validationExpr = parse_field_validation_blr(tdbb,
&FLD.RDB$VALIDATION_BLR, name);
2008-01-16 10:29:37 +01:00
}
}
}
}
END_FOR
2008-01-16 10:29:37 +01:00
if (!found)
{
ERR_post(Arg::Gds(isc_domnotdef) << Arg::Str(name));
2008-01-16 10:29:37 +01:00
}
}
MetaName MET_get_relation_field(thread_db* tdbb, MemoryPool& csbPool, const MetaName& relationName,
const MetaName& fieldName, dsc* desc, FieldInfo* fieldInfo)
2008-01-16 10:29:37 +01:00
{
/**************************************
*
* M E T _ g e t _ r e l a t i o n _ f i e l d
*
**************************************
*
* Functional description
* Get relation field descriptor and informations.
* Returns field source name.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2008-01-16 10:29:37 +01:00
bool found = false;
2010-06-12 18:32:46 +02:00
MetaName sourceName;
AutoCacheRequest handle(tdbb, irq_l_relfield, IRQ_REQUESTS);
2008-01-16 10:29:37 +01:00
FOR(REQUEST_HANDLE handle)
RFL IN RDB$RELATION_FIELDS CROSS
FLD IN RDB$FIELDS WITH
RFL.RDB$RELATION_NAME EQ relationName.c_str() AND
RFL.RDB$FIELD_NAME EQ fieldName.c_str() AND
FLD.RDB$FIELD_NAME EQ RFL.RDB$FIELD_SOURCE
{
2008-01-16 10:29:37 +01:00
if (DSC_make_descriptor(desc,
FLD.RDB$FIELD_TYPE,
FLD.RDB$FIELD_SCALE,
FLD.RDB$FIELD_LENGTH,
FLD.RDB$FIELD_SUB_TYPE,
FLD.RDB$CHARACTER_SET_ID,
(RFL.RDB$COLLATION_ID.NULL ? FLD.RDB$COLLATION_ID : RFL.RDB$COLLATION_ID)))
2008-01-16 10:29:37 +01:00
{
found = true;
sourceName = RFL.RDB$FIELD_SOURCE;
if (fieldInfo)
{
fieldInfo->nullable = RFL.RDB$NULL_FLAG.NULL ?
2008-12-18 11:57:12 +01:00
(FLD.RDB$NULL_FLAG.NULL || FLD.RDB$NULL_FLAG == 0) : RFL.RDB$NULL_FLAG == 0;
2008-01-16 10:29:37 +01:00
Jrd::ContextPoolHolder context(tdbb, &csbPool);
bid* defaultId = NULL;
if (!RFL.RDB$DEFAULT_VALUE.NULL)
defaultId = &RFL.RDB$DEFAULT_VALUE;
else if (!FLD.RDB$DEFAULT_VALUE.NULL)
defaultId = &FLD.RDB$DEFAULT_VALUE;
if (defaultId)
fieldInfo->defaultValue = parse_field_default_blr(tdbb, defaultId);
else
fieldInfo->defaultValue = NULL;
if (FLD.RDB$VALIDATION_BLR.NULL)
fieldInfo->validationExpr = NULL;
else
{
fieldInfo->validationExpr = parse_field_validation_blr(tdbb,
&FLD.RDB$VALIDATION_BLR, RFL.RDB$FIELD_SOURCE);
}
}
}
}
END_FOR
if (!found)
{
ERR_post(Arg::Gds(isc_dyn_column_does_not_exist) << Arg::Str(fieldName) <<
Arg::Str(relationName));
}
2008-01-16 10:29:37 +01:00
return sourceName;
}
void MET_update_partners(thread_db* tdbb)
{
/**************************************
*
* M E T _ u p d a t e _ p a r t n e r s
*
**************************************
*
* Functional description
* Mark all relations to update their links to FK partners
* Called when any index is deleted because engine don't know
* was it used in any FK or not
*
**************************************/
SET_TDBB(tdbb);
2011-05-09 12:15:19 +02:00
Jrd::Attachment* attachment = tdbb->getAttachment();
2011-05-09 12:15:19 +02:00
vec<jrd_rel*>* relations = attachment->att_relations;
vec<jrd_rel*>::iterator ptr = relations->begin();
2006-02-10 04:28:43 +01:00
for (const vec<jrd_rel*>::const_iterator end = relations->end(); ptr < end; ++ptr)
{
jrd_rel* relation = *ptr;
if (!relation) {
2003-12-22 11:00:59 +01:00
continue;
}
// signal other processes
LCK_lock(tdbb, relation->rel_partners_lock, LCK_EX, LCK_WAIT);
LCK_release(tdbb, relation->rel_partners_lock);
relation->rel_flags |= REL_check_partners;
}
}
2004-05-09 07:48:33 +02:00
void adjust_dependencies(jrd_prc* procedure)
{
2009-11-07 12:58:54 +01:00
if (procedure->prc_int_use_count == -1)
{
// Already processed
return;
}
2003-12-22 11:00:59 +01:00
procedure->prc_int_use_count = -1; // Mark as undeletable
if (procedure->getStatement())
2009-11-07 12:58:54 +01:00
{
// Loop over procedures from resource list of request
ResourceList& list = procedure->getStatement()->resources;
size_t i;
for (list.find(Resource(Resource::rsc_procedure, 0, NULL, NULL, NULL), i);
2008-12-18 11:57:12 +01:00
i < list.getCount(); i++)
{
2004-05-09 07:48:33 +02:00
Resource& resource = list[i];
2005-05-28 00:45:31 +02:00
if (resource.rsc_type != Resource::rsc_procedure)
break;
procedure = resource.rsc_prc;
2009-11-07 12:58:54 +01:00
if (procedure->prc_int_use_count == procedure->prc_use_count)
{
// Mark it and all dependent procedures as undeletable
2005-05-28 00:45:31 +02:00
adjust_dependencies(procedure);
}
}
}
}
2004-05-09 07:48:33 +02:00
#ifdef DEV_BUILD
void MET_verify_cache(thread_db* tdbb)
{
/**************************************
*
* M E T _ v e r i f y _ c a c h e
*
**************************************
*
* Functional description
* Check if all links between procedures are properly counted
*
**************************************/
SET_TDBB(tdbb);
2011-05-09 12:15:19 +02:00
Jrd::Attachment* att = tdbb->getAttachment();
if (!att)
return;
2005-05-28 00:45:31 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* procedures = att->att_procedures;
2009-11-07 12:58:54 +01:00
if (procedures)
{
2003-12-22 11:00:59 +01:00
jrd_prc* procedure;
vec<jrd_prc*>::iterator ptr, end;
2005-05-28 00:45:31 +02:00
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ++ptr)
{
if ( (procedure = *ptr) && procedure->getStatement() /*&&
2005-05-28 00:45:31 +02:00
!(procedure->prc_flags & PRC_obsolete)*/ )
{
2003-11-04 00:59:24 +01:00
fb_assert(procedure->prc_int_use_count == 0);
}
}
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// Walk procedures and calculate internal dependencies
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) && procedure->getStatement() /*&&
2005-05-28 00:45:31 +02:00
!(procedure->prc_flags & PRC_obsolete)*/ )
{
inc_int_use_count(procedure->getStatement());
}
}
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// Walk procedures again and check dependencies
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) && procedure->getStatement() && /*
!(procedure->prc_flags & PRC_obsolete) && */
procedure->prc_use_count < procedure->prc_int_use_count)
{
char buffer[1024], *buf = buffer;
2005-05-28 00:45:31 +02:00
buf += sprintf(buf, "Procedure %d:%s is not properly counted (use count=%d, prc use=%d). Used by: \n",
procedure->getId(), procedure->getName().toString().c_str(),
procedure->prc_use_count, procedure->prc_int_use_count);
vec<jrd_prc*>::const_iterator ptr2 = procedures->begin();
for (const vec<jrd_prc*>::const_iterator end2 = procedures->end();
2006-02-10 04:28:43 +01:00
ptr2 < end2; ++ptr2)
{
const jrd_prc* prc = *ptr2;
if (prc && prc->getStatement() /*&& !(prc->prc_flags & PRC_obsolete)*/ )
{
// Loop over procedures from resource list of request
const ResourceList& list = prc->getStatement()->resources;
size_t i;
for (list.find(Resource(Resource::rsc_procedure, 0, NULL, NULL, NULL), i);
2005-05-28 00:45:31 +02:00
i < list.getCount(); i++)
{
const Resource& resource = list[i];
2005-05-28 00:45:31 +02:00
if (resource.rsc_type != Resource::rsc_procedure)
break;
if (resource.rsc_prc == procedure)
{
2009-12-28 12:46:55 +01:00
// Do not enable this code in production builds unless
// the possible B.O. is fixed here.
buf += sprintf(buf, "%d:%s\n", prc->getId(),
prc->getName().toString().c_str());
}
}
}
}
gds__log(buffer);
2003-11-04 00:59:24 +01:00
fb_assert(false);
}
}
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// Fix back int_use_count
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) )
2005-05-28 00:45:31 +02:00
{
procedure->prc_int_use_count = 0;
}
2005-05-28 00:45:31 +02:00
}
}
}
#endif
2004-05-09 07:48:33 +02:00
2004-04-10 21:46:48 +02:00
void MET_clear_cache(thread_db* tdbb)
{
/**************************************
*
* M E T _ c l e a r _ c a c h e
*
**************************************
*
* Functional description
2005-05-28 00:45:31 +02:00
* Remove all unused objects from metadata cache to
2004-04-10 21:46:48 +02:00
* release resources they use
*
**************************************/
SET_TDBB(tdbb);
#ifdef DEV_BUILD
MET_verify_cache(tdbb);
#endif
2003-12-22 11:00:59 +01:00
2011-05-09 12:15:19 +02:00
Jrd::Attachment* att = tdbb->getAttachment();
2003-12-22 11:00:59 +01:00
2013-11-19 14:19:11 +01:00
for (unsigned i = 0; i < DB_TRIGGER_MAX; i++) {
2011-05-09 12:15:19 +02:00
release_cached_triggers(tdbb, att->att_triggers[i]);
}
2011-05-09 12:15:19 +02:00
vec<jrd_rel*>* relations = att->att_relations;
2009-03-15 21:09:15 +01:00
if (relations)
{ // scope
vec<jrd_rel*>::iterator ptr, end;
2006-02-10 04:28:43 +01:00
for (ptr = relations->begin(), end = relations->end(); ptr < end; ++ptr)
{
jrd_rel* relation = *ptr;
if (!relation)
continue;
release_cached_triggers(tdbb, relation->rel_pre_store);
release_cached_triggers(tdbb, relation->rel_post_store);
release_cached_triggers(tdbb, relation->rel_pre_erase);
release_cached_triggers(tdbb, relation->rel_post_erase);
release_cached_triggers(tdbb, relation->rel_pre_modify);
release_cached_triggers(tdbb, relation->rel_post_modify);
}
} // scope
2005-05-28 00:45:31 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* procedures = att->att_procedures;
if (procedures)
{
2003-12-22 11:00:59 +01:00
jrd_prc* procedure;
vec<jrd_prc*>::iterator ptr, end;
2009-11-21 04:23:47 +01:00
// Walk procedures and calculate internal dependencies
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ++ptr)
{
if ( (procedure = *ptr) && procedure->getStatement() &&
2005-05-28 00:45:31 +02:00
!(procedure->prc_flags & PRC_obsolete) )
{
inc_int_use_count(procedure->getStatement());
}
}
2005-05-28 00:45:31 +02:00
// Walk procedures again and adjust dependencies for procedures
// which will not be removed.
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) && procedure->getStatement() &&
2008-12-18 11:57:12 +01:00
!(procedure->prc_flags & PRC_obsolete) &&
procedure->prc_use_count != procedure->prc_int_use_count )
{
adjust_dependencies(procedure);
}
}
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// Deallocate all used requests
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) )
{
2005-05-28 00:45:31 +02:00
if ( procedure->getStatement() && !(procedure->prc_flags & PRC_obsolete) &&
2005-05-28 00:45:31 +02:00
procedure->prc_int_use_count >= 0 &&
procedure->prc_use_count == procedure->prc_int_use_count )
{
procedure->releaseStatement(tdbb);
if (procedure->prc_existence_lock) //// TODO: verify why this IF is necessary now
LCK_release(tdbb, procedure->prc_existence_lock);
procedure->prc_existence_lock = NULL;
procedure->prc_flags |= PRC_obsolete;
}
// Leave it in state 0 to avoid extra pass next time to clear it
// Note: we need to adjust prc_int_use_count for all procedures
// in cache because any of them may have been affected from
// dependencies earlier. Even procedures that were not scanned yet !
procedure->prc_int_use_count = 0;
}
}
}
#ifdef DEV_BUILD
MET_verify_cache(tdbb);
#endif
}
2004-05-09 07:48:33 +02:00
bool MET_procedure_in_use(thread_db* tdbb, jrd_prc* proc)
{
/**************************************
*
* M E T _ p r o c e d u r e _ i n _ u s e
*
**************************************
*
* Functional description
2005-05-28 00:45:31 +02:00
* Determine if procedure is used by any user requests or transactions.
* Return false if procedure is used only inside cache or not used at all.
*
**************************************/
SET_TDBB(tdbb);
#ifdef DEV_BUILD
MET_verify_cache(tdbb);
#endif
2011-05-09 12:15:19 +02:00
Jrd::Attachment* att = tdbb->getAttachment();
2005-05-28 00:45:31 +02:00
// This should not really happen
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* procedures = att->att_procedures;
if (!procedures) {
return false;
}
2011-05-09 12:15:19 +02:00
vec<jrd_rel*>* relations = att->att_relations;
{ // scope
vec<jrd_rel*>::iterator ptr, end;
2006-02-10 04:28:43 +01:00
for (ptr = relations->begin(), end = relations->end(); ptr < end; ++ptr)
{
jrd_rel* relation = *ptr;
if (!relation) {
continue;
}
post_used_procedures(relation->rel_pre_store);
post_used_procedures(relation->rel_post_store);
post_used_procedures(relation->rel_pre_erase);
post_used_procedures(relation->rel_post_erase);
post_used_procedures(relation->rel_pre_modify);
post_used_procedures(relation->rel_post_modify);
}
} // scope
2003-12-22 11:00:59 +01:00
jrd_prc* procedure;
vec<jrd_prc*>::iterator ptr, end;
2009-11-21 04:23:47 +01:00
// Walk procedures and calculate internal dependencies
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ++ptr)
{
if ((procedure = *ptr) && procedure->getStatement() && !(procedure->prc_flags & PRC_obsolete))
inc_int_use_count(procedure->getStatement());
}
2005-05-28 00:45:31 +02:00
// Walk procedures again and adjust dependencies for procedures
// which will not be removed.
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) && procedure->getStatement() &&
2008-12-18 11:57:12 +01:00
!(procedure->prc_flags & PRC_obsolete) &&
procedure->prc_use_count != procedure->prc_int_use_count && procedure != proc )
{
adjust_dependencies(procedure);
}
}
2005-05-28 00:45:31 +02:00
const bool result = proc->prc_use_count != proc->prc_int_use_count;
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// Fix back int_use_count
2008-12-18 11:57:12 +01:00
for (ptr = procedures->begin(), end = procedures->end(); ptr < end; ptr++)
{
if ( (procedure = *ptr) )
2005-05-28 00:45:31 +02:00
{
procedure->prc_int_use_count = 0;
}
}
#ifdef DEV_BUILD
MET_verify_cache(tdbb);
#endif
return result;
}
2001-12-24 03:51:06 +01:00
void MET_activate_shadow(thread_db* tdbb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ a c t i v a t e _ s h a d o w
*
**************************************
*
* Functional description
* Activate the current database, which presumably
* was formerly a shadow, by deleting all records
* corresponding to the shadow that this database
* represents.
* Get rid of write ahead log for the activated shadow.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Erase any secondary files of the primary database of the shadow being activated.
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle) X IN RDB$FILES
WITH X.RDB$SHADOW_NUMBER NOT MISSING
2005-05-28 00:45:31 +02:00
AND X.RDB$SHADOW_NUMBER EQ 0
2008-12-18 11:57:12 +01:00
ERASE X;
END_FOR
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
PageSpace* pageSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
const char* dbb_file_name = pageSpace->file->fil_string;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// go through files looking for any that expand to the current database name
2003-12-22 11:00:59 +01:00
SCHAR expanded_name[MAXPATHLEN];
AutoRequest handle2;
handle.reset();
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle) X IN RDB$FILES
WITH X.RDB$SHADOW_NUMBER NOT MISSING
AND X.RDB$SHADOW_NUMBER NE 0
2008-12-18 11:57:12 +01:00
PIO_expand(X.RDB$FILE_NAME, (USHORT)strlen(X.RDB$FILE_NAME),
expanded_name, sizeof(expanded_name));
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (!strcmp(expanded_name, dbb_file_name))
{
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle2) Y IN RDB$FILES
WITH X.RDB$SHADOW_NUMBER EQ Y.RDB$SHADOW_NUMBER
MODIFY Y
Y.RDB$SHADOW_NUMBER = 0;
END_MODIFY
END_FOR
2010-04-23 03:59:21 +02:00
2001-05-23 15:26:42 +02:00
ERASE X;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
2009-11-27 07:59:21 +01:00
ULONG MET_align(const dsc* desc, ULONG value)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ a l i g n
*
**************************************
*
* Functional description
* Align value (presumed offset) on appropriate border
* and return.
*
**************************************/
2003-12-22 11:00:59 +01:00
USHORT alignment = desc->dsc_length;
2009-01-20 09:33:59 +01:00
switch (desc->dsc_dtype)
{
2001-05-23 15:26:42 +02:00
case dtype_text:
case dtype_cstring:
return value;
case dtype_varying:
alignment = sizeof(USHORT);
break;
}
alignment = MIN(alignment, FORMAT_ALIGNMENT);
2001-05-23 15:26:42 +02:00
return FB_ALIGN(value, alignment);
}
2006-07-12 03:38:41 +02:00
DeferredWork* MET_change_fields(thread_db* tdbb, jrd_tra* transaction, const dsc* field_source)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ c h a n g e _ f i e l d s
*
**************************************
*
* Functional description
* Somebody is modifying RDB$FIELDS.
* Find all relations affected and schedule a format update.
* Find all procedures and triggers and schedule a BLR validate.
2009-12-26 14:22:52 +01:00
* Find all functions and schedule a BLR validate.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
dsc relation_name;
DeferredWork* dw = NULL;
AutoCacheRequest request(tdbb, irq_m_fields, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$RELATION_FIELDS WITH
2002-12-22 14:08:50 +01:00
X.RDB$FIELD_SOURCE EQ field_source->dsc_address
{
2009-11-28 20:39:23 +01:00
relation_name.makeText(sizeof(X.RDB$RELATION_NAME), CS_METADATA,
(UCHAR*) X.RDB$RELATION_NAME);
SCL_check_relation(tdbb, &relation_name, SCL_control);
dw = DFW_post_work(transaction, dfw_update_format, &relation_name, 0);
AutoCacheRequest request2(tdbb, irq_m_fields4, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request2)
RFL IN RDB$RELATION_FIELDS CROSS
DEP IN RDB$DEPENDENCIES CROSS
PRC IN RDB$PROCEDURES
WITH RFL.RDB$FIELD_SOURCE EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_NAME EQ RFL.RDB$RELATION_NAME AND
DEP.RDB$FIELD_NAME EQ RFL.RDB$FIELD_NAME AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_relation AND
DEP.RDB$DEPENDENT_TYPE EQ obj_procedure AND
DEP.RDB$DEPENDENT_NAME EQ PRC.RDB$PROCEDURE_NAME AND
PRC.RDB$PACKAGE_NAME MISSING
{
MetaName proc_name(PRC.RDB$PROCEDURE_NAME);
dsc desc;
desc.makeText(proc_name.length(), CS_METADATA, (UCHAR*) proc_name.c_str());
DeferredWork* dw2 =
DFW_post_work(transaction, dfw_modify_procedure, &desc, PRC.RDB$PROCEDURE_ID);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
request2.reset(tdbb, irq_m_fields5, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request2)
RFL IN RDB$RELATION_FIELDS CROSS
DEP IN RDB$DEPENDENCIES CROSS
TRG IN RDB$TRIGGERS
WITH RFL.RDB$FIELD_SOURCE EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_NAME EQ RFL.RDB$RELATION_NAME AND
DEP.RDB$FIELD_NAME EQ RFL.RDB$FIELD_NAME AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_relation AND
DEP.RDB$DEPENDENT_TYPE EQ obj_trigger AND
DEP.RDB$DEPENDENT_NAME EQ TRG.RDB$TRIGGER_NAME
{
MetaName trigger_name(TRG.RDB$TRIGGER_NAME);
MetaName trigger_relation_name(TRG.RDB$RELATION_NAME);
dsc desc;
desc.makeText(trigger_name.length(), CS_METADATA, (UCHAR*) trigger_name.c_str());
DeferredWork* dw2 = DFW_post_work(transaction, dfw_modify_trigger, &desc, 0);
DFW_post_work_arg(transaction, dw2, NULL, TRG.RDB$TRIGGER_TYPE, dfw_arg_trg_type);
desc.dsc_length = trigger_relation_name.length();
desc.dsc_address = (UCHAR*) trigger_relation_name.c_str();
DFW_post_work_arg(transaction, dw2, &desc, 0, dfw_arg_check_blr);
}
END_FOR
request2.reset(tdbb, irq_m_fields8, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request2)
RFL IN RDB$RELATION_FIELDS CROSS
DEP IN RDB$DEPENDENCIES CROSS
FUN IN RDB$FUNCTIONS
WITH RFL.RDB$FIELD_SOURCE EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_NAME EQ RFL.RDB$RELATION_NAME AND
DEP.RDB$FIELD_NAME EQ RFL.RDB$FIELD_NAME AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_relation AND
DEP.RDB$DEPENDENT_TYPE EQ obj_udf AND
DEP.RDB$DEPENDENT_NAME EQ FUN.RDB$FUNCTION_NAME AND
FUN.RDB$PACKAGE_NAME MISSING
{
MetaName name(FUN.RDB$FUNCTION_NAME);
dsc desc;
desc.makeText(name.length(), CS_METADATA, (UCHAR*) name.c_str());
DeferredWork* dw2 =
DFW_post_work(transaction, dfw_modify_function, &desc, FUN.RDB$FUNCTION_ID);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
}
END_FOR
request.reset(tdbb, irq_m_fields2, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
DEP IN RDB$DEPENDENCIES CROSS
PRC IN RDB$PROCEDURES
WITH DEP.RDB$DEPENDED_ON_NAME EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_field AND
DEP.RDB$DEPENDENT_TYPE EQ obj_procedure AND
DEP.RDB$DEPENDENT_NAME EQ PRC.RDB$PROCEDURE_NAME AND
PRC.RDB$PACKAGE_NAME MISSING
{
2010-06-12 18:32:46 +02:00
MetaName proc_name(PRC.RDB$PROCEDURE_NAME);
dsc desc;
2009-11-28 20:39:23 +01:00
desc.makeText(proc_name.length(), CS_METADATA, (UCHAR*) proc_name.c_str());
DeferredWork* dw2 =
DFW_post_work(transaction, dfw_modify_procedure, &desc, PRC.RDB$PROCEDURE_ID);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
request.reset(tdbb, irq_m_fields6, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
DEP IN RDB$DEPENDENCIES CROSS
PRC IN RDB$PROCEDURES
WITH DEP.RDB$DEPENDED_ON_NAME EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_field AND
(DEP.RDB$DEPENDENT_TYPE EQ obj_package_header OR
DEP.RDB$DEPENDENT_TYPE EQ obj_package_body) AND
DEP.RDB$DEPENDENT_NAME EQ PRC.RDB$PACKAGE_NAME
{
2010-06-12 18:32:46 +02:00
MetaName proc_name(PRC.RDB$PROCEDURE_NAME);
dsc desc;
2009-11-28 20:39:23 +01:00
desc.makeText(proc_name.length(), CS_METADATA, (UCHAR*) proc_name.c_str());
DeferredWork* dw2 = DFW_post_work(transaction, dfw_modify_procedure, &desc,
PRC.RDB$PROCEDURE_ID, PRC.RDB$PACKAGE_NAME);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
request.reset(tdbb, irq_m_fields3, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
DEP IN RDB$DEPENDENCIES CROSS
TRG IN RDB$TRIGGERS
WITH DEP.RDB$DEPENDED_ON_NAME EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_field AND
DEP.RDB$DEPENDENT_TYPE EQ obj_trigger AND
DEP.RDB$DEPENDENT_NAME EQ TRG.RDB$TRIGGER_NAME
{
2010-06-12 18:32:46 +02:00
MetaName trigger_name(TRG.RDB$TRIGGER_NAME);
MetaName trigger_relation_name(TRG.RDB$RELATION_NAME);
2008-01-16 10:29:37 +01:00
dsc desc;
2009-11-28 20:39:23 +01:00
desc.makeText(trigger_name.length(), CS_METADATA, (UCHAR*) trigger_name.c_str());
DeferredWork* dw2 = DFW_post_work(transaction, dfw_modify_trigger, &desc, 0);
DFW_post_work_arg(transaction, dw2, NULL, TRG.RDB$TRIGGER_TYPE, dfw_arg_trg_type);
desc.dsc_length = trigger_relation_name.length();
desc.dsc_address = (UCHAR*) trigger_relation_name.c_str();
DFW_post_work_arg(transaction, dw2, &desc, 0, dfw_arg_check_blr);
}
END_FOR
request.reset(tdbb, irq_m_fields7, IRQ_REQUESTS);
2009-12-25 14:30:38 +01:00
FOR(REQUEST_HANDLE request)
DEP IN RDB$DEPENDENCIES CROSS
FUN IN RDB$FUNCTIONS
WITH DEP.RDB$DEPENDED_ON_NAME EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_field AND
DEP.RDB$DEPENDENT_TYPE EQ obj_udf AND
DEP.RDB$DEPENDENT_NAME EQ FUN.RDB$FUNCTION_NAME AND
FUN.RDB$PACKAGE_NAME MISSING
{
2010-06-12 18:32:46 +02:00
MetaName name(FUN.RDB$FUNCTION_NAME);
2009-12-25 14:30:38 +01:00
dsc desc;
desc.makeText(name.length(), CS_METADATA, (UCHAR*) name.c_str());
DeferredWork* dw2 =
DFW_post_work(transaction, dfw_modify_function, &desc, FUN.RDB$FUNCTION_ID);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
2009-12-25 14:30:38 +01:00
request.reset(tdbb, irq_m_fields9, IRQ_REQUESTS);
2009-12-25 14:30:38 +01:00
FOR(REQUEST_HANDLE request)
DEP IN RDB$DEPENDENCIES CROSS
FUN IN RDB$FUNCTIONS
WITH DEP.RDB$DEPENDED_ON_NAME EQ field_source->dsc_address AND
DEP.RDB$DEPENDED_ON_TYPE EQ obj_field AND
(DEP.RDB$DEPENDENT_TYPE EQ obj_package_header OR
DEP.RDB$DEPENDENT_TYPE EQ obj_package_body) AND
DEP.RDB$DEPENDENT_NAME EQ FUN.RDB$PACKAGE_NAME
{
2010-06-12 18:32:46 +02:00
MetaName name(FUN.RDB$FUNCTION_NAME);
2009-12-25 14:30:38 +01:00
dsc desc;
desc.makeText(name.length(), CS_METADATA, (UCHAR*) name.c_str());
DeferredWork* dw2 = DFW_post_work(transaction, dfw_modify_function, &desc,
FUN.RDB$FUNCTION_ID, FUN.RDB$PACKAGE_NAME);
DFW_post_work_arg(transaction, dw2, NULL, 0, dfw_arg_check_blr);
}
END_FOR
2009-12-25 14:30:38 +01:00
return dw;
2001-05-23 15:26:42 +02:00
}
2004-03-30 06:10:52 +02:00
Format* MET_current(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ c u r r e n t
*
**************************************
*
* Functional description
* Get the current format for a relation. The current format is the
* format in which new records are to be stored.
*
**************************************/
if (relation->rel_current_format)
return relation->rel_current_format;
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
if (!(relation->rel_flags & REL_scanned))
{
AutoCacheRequest request(tdbb, irq_l_curr_format, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
REL IN RDB$RELATIONS WITH REL.RDB$RELATION_ID EQ relation->rel_id
{
relation->rel_current_fmt = REL.RDB$FORMAT;
}
END_FOR
}
// Usually, format numbers start with one and they are present in RDB$FORMATS.
// However, system tables have zero as their initial format and they don't have
// any related records in RDB$FORMATS, instead their rel_formats[0] is initialized
// directly (see ini.epp). Every other case of zero format number found for an already
// scanned table must be catched here and investigated.
fb_assert(relation->rel_current_fmt || relation->isSystem());
relation->rel_current_format = MET_format(tdbb, relation, relation->rel_current_fmt);
2001-05-23 15:26:42 +02:00
return relation->rel_current_format;
2001-05-23 15:26:42 +02:00
}
void MET_delete_dependencies(thread_db* tdbb,
2010-06-12 18:32:46 +02:00
const MetaName& object_name,
int dependency_type,
jrd_tra* transaction)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ d e l e t e _ d e p e n d e n c i e s
*
**************************************
*
* Functional description
2005-05-28 00:45:31 +02:00
* Delete all dependencies for the specified
2001-05-23 15:26:42 +02:00
* object of given type.
*
**************************************/
SET_TDBB(tdbb);
// if (!object_name)
// return;
//
AutoCacheRequest request(tdbb, irq_d_deps, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request TRANSACTION_HANDLE transaction)
2001-05-23 15:26:42 +02:00
DEP IN RDB$DEPENDENCIES
WITH DEP.RDB$DEPENDENT_NAME = object_name.c_str()
2001-05-23 15:26:42 +02:00
AND DEP.RDB$DEPENDENT_TYPE = dependency_type
{
2001-05-23 15:26:42 +02:00
ERASE DEP;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_delete_shadow(thread_db* tdbb, USHORT shadow_number)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ d e l e t e _ s h a d o w
*
**************************************
*
2005-05-28 00:45:31 +02:00
* Functional description
2001-05-23 15:26:42 +02:00
* When any of the shadows in RDB$FILES for a particular
* shadow are deleted, stop shadowing to that file and
* remove all other files from the same shadow.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
2005-05-28 00:45:31 +02:00
X IN RDB$FILES WITH X.RDB$SHADOW_NUMBER EQ shadow_number
2008-12-18 11:57:12 +01:00
ERASE X;
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
for (Shadow* shadow = dbb->dbb_shadow; shadow; shadow = shadow->sdw_next)
{
if (shadow->sdw_number == shadow_number) {
2001-05-23 15:26:42 +02:00
shadow->sdw_flags |= SDW_shutdown;
}
2003-12-22 11:00:59 +01:00
}
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// notify other processes to check for shadow deletion
if (SDW_lck_update(tdbb, 0))
SDW_notify(tdbb);
2001-05-23 15:26:42 +02:00
}
2010-01-09 23:15:18 +01:00
bool MET_dsql_cache_use(thread_db* tdbb, int type, const MetaName& name, const MetaName& package)
{
2010-01-09 23:15:18 +01:00
DSqlCacheItem* item = get_dsql_cache_item(tdbb, type, QualifiedName(name, package));
2008-03-02 08:35:37 +01:00
const bool obsolete = item->obsolete;
if (!item->locked)
{
// lock to be notified by others when we should mark as obsolete
LCK_lock(tdbb, item->lock, LCK_SR, LCK_WAIT);
item->locked = true;
}
item->obsolete = false;
return obsolete;
}
2010-01-09 23:15:18 +01:00
void MET_dsql_cache_release(thread_db* tdbb, int type, const MetaName& name, const MetaName& package)
{
2010-01-09 23:15:18 +01:00
DSqlCacheItem* item = get_dsql_cache_item(tdbb, type, QualifiedName(name, package));
// release lock
LCK_release(tdbb, item->lock);
// notify others through AST to mark as obsolete
const USHORT key_length = item->lock->lck_length;
AutoPtr<Lock> temp_lock(FB_NEW_RPT(*tdbb->getDefaultPool(), key_length)
Lock(tdbb, key_length, LCK_dsql_cache));
memcpy(temp_lock->lck_key.lck_string, item->lock->lck_key.lck_string, key_length);
if (LCK_lock(tdbb, temp_lock, LCK_EX, LCK_WAIT))
LCK_release(tdbb, temp_lock);
item->locked = false;
item->obsolete = false;
}
2003-12-22 11:00:59 +01:00
void MET_error(const TEXT* string, ...)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ e r r o r
*
**************************************
*
* Functional description
* Post an error in a metadata update
* Oh, wow.
*
**************************************/
TEXT s[128];
va_list ptr;
va_start(ptr, string);
VSNPRINTF(s, sizeof(s), string, ptr);
va_end(ptr);
2001-05-23 15:26:42 +02:00
ERR_post(Arg::Gds(isc_no_meta_update) <<
Arg::Gds(isc_random) << Arg::Str(s));
2001-05-23 15:26:42 +02:00
}
2004-03-30 06:10:52 +02:00
Format* MET_format(thread_db* tdbb, jrd_rel* relation, USHORT number)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ f o r m a t
*
**************************************
*
* Functional description
* Lookup a format for given relation.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
2004-03-30 06:10:52 +02:00
Format* format;
vec<Format*>* formats = relation->rel_formats;
2008-12-18 11:57:12 +01:00
if (formats && (number < formats->count()) && (format = (*formats)[number]))
2001-05-23 15:26:42 +02:00
{
return format;
}
format = NULL;
AutoCacheRequest request(tdbb, irq_r_format, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$FORMATS WITH X.RDB$RELATION_ID EQ relation->rel_id AND
2005-05-28 00:45:31 +02:00
X.RDB$FORMAT EQ number
{
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), &X.RDB$DESCRIPTOR);
2005-05-28 00:45:31 +02:00
// Use generic representation of formats with 32-bit offsets
2005-05-28 00:45:31 +02:00
HalfStaticArray<UCHAR, BUFFER_MEDIUM> buffer;
blob->BLB_get_data(tdbb, buffer.getBuffer(blob->blb_length), blob->blb_length);
unsigned bufferPos = 2;
USHORT count = buffer[0] | (buffer[1] << 8);
2005-05-28 00:45:31 +02:00
2011-05-09 12:15:19 +02:00
format = Format::newFormat(*relation->rel_pool, count);
2010-06-12 18:32:46 +02:00
Array<Ods::Descriptor> odsDescs;
Ods::Descriptor* odsDesc = odsDescs.getBuffer(count);
memcpy(odsDesc, buffer.begin() + bufferPos, count * sizeof(Ods::Descriptor));
for (Format::fmt_desc_iterator desc = format->fmt_desc.begin();
desc < format->fmt_desc.end(); ++desc, ++odsDesc)
{
*desc = *odsDesc;
if (odsDesc->dsc_offset)
format->fmt_length = odsDesc->dsc_offset + desc->dsc_length;
}
2005-05-28 00:45:31 +02:00
const UCHAR* p = buffer.begin() + bufferPos + count * sizeof(Ods::Descriptor);
count = p[0] | (p[1] << 8);
p += 2;
while (count-- > 0)
{
USHORT offset = p[0] | (p[1] << 8);
p += 2;
const Ods::Descriptor* odsDflDesc = (Ods::Descriptor*) p;
p = (UCHAR*) (odsDflDesc + 1);
dsc desc = *odsDflDesc;
desc.dsc_address = const_cast<UCHAR*>(p);
EVL_make_value(tdbb, &desc, &format->fmt_defaults[offset], dbb->dbb_permanent);
p += desc.dsc_length;
2001-05-23 15:26:42 +02:00
}
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-26 01:20:59 +01:00
if (!format)
2011-05-09 12:15:19 +02:00
format = Format::newFormat(*relation->rel_pool);
2001-05-23 15:26:42 +02:00
format->fmt_version = number;
2009-11-21 04:23:47 +01:00
// Link the format block into the world
2001-05-23 15:26:42 +02:00
formats = relation->rel_formats =
2011-05-09 12:15:19 +02:00
vec<Format*>::newVector(*relation->rel_pool, relation->rel_formats, number + 1);
(*formats)[number] = format;
2001-05-23 15:26:42 +02:00
return format;
}
bool MET_get_char_coll_subtype(thread_db* tdbb, USHORT* id, const UCHAR* name, USHORT length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
2005-05-28 00:45:31 +02:00
* M E T _ g e t _ c h a r _ c o l l _ s u b t y p e
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Character types can be specified as either:
* a) A POSIX style locale name "<collation>.<characterset>"
* or
* b) A simple <characterset> name (using default collation)
* c) A simple <collation> name (use charset for collation)
*
* Given an ASCII7 string which could be any of the above, try to
* resolve the name in the order a, b, c
* a) is only tried iff the name contains a period.
* (in which case b) and c) are not tried).
*
* Return:
* 1 if no errors (and *id is set).
* 0 if the name could not be resolved.
*
**************************************/
SET_TDBB(tdbb);
2003-11-04 00:59:24 +01:00
fb_assert(id != NULL);
fb_assert(name != NULL);
2001-05-23 15:26:42 +02:00
2003-12-22 11:00:59 +01:00
const UCHAR* const end_name = name + length;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Force key to uppercase, following C locale rules for uppercasing
// At the same time, search for the first period in the string (if any)
UCHAR buffer[MAX_SQL_IDENTIFIER_SIZE]; // BASED ON RDB$COLLATION_NAME
2003-12-22 11:00:59 +01:00
UCHAR* p = buffer;
UCHAR* period = NULL;
2009-11-07 12:58:54 +01:00
for (; name < end_name && p < buffer + sizeof(buffer) - 1; p++, name++)
{
2001-05-23 15:26:42 +02:00
*p = UPPER7(*name);
if ((*p == '.') && !period) {
2001-05-23 15:26:42 +02:00
period = p;
}
2001-05-23 15:26:42 +02:00
}
*p = 0;
2009-11-21 04:23:47 +01:00
// Is there a period, separating collation name from character set?
2009-11-07 12:58:54 +01:00
if (period)
{
2001-05-23 15:26:42 +02:00
*period = 0;
return resolve_charset_and_collation(tdbb, id, period + 1, buffer);
}
2009-11-21 04:23:47 +01:00
// Is it a character set name (implying charset default collation sequence)
2008-01-16 10:29:37 +01:00
bool res = resolve_charset_and_collation(tdbb, id, buffer, NULL);
2009-11-07 12:58:54 +01:00
if (!res)
{
2009-11-21 04:23:47 +01:00
// Is it a collation name (implying implementation-default character set)
2008-01-16 10:29:37 +01:00
res = resolve_charset_and_collation(tdbb, id, NULL, buffer);
2001-05-23 15:26:42 +02:00
}
2008-01-16 10:29:37 +01:00
return res;
2001-05-23 15:26:42 +02:00
}
2006-04-05 18:34:18 +02:00
bool MET_get_char_coll_subtype_info(thread_db* tdbb, USHORT id, SubtypeInfo* info)
2005-05-28 00:45:31 +02:00
{
/**************************************
*
* M E T _ g e t _ c h a r _ c o l l _ s u b t y p e _ i n f o
*
**************************************
*
* Functional description
* Get charset and collation informations
* for a subtype ID.
*
**************************************/
fb_assert(info != NULL);
const USHORT charset_id = id & 0x00FF;
const USHORT collation_id = id >> 8;
2005-05-28 00:45:31 +02:00
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2005-05-28 00:45:31 +02:00
AutoCacheRequest request(tdbb, irq_l_subtype, IRQ_REQUESTS);
2005-05-28 00:45:31 +02:00
bool found = false;
FOR(REQUEST_HANDLE request) FIRST 1
CL IN RDB$COLLATIONS CROSS
CS IN RDB$CHARACTER_SETS
WITH CL.RDB$CHARACTER_SET_ID EQ charset_id AND
CL.RDB$COLLATION_ID EQ collation_id AND
CS.RDB$CHARACTER_SET_ID EQ CL.RDB$CHARACTER_SET_ID
{
found = true;
2005-05-28 00:45:31 +02:00
info->charsetName = CS.RDB$CHARACTER_SET_NAME;
info->collationName = CL.RDB$COLLATION_NAME;
2005-05-28 00:45:31 +02:00
if (CL.RDB$BASE_COLLATION_NAME.NULL)
info->baseCollationName = info->collationName;
else
info->baseCollationName = CL.RDB$BASE_COLLATION_NAME;
2005-05-28 00:45:31 +02:00
if (CL.RDB$SPECIFIC_ATTRIBUTES.NULL)
info->specificAttributes.clear();
else
{
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), &CL.RDB$SPECIFIC_ATTRIBUTES);
const ULONG length = blob->blb_length;
2005-05-28 00:45:31 +02:00
// ASF: Here info->specificAttributes is in UNICODE_FSS charset.
// It will be converted to the collation charset in intl.cpp
blob->BLB_get_data(tdbb, info->specificAttributes.getBuffer(length), length);
}
2005-05-28 00:45:31 +02:00
info->attributes = (USHORT)CL.RDB$COLLATION_ATTRIBUTES;
info->ignoreAttributes = CL.RDB$COLLATION_ATTRIBUTES.NULL;
}
END_FOR
2005-05-28 00:45:31 +02:00
return found;
}
DmlNode* MET_get_dependencies(thread_db* tdbb,
jrd_rel* relation,
const UCHAR* blob,
const ULONG blob_length,
CompilerScratch* view_csb,
bid* blob_id,
JrdStatement** statementPtr,
CompilerScratch** csb_ptr,
const MetaName& object_name,
int type,
USHORT flags,
jrd_tra* transaction,
const MetaName& domain_validation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ g e t _ d e p e n d e n c i e s
*
**************************************
*
* Functional description
2005-05-28 00:45:31 +02:00
* Get dependencies for an object by parsing
2001-05-23 15:26:42 +02:00
* the blr used in its definition.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
fb_assert(domain_validation.isEmpty() || object_name == domain_validation); // for now
CompilerScratch* csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), 5, domain_validation);
csb->csb_g_flags |= (csb_get_dependencies | flags);
2001-05-23 15:26:42 +02:00
DmlNode* node;
2001-05-23 15:26:42 +02:00
if (blob)
{
node = PAR_blr(tdbb, relation, blob, blob_length, view_csb, &csb, statementPtr,
(type == obj_trigger && relation != NULL), 0);
2001-05-23 15:26:42 +02:00
}
else
{
node = MET_parse_blob(tdbb, relation, blob_id, &csb, statementPtr,
(type == obj_trigger && relation != NULL), type == obj_validation);
2001-05-23 15:26:42 +02:00
}
2006-08-30 06:54:44 +02:00
if (type == obj_computed)
2001-05-23 15:26:42 +02:00
{
MetaName domainName;
AutoRequest handle;
2010-04-23 03:59:21 +02:00
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
RLF IN RDB$RELATION_FIELDS CROSS
FLD IN RDB$FIELDS WITH
RLF.RDB$FIELD_SOURCE EQ FLD.RDB$FIELD_NAME AND
RLF.RDB$RELATION_NAME EQ relation->rel_name.c_str() AND
RLF.RDB$FIELD_NAME EQ object_name.c_str()
2010-04-23 03:59:21 +02:00
{
domainName = FLD.RDB$FIELD_NAME;
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
MET_delete_dependencies(tdbb, domainName, type, transaction);
store_dependencies(tdbb, csb, relation, domainName, type, transaction);
}
else
{
MET_delete_dependencies(tdbb, object_name, type, transaction);
store_dependencies(tdbb, csb, relation, object_name, type, transaction);
}
2001-05-23 15:26:42 +02:00
if (csb_ptr)
*csb_ptr = csb;
2001-05-23 15:26:42 +02:00
else
delete csb;
2001-05-23 15:26:42 +02:00
return node;
}
jrd_fld* MET_get_field(jrd_rel* relation, USHORT id)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ g e t _ f i e l d
*
**************************************
*
* Functional description
* Get the field block for a field if possible. If not,
* return NULL;
*
**************************************/
vec<jrd_fld*>* vector;
2001-05-23 15:26:42 +02:00
2008-12-18 11:57:12 +01:00
if (!relation || !(vector = relation->rel_fields) || id >= vector->count())
2003-12-22 11:00:59 +01:00
{
return NULL;
}
2001-05-23 15:26:42 +02:00
return (*vector)[id];
2001-05-23 15:26:42 +02:00
}
void MET_get_shadow_files(thread_db* tdbb, bool delete_files)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ g e t _ s h a d o w _ f i l e s
*
**************************************
*
* Functional description
* Check the shadows found in the database against
2005-05-28 00:45:31 +02:00
* our in-memory list: if any new shadow files have
2001-05-23 15:26:42 +02:00
* been defined since the last time we looked, start
* shadowing to them; if any have been deleted, stop
* shadowing to them.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2010-04-23 03:59:21 +02:00
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle) X IN RDB$FILES
WITH X.RDB$SHADOW_NUMBER NOT MISSING
AND X.RDB$SHADOW_NUMBER NE 0
AND X.RDB$FILE_SEQUENCE EQ 0
2010-04-23 03:59:21 +02:00
{
2008-12-18 11:57:12 +01:00
if ((X.RDB$FILE_FLAGS & FILE_shadow) && !(X.RDB$FILE_FLAGS & FILE_inactive))
2001-05-23 15:26:42 +02:00
{
2003-12-22 11:00:59 +01:00
const USHORT file_flags = X.RDB$FILE_FLAGS;
SDW_start(tdbb, X.RDB$FILE_NAME, X.RDB$SHADOW_NUMBER, file_flags, delete_files);
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// if the shadow exists, mark the appropriate shadow
// block as found for the purposes of this routine;
// if the shadow was conditional and is no longer, note it
2001-05-23 15:26:42 +02:00
for (Shadow* shadow = dbb->dbb_shadow; shadow; shadow = shadow->sdw_next)
2001-05-23 15:26:42 +02:00
{
2008-12-18 11:57:12 +01:00
if ((shadow->sdw_number == X.RDB$SHADOW_NUMBER) && !(shadow->sdw_flags & SDW_IGNORE))
2001-05-23 15:26:42 +02:00
{
shadow->sdw_flags |= SDW_found;
if (!(file_flags & FILE_conditional)) {
2001-05-23 15:26:42 +02:00
shadow->sdw_flags &= ~SDW_conditional;
}
2001-05-23 15:26:42 +02:00
break;
}
}
}
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// if any current shadows were not defined in database, mark
// them to be shutdown since they don't exist anymore
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
for (Shadow* shadow = dbb->dbb_shadow; shadow; shadow = shadow->sdw_next)
{
if (!(shadow->sdw_flags & SDW_found))
2001-05-23 15:26:42 +02:00
shadow->sdw_flags |= SDW_shutdown;
else
2001-05-23 15:26:42 +02:00
shadow->sdw_flags &= ~SDW_found;
2003-12-22 11:00:59 +01:00
}
2001-05-23 15:26:42 +02:00
SDW_check(tdbb);
2001-05-23 15:26:42 +02:00
}
void MET_load_db_triggers(thread_db* tdbb, int type)
{
/**************************************
*
2006-11-06 02:07:31 +01:00
* M E T _ l o a d _ d b _ t r i g g e r s
*
**************************************
*
* Functional description
* Load database triggers from RDB$TRIGGERS.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
CHECK_DBB(dbb);
2011-05-09 12:15:19 +02:00
if ((attachment->att_flags & ATT_no_db_triggers) ||
attachment->att_triggers[type] != NULL)
{
return;
}
2011-05-09 12:15:19 +02:00
attachment->att_triggers[type] = FB_NEW(*attachment->att_pool)
trig_vec(*attachment->att_pool);
AutoRequest trigger_request;
int encoded_type = type | TRIGGER_TYPE_DB;
FOR(REQUEST_HANDLE trigger_request)
TRG IN RDB$TRIGGERS
WITH TRG.RDB$RELATION_NAME MISSING AND
TRG.RDB$TRIGGER_TYPE EQ encoded_type AND
TRG.RDB$TRIGGER_INACTIVE EQ 0
SORTED BY TRG.RDB$TRIGGER_SEQUENCE
2010-04-23 03:59:21 +02:00
{
2011-05-09 12:15:19 +02:00
MET_load_trigger(tdbb, NULL, TRG.RDB$TRIGGER_NAME, &attachment->att_triggers[type]);
2010-04-23 03:59:21 +02:00
}
END_FOR
}
// Load DDL triggers from RDB$TRIGGERS.
void MET_load_ddl_triggers(thread_db* tdbb)
{
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
CHECK_DBB(dbb);
2011-05-09 12:15:19 +02:00
if ((attachment->att_flags & ATT_no_db_triggers) ||
attachment->att_ddl_triggers != NULL)
{
return;
}
2011-05-09 12:15:19 +02:00
attachment->att_ddl_triggers = FB_NEW(*tdbb->getDatabase()->dbb_permanent)
trig_vec(*tdbb->getDatabase()->dbb_permanent);
AutoRequest trigger_request;
FOR(REQUEST_HANDLE trigger_request)
TRG IN RDB$TRIGGERS
WITH TRG.RDB$RELATION_NAME MISSING AND
TRG.RDB$TRIGGER_INACTIVE EQ 0
SORTED BY TRG.RDB$TRIGGER_SEQUENCE
{
if ((TRG.RDB$TRIGGER_TYPE & TRIGGER_TYPE_MASK) == TRIGGER_TYPE_DDL)
{
MET_load_trigger(tdbb, NULL, TRG.RDB$TRIGGER_NAME,
2011-05-09 12:15:19 +02:00
&attachment->att_ddl_triggers);
}
}
END_FOR
}
void MET_load_trigger(thread_db* tdbb,
2005-05-28 00:45:31 +02:00
jrd_rel* relation,
2010-06-12 18:32:46 +02:00
const MetaName& trigger_name,
trig_vec** triggers)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o a d _ t r i g g e r
*
**************************************
*
* Functional description
* Load triggers from RDB$TRIGGERS. If a requested,
* also load triggers from RDB$RELATIONS.
*
**************************************/
TEXT errmsg[MAX_ERRMSG_LEN + 1];
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
if (relation)
{
if (relation->rel_flags & REL_sys_trigs_being_loaded)
return;
// No need to load table triggers for ReadOnly databases,
// since INSERT/DELETE/UPDATE statements are not going to be allowed
// hvlad: GTT with ON COMMIT DELETE ROWS clause is writable
if (dbb->readOnly() && !(relation->rel_flags & REL_temp_tran))
return;
}
2001-05-23 15:26:42 +02:00
// Scan RDB$TRIGGERS next
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_s_triggers, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
TRG IN RDB$TRIGGERS
WITH TRG.RDB$TRIGGER_NAME EQ trigger_name.c_str() AND
(TRG.RDB$TRIGGER_INACTIVE MISSING OR TRG.RDB$TRIGGER_INACTIVE EQ 0)
{
2009-11-21 04:23:47 +01:00
// check if the trigger is to be fired without any permissions
// checks. Verify such a claim
USHORT trig_flags = (USHORT) TRG.RDB$FLAGS;
2009-11-21 04:23:47 +01:00
// if there is an ignore permission flag, see if it is legit
2008-12-18 11:57:12 +01:00
if ((TRG.RDB$FLAGS & TRG_ignore_perm) && !verify_TRG_ignore_perm(tdbb, trigger_name))
{
fb_msg_format(NULL, JRD_BUGCHK, 304, sizeof(errmsg),
errmsg, MsgFormat::SafeArg() << trigger_name.c_str());
ERR_log(JRD_BUGCHK, 304, errmsg);
2001-05-23 15:26:42 +02:00
trig_flags &= ~TRG_ignore_perm;
}
2001-05-23 15:26:42 +02:00
bid debug_blob_id;
debug_blob_id.clear();
bid extBodyId;
extBodyId.clear();
if (!TRG.RDB$DEBUG_INFO.NULL) // ODS_11_1
debug_blob_id = TRG.RDB$DEBUG_INFO;
2010-06-12 18:32:46 +02:00
MetaName engine;
string entryPoint;
if (!TRG.RDB$ENGINE_NAME.NULL) // ODS_12_0
{
engine = TRG.RDB$ENGINE_NAME;
extBodyId = TRG.RDB$TRIGGER_SOURCE;
}
if (!TRG.RDB$ENTRYPOINT.NULL) // ODS_12_0
entryPoint = TRG.RDB$ENTRYPOINT;
if (TRG.RDB$RELATION_NAME.NULL)
{
if ((TRG.RDB$TRIGGER_TYPE & TRIGGER_TYPE_MASK) == TRIGGER_TYPE_DB ||
(TRG.RDB$TRIGGER_TYPE & TRIGGER_TYPE_MASK) == TRIGGER_TYPE_DDL)
{
// this is a database trigger
get_trigger(tdbb,
relation,
&TRG.RDB$TRIGGER_BLR,
&debug_blob_id,
triggers,
TRG.RDB$TRIGGER_NAME,
TRG.RDB$TRIGGER_TYPE & ~TRIGGER_TYPE_MASK,
(bool) TRG.RDB$SYSTEM_FLAG,
trig_flags,
engine,
entryPoint,
&extBodyId);
2001-05-23 15:26:42 +02:00
}
}
else
{
// dimitr: support for the universal triggers
int trigger_action, slot_index = 0;
while ((trigger_action = TRIGGER_ACTION_SLOT(TRG.RDB$TRIGGER_TYPE, ++slot_index)) > 0)
{
get_trigger(tdbb,
relation,
&TRG.RDB$TRIGGER_BLR,
&debug_blob_id,
triggers + trigger_action,
TRG.RDB$TRIGGER_NAME,
(UCHAR) trigger_action,
2003-12-03 09:19:24 +01:00
(bool) TRG.RDB$SYSTEM_FLAG,
trig_flags,
engine,
entryPoint,
&extBodyId);
}
2001-05-23 15:26:42 +02:00
}
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_lookup_cnstrt_for_index(thread_db* tdbb,
2010-06-12 18:32:46 +02:00
MetaName& constraint_name,
const MetaName& index_name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ c n s t r t _ f o r _ i n d e x
*
**************************************
*
* Functional description
* Lookup constraint name from index name, if one exists.
* Calling routine must pass a buffer of at least 32 bytes.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
constraint_name = "";
AutoCacheRequest request(tdbb, irq_l_cnstrt, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$RELATION_CONSTRAINTS WITH X.RDB$INDEX_NAME EQ index_name.c_str()
{
constraint_name = X.RDB$CONSTRAINT_NAME;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_lookup_cnstrt_for_trigger(thread_db* tdbb,
2010-06-12 18:32:46 +02:00
MetaName& constraint_name,
MetaName& relation_name,
const MetaName& trigger_name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
2005-05-28 00:45:31 +02:00
* M E T _ l o o k u p _ c n s t r t _ f o r _ t r i g g e r
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Lookup constraint name from trigger name, if one exists.
* Calling routine must pass a buffer of at least 32 bytes.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
constraint_name = "";
relation_name = "";
AutoCacheRequest request(tdbb, irq_l_check, IRQ_REQUESTS);
AutoCacheRequest request2(tdbb, irq_l_check2, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// utilize two requests rather than one so that we
// guarantee we always return the name of the relation
// that the trigger is defined on, even if we don't
// have a check constraint defined for that trigger
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
Y IN RDB$TRIGGERS WITH
Y.RDB$TRIGGER_NAME EQ trigger_name.c_str()
{
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request2)
X IN RDB$CHECK_CONSTRAINTS WITH
X.RDB$TRIGGER_NAME EQ Y.RDB$TRIGGER_NAME
{
constraint_name = X.RDB$CONSTRAINT_NAME;
}
END_FOR
2001-05-23 15:26:42 +02:00
relation_name = Y.RDB$RELATION_NAME;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_lookup_exception(thread_db* tdbb,
SLONG number,
2010-06-12 18:32:46 +02:00
MetaName& name,
string* message)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ e x c e p t i o n
*
**************************************
*
* Functional description
* Lookup exception by number and return its name and message.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// We need to look up exception in RDB$EXCEPTIONS
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_exception, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
name = "";
if (message)
*message = "";
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$EXCEPTIONS WITH X.RDB$EXCEPTION_NUMBER = number
{
2001-05-23 15:26:42 +02:00
if (!X.RDB$EXCEPTION_NAME.NULL)
name = X.RDB$EXCEPTION_NAME;
2001-05-23 15:26:42 +02:00
if (!X.RDB$MESSAGE.NULL && message)
*message = X.RDB$MESSAGE;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
bool MET_load_exception(thread_db* tdbb, ExceptionItem& item)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o a d _ e x c e p t i o n
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Lookup exception by name and fill the passed instance.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// We need to look up exception in RDB$EXCEPTIONS
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_except_no, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$EXCEPTIONS WITH X.RDB$EXCEPTION_NAME = item.name.c_str()
{
item.type = ExceptionItem::XCP_CODE;
item.code = X.RDB$EXCEPTION_NUMBER;
item.secName = X.RDB$SECURITY_CLASS;
return true;
}
END_FOR
2001-05-23 15:26:42 +02:00
return false;
2001-05-23 15:26:42 +02:00
}
2010-10-15 03:55:57 +02:00
int MET_lookup_field(thread_db* tdbb, jrd_rel* relation, const MetaName& name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ f i e l d
*
**************************************
*
* Functional description
* Look up a field name.
*
2003-02-20 01:10:25 +01:00
* if the field is not found return -1
*
*****************************************/
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Start by checking field names that we already know
vec<jrd_fld*>* vector = relation->rel_fields;
if (vector)
{
int id = 0;
vec<jrd_fld*>::iterator fieldIter = vector->begin();
2008-12-18 11:57:12 +01:00
for (const vec<jrd_fld*>::const_iterator end = vector->end(); fieldIter < end;
++fieldIter, ++id)
2003-12-22 11:00:59 +01:00
{
2005-05-28 00:45:31 +02:00
if (*fieldIter)
{
jrd_fld* field = *fieldIter;
if (field->fld_name == name)
{
return id;
}
}
}
}
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Not found. Next, try system relations directly
2001-05-23 15:26:42 +02:00
int id = -1;
2001-05-23 15:26:42 +02:00
if (relation->rel_flags & REL_deleted)
{
return id;
}
AutoCacheRequest request(tdbb, irq_l_field, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$RELATION_FIELDS WITH
X.RDB$RELATION_NAME EQ relation->rel_name.c_str() AND
2005-09-14 14:02:13 +02:00
X.RDB$FIELD_ID NOT MISSING AND
2005-05-28 00:45:31 +02:00
X.RDB$FIELD_NAME EQ name.c_str()
{
id = X.RDB$FIELD_ID;
2001-05-23 15:26:42 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return id;
}
BlobFilter* MET_lookup_filter(thread_db* tdbb, SSHORT from, SSHORT to)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ f i l t e r
*
**************************************
*
* Functional description
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
2003-12-22 11:00:59 +01:00
FPTR_BFILTER_CALLBACK filter = NULL;
BlobFilter* blf = NULL;
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_r_filters, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$FILTERS WITH X.RDB$INPUT_SUB_TYPE EQ from AND
2008-12-18 11:57:12 +01:00
X.RDB$OUTPUT_SUB_TYPE EQ to
{
filter = (FPTR_BFILTER_CALLBACK)
2008-01-29 11:11:52 +01:00
Module::lookup(X.RDB$MODULE_NAME, X.RDB$ENTRYPOINT, dbb->dbb_modules);
2001-05-23 15:26:42 +02:00
if (filter)
{
blf = FB_NEW(*dbb->dbb_permanent) BlobFilter(*dbb->dbb_permanent);
blf->blf_next = NULL;
blf->blf_from = from;
blf->blf_to = to;
blf->blf_filter = filter;
blf->blf_exception_message.printf(EXCEPTION_MESSAGE,
2001-05-23 15:26:42 +02:00
X.RDB$FUNCTION_NAME, X.RDB$ENTRYPOINT, X.RDB$MODULE_NAME);
}
}
END_FOR
2001-05-23 15:26:42 +02:00
return blf;
2001-05-23 15:26:42 +02:00
}
bool MET_load_generator(thread_db* tdbb, GeneratorItem& item)
{
/**************************************
*
* M E T _ l o a d _ g e n e r a t o r
*
**************************************
*
* Functional description
* Lookup generator ID by its name and load its metadata into the passed object.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
if (item.name == "RDB$GENERATORS")
{
item.id = 0;
return true;
}
AutoCacheRequest request(tdbb, irq_r_gen_id, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
X IN RDB$GENERATORS WITH X.RDB$GENERATOR_NAME EQ item.name.c_str()
{
item.id = X.RDB$GENERATOR_ID;
item.secName = X.RDB$SECURITY_CLASS;
return true;
}
END_FOR
return false;
}
SLONG MET_lookup_generator(thread_db* tdbb, const MetaName& name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ g e n e r a t o r
*
**************************************
*
* Functional description
* Lookup generator ID by its name.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
if (name == "RDB$GENERATORS")
2001-05-23 15:26:42 +02:00
return 0;
AutoCacheRequest request(tdbb, irq_l_gen_id, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$GENERATORS WITH X.RDB$GENERATOR_NAME EQ name.c_str()
{
return X.RDB$GENERATOR_ID;
}
END_FOR
2001-05-23 15:26:42 +02:00
return -1;
2001-05-23 15:26:42 +02:00
}
void MET_lookup_generator_id(thread_db* tdbb, SLONG gen_id, MetaName& name)
2002-07-01 18:59:09 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ g e n e r a t o r _ i d
*
**************************************
*
* Functional description
* Lookup generator (aka gen_id) by ID. It will load
* the name in the third parameter.
*
**************************************/
2008-12-18 11:57:12 +01:00
SET_TDBB (tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2005-05-28 00:45:31 +02:00
2009-11-07 12:58:54 +01:00
if (!gen_id)
{
2008-12-18 11:57:12 +01:00
name = "RDB$GENERATORS";
return;
}
2002-07-01 18:59:09 +02:00
2008-12-18 11:57:12 +01:00
name = "";
2005-05-28 00:45:31 +02:00
AutoCacheRequest request (tdbb, irq_r_gen_id_num, IRQ_REQUESTS);
2005-05-28 00:45:31 +02:00
2008-12-18 11:57:12 +01:00
FOR (REQUEST_HANDLE request)
X IN RDB$GENERATORS WITH X.RDB$GENERATOR_ID EQ gen_id
{
2008-12-18 11:57:12 +01:00
name = X.RDB$GENERATOR_NAME;
}
END_FOR
2002-07-01 18:59:09 +02:00
}
2001-05-23 15:26:42 +02:00
void MET_lookup_index(thread_db* tdbb,
2010-06-12 18:32:46 +02:00
MetaName& index_name,
const MetaName& relation_name,
USHORT number)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ i n d e x
*
**************************************
*
* Functional description
* Lookup index name from relation and index number.
* Calling routine must pass a buffer of at least 32 bytes.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
index_name = "";
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_index, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$INDICES WITH X.RDB$RELATION_NAME EQ relation_name.c_str()
2005-05-28 00:45:31 +02:00
AND X.RDB$INDEX_ID EQ number
{
index_name = X.RDB$INDEX_NAME;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
SLONG MET_lookup_index_name(thread_db* tdbb,
2010-06-12 18:32:46 +02:00
const MetaName& index_name,
2003-12-22 11:00:59 +01:00
SLONG* relation_id, SSHORT* status)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ i n d e x _ n a m e
*
**************************************
*
* Functional description
* Lookup index id from index name.
*
**************************************/
SLONG id = -1;
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_index_name, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
*status = MET_object_unknown;
FOR(REQUEST_HANDLE request)
X IN RDB$INDICES WITH
X.RDB$INDEX_NAME EQ index_name.c_str()
{
if (X.RDB$INDEX_INACTIVE == 0)
2001-05-23 15:26:42 +02:00
*status = MET_object_active;
else
2001-05-23 15:26:42 +02:00
*status = MET_object_inactive;
id = X.RDB$INDEX_ID - 1;
2003-12-22 11:00:59 +01:00
const jrd_rel* relation = MET_lookup_relation(tdbb, X.RDB$RELATION_NAME);
2001-05-23 15:26:42 +02:00
*relation_id = relation->rel_id;
}
END_FOR
2001-05-23 15:26:42 +02:00
return id;
}
2009-11-21 06:20:18 +01:00
bool MET_lookup_partner(thread_db* tdbb, jrd_rel* relation, index_desc* idx, const TEXT* index_name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ p a r t n e r
*
**************************************
*
* Functional description
* Find partner index participating in a
* foreign key relationship.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
if (relation->rel_flags & REL_check_partners) {
scan_partners(tdbb, relation);
2001-05-23 15:26:42 +02:00
}
if (idx->idx_flags & idx_foreign)
{
if (index_name)
2001-05-23 15:26:42 +02:00
{
2009-11-21 04:23:47 +01:00
// Since primary key index names aren't being cached, do a long
// hard lookup. This is only called during index create for foreign keys.
2001-05-23 15:26:42 +02:00
bool found = false;
AutoRequest request;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
IDX IN RDB$INDICES CROSS
IND IN RDB$INDICES WITH
IDX.RDB$RELATION_NAME EQ relation->rel_name.c_str() AND
2001-05-23 15:26:42 +02:00
(IDX.RDB$INDEX_ID EQ idx->idx_id + 1 OR
2002-12-22 14:08:50 +01:00
IDX.RDB$INDEX_NAME EQ index_name) AND
2001-05-23 15:26:42 +02:00
IND.RDB$INDEX_NAME EQ IDX.RDB$FOREIGN_KEY AND
2005-12-12 14:58:19 +01:00
IND.RDB$UNIQUE_FLAG = 1
2010-04-23 03:59:21 +02:00
{
2008-12-18 11:57:12 +01:00
const jrd_rel* partner_relation = MET_lookup_relation(tdbb, IND.RDB$RELATION_NAME);
2001-05-23 15:26:42 +02:00
2008-12-18 11:57:12 +01:00
if (partner_relation && !IDX.RDB$INDEX_INACTIVE && !IND.RDB$INDEX_INACTIVE)
2001-05-23 15:26:42 +02:00
{
idx->idx_primary_relation = partner_relation->rel_id;
idx->idx_primary_index = IND.RDB$INDEX_ID - 1;
fb_assert(idx->idx_primary_index != idx_invalid);
found = true;
2001-05-23 15:26:42 +02:00
}
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return found;
}
frgn* references = &relation->rel_foreign_refs;
2001-05-23 15:26:42 +02:00
if (references->frgn_reference_ids)
{
for (unsigned int index_number = 0;
index_number < references->frgn_reference_ids->count();
2008-12-18 11:57:12 +01:00
index_number++)
2001-05-23 15:26:42 +02:00
{
if (idx->idx_id == (*references->frgn_reference_ids)[index_number])
2001-05-23 15:26:42 +02:00
{
idx->idx_primary_relation = (*references->frgn_relations)[index_number];
idx->idx_primary_index = (*references->frgn_indexes)[index_number];
return true;
2001-05-23 15:26:42 +02:00
}
}
}
return false;
2001-05-23 15:26:42 +02:00
}
else if (idx->idx_flags & (idx_primary | idx_unique))
{
const prim* dependencies = &relation->rel_primary_dpnds;
2001-05-23 15:26:42 +02:00
if (dependencies->prim_reference_ids)
{
for (unsigned int index_number = 0;
index_number < dependencies->prim_reference_ids->count();
2001-05-23 15:26:42 +02:00
index_number++)
{
if (idx->idx_id == (*dependencies->prim_reference_ids)[index_number])
2001-05-23 15:26:42 +02:00
{
2008-12-18 11:57:12 +01:00
idx->idx_foreign_primaries = relation->rel_primary_dpnds.prim_reference_ids;
idx->idx_foreign_relations = relation->rel_primary_dpnds.prim_relations;
idx->idx_foreign_indexes = relation->rel_primary_dpnds.prim_indexes;
return true;
2001-05-23 15:26:42 +02:00
}
}
}
return false;
2001-05-23 15:26:42 +02:00
}
return false;
2001-05-23 15:26:42 +02:00
}
jrd_prc* MET_lookup_procedure(thread_db* tdbb, const QualifiedName& name, bool noscan)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ p r o c e d u r e
*
**************************************
*
* Functional description
* Lookup procedure by name. Name passed in is
* ASCIZ name.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2003-12-22 11:00:59 +01:00
jrd_prc* check_procedure = NULL;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// See if we already know the procedure by name
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* procedures = attachment->att_procedures;
2009-11-07 12:58:54 +01:00
if (procedures)
{
2008-12-18 11:57:12 +01:00
vec<jrd_prc*>::iterator ptr = procedures->begin();
for (const vec<jrd_prc*>::const_iterator end = procedures->end(); ptr < end; ++ptr)
2002-11-16 21:20:44 +01:00
{
jrd_prc* procedure = *ptr;
2008-12-18 11:57:12 +01:00
if (procedure && !(procedure->prc_flags & PRC_obsolete) &&
((procedure->prc_flags & PRC_scanned) || noscan) &&
!(procedure->prc_flags & PRC_being_scanned) &&
!(procedure->prc_flags & PRC_being_altered))
{
if (procedure->getName() == name)
2002-11-16 21:20:44 +01:00
{
2005-05-28 00:45:31 +02:00
if (procedure->prc_flags & PRC_check_existence)
{
check_procedure = procedure;
2008-12-18 11:57:12 +01:00
LCK_lock(tdbb, check_procedure->prc_existence_lock, LCK_SR, LCK_WAIT);
break;
}
2008-01-16 10:29:37 +01:00
return procedure;
}
}
2001-05-23 15:26:42 +02:00
}
}
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// We need to look up the procedure name in RDB$PROCEDURES
2001-05-23 15:26:42 +02:00
jrd_prc* procedure = NULL;
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_procedure, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
P IN RDB$PROCEDURES
WITH P.RDB$PROCEDURE_NAME EQ name.identifier.c_str() AND
P.RDB$PACKAGE_NAME EQUIV NULLIF(name.package.c_str(), '')
{
2008-12-18 11:57:12 +01:00
procedure = MET_procedure(tdbb, P.RDB$PROCEDURE_ID, noscan, 0);
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (check_procedure)
{
check_procedure->prc_flags &= ~PRC_check_existence;
2009-11-07 12:58:54 +01:00
if (check_procedure != procedure)
{
LCK_release(tdbb, check_procedure->prc_existence_lock);
check_procedure->prc_flags |= PRC_obsolete;
}
}
2001-05-23 15:26:42 +02:00
return procedure;
}
jrd_prc* MET_lookup_procedure_id(thread_db* tdbb, SSHORT id,
2009-11-21 06:20:18 +01:00
bool return_deleted, bool noscan, USHORT flags)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ p r o c e d u r e _ i d
*
**************************************
*
* Functional description
* Lookup procedure by id.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2003-12-22 11:00:59 +01:00
jrd_prc* check_procedure = NULL;
2001-05-23 15:26:42 +02:00
jrd_prc* procedure;
2005-05-28 00:45:31 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* procedures = attachment->att_procedures;
2008-12-18 11:57:12 +01:00
if (procedures && id < (SSHORT) procedures->count() && (procedure = (*procedures)[id]) &&
procedure->getId() == id &&
2008-12-18 11:57:12 +01:00
!(procedure->prc_flags & PRC_being_scanned) &&
((procedure->prc_flags & PRC_scanned) || noscan) &&
!(procedure->prc_flags & PRC_being_altered) &&
(!(procedure->prc_flags & PRC_obsolete) || return_deleted))
{
2009-11-07 12:58:54 +01:00
if (procedure->prc_flags & PRC_check_existence)
{
check_procedure = procedure;
LCK_lock(tdbb, check_procedure->prc_existence_lock, LCK_SR, LCK_WAIT);
}
else {
return procedure;
}
}
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// We need to look up the procedure name in RDB$PROCEDURES
2001-05-23 15:26:42 +02:00
procedure = NULL;
AutoCacheRequest request(tdbb, irq_l_proc_id, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
P IN RDB$PROCEDURES WITH P.RDB$PROCEDURE_ID EQ id
{
procedure = MET_procedure(tdbb, P.RDB$PROCEDURE_ID, noscan, flags);
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (check_procedure)
{
check_procedure->prc_flags &= ~PRC_check_existence;
2009-11-07 12:58:54 +01:00
if (check_procedure != procedure)
{
LCK_release(tdbb, check_procedure->prc_existence_lock);
check_procedure->prc_flags |= PRC_obsolete;
}
}
2001-05-23 15:26:42 +02:00
return procedure;
}
2010-06-12 18:32:46 +02:00
jrd_rel* MET_lookup_relation(thread_db* tdbb, const MetaName& name)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ r e l a t i o n
*
**************************************
*
* Functional description
* Lookup relation by name. Name passed in is
* ASCIZ name.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// See if we already know the relation by name
2001-05-23 15:26:42 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_rel*>* relations = attachment->att_relations;
jrd_rel* check_relation = NULL;
2001-05-23 15:26:42 +02:00
vec<jrd_rel*>::iterator ptr = relations->begin();
2006-02-10 04:28:43 +01:00
for (const vec<jrd_rel*>::const_iterator end = relations->end(); ptr < end; ++ptr)
2001-05-23 15:26:42 +02:00
{
jrd_rel* const relation = *ptr;
if (relation)
2001-05-23 15:26:42 +02:00
{
2011-05-11 03:18:28 +02:00
if (relation->rel_flags & REL_deleting)
Jrd::Attachment::CheckoutLockGuard guard(attachment, relation->rel_drop_mutex, FB_FUNCTION);
if (!(relation->rel_flags & REL_deleted))
2001-05-23 15:26:42 +02:00
{
// dimitr: for non-system relations we should also check
// REL_scanned and REL_being_scanned flags. Look
// at MET_lookup_procedure for example.
if (!(relation->rel_flags & REL_system) &&
(!(relation->rel_flags & REL_scanned) || (relation->rel_flags & REL_being_scanned)))
2001-05-23 15:26:42 +02:00
{
continue;
}
2008-01-16 10:29:37 +01:00
if (relation->rel_name == name)
{
if (relation->rel_flags & REL_check_existence)
{
check_relation = relation;
LCK_lock(tdbb, check_relation->rel_existence_lock, LCK_SR, LCK_WAIT);
break;
}
return relation;
}
2001-05-23 15:26:42 +02:00
}
}
}
2009-11-21 04:23:47 +01:00
// We need to look up the relation name in RDB$RELATIONS
2001-05-23 15:26:42 +02:00
jrd_rel* relation = NULL;
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_l_relation, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$RELATIONS WITH X.RDB$RELATION_NAME EQ name.c_str()
{
2001-05-23 15:26:42 +02:00
relation = MET_relation(tdbb, X.RDB$RELATION_ID);
if (relation->rel_name.length() == 0) {
relation->rel_name = name;
}
relation->rel_flags |= get_rel_flags_from_FLAGS(X.RDB$FLAGS);
2011-06-04 04:18:55 +02:00
if (!X.RDB$RELATION_TYPE.NULL)
{
relation->rel_flags |= MET_get_rel_flags_from_TYPE(X.RDB$RELATION_TYPE);
}
2001-05-23 15:26:42 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (check_relation)
{
check_relation->rel_flags &= ~REL_check_existence;
2009-11-07 12:58:54 +01:00
if (check_relation != relation)
{
2001-05-23 15:26:42 +02:00
LCK_release(tdbb, check_relation->rel_existence_lock);
LCK_release(tdbb, check_relation->rel_partners_lock);
LCK_release(tdbb, check_relation->rel_rescan_lock);
check_relation->rel_flags &= ~REL_check_partners;
2001-05-23 15:26:42 +02:00
check_relation->rel_flags |= REL_deleted;
}
}
return relation;
}
jrd_rel* MET_lookup_relation_id(thread_db* tdbb, SLONG id, bool return_deleted)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ l o o k u p _ r e l a t i o n _ i d
*
**************************************
*
* Functional description
* Lookup relation by id. Make sure it really exists.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// System relations are above suspicion
2001-05-23 15:26:42 +02:00
if (id < (int) rel_MAX)
{
2004-11-01 04:18:42 +01:00
fb_assert(id < MAX_USHORT);
2001-05-23 15:26:42 +02:00
return MET_relation(tdbb, (USHORT) id);
}
jrd_rel* check_relation = NULL;
jrd_rel* relation;
2011-05-09 12:15:19 +02:00
vec<jrd_rel*>* vector = attachment->att_relations;
2008-12-18 11:57:12 +01:00
if (vector && (id < (SLONG) vector->count()) && (relation = (*vector)[id]))
2001-05-23 15:26:42 +02:00
{
2011-05-11 03:18:28 +02:00
if (relation->rel_flags & REL_deleting)
Jrd::Attachment::CheckoutLockGuard guard(attachment, relation->rel_drop_mutex, FB_FUNCTION);
2001-05-23 15:26:42 +02:00
if (relation->rel_flags & REL_deleted)
2008-01-16 10:29:37 +01:00
return return_deleted ? relation : NULL;
if (relation->rel_flags & REL_check_existence)
{
check_relation = relation;
LCK_lock(tdbb, check_relation->rel_existence_lock, LCK_SR, LCK_WAIT);
2001-05-23 15:26:42 +02:00
}
else
2008-01-16 10:29:37 +01:00
return relation;
2001-05-23 15:26:42 +02:00
}
2009-11-21 04:23:47 +01:00
// We need to look up the relation id in RDB$RELATIONS
2001-05-23 15:26:42 +02:00
relation = NULL;
AutoCacheRequest request(tdbb, irq_l_rel_id, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
X IN RDB$RELATIONS WITH X.RDB$RELATION_ID EQ id
{
2001-05-23 15:26:42 +02:00
relation = MET_relation(tdbb, X.RDB$RELATION_ID);
if (relation->rel_name.length() == 0) {
2008-07-01 03:12:02 +02:00
relation->rel_name = X.RDB$RELATION_NAME;
}
relation->rel_flags |= get_rel_flags_from_FLAGS(X.RDB$FLAGS);
2011-06-04 04:18:55 +02:00
if (!X.RDB$RELATION_TYPE.NULL)
{
relation->rel_flags |= MET_get_rel_flags_from_TYPE(X.RDB$RELATION_TYPE);
}
}
END_FOR
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (check_relation)
{
check_relation->rel_flags &= ~REL_check_existence;
2009-11-07 12:58:54 +01:00
if (check_relation != relation)
{
2001-05-23 15:26:42 +02:00
LCK_release(tdbb, check_relation->rel_existence_lock);
LCK_release(tdbb, check_relation->rel_partners_lock);
LCK_release(tdbb, check_relation->rel_rescan_lock);
check_relation->rel_flags &= ~REL_check_partners;
2001-05-23 15:26:42 +02:00
check_relation->rel_flags |= REL_deleted;
}
}
return relation;
}
DmlNode* MET_parse_blob(thread_db* tdbb,
2003-12-22 11:00:59 +01:00
jrd_rel* relation,
bid* blob_id,
CompilerScratch** csb_ptr,
JrdStatement** statementPtr,
const bool trigger,
bool validationExpr)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ p a r s e _ b l o b
*
**************************************
*
* Functional description
* Parse blr, returning a compiler scratch block with the results.
*
* if ignore_perm is true then, the request generated must be set to
* ignore all permissions checks. In this case, we call PAR_blr
* passing it the csb_ignore_perm flag to generate a request
* which must go through without checking any permissions.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), blob_id);
ULONG length = blob->blb_length + 10;
2010-06-12 18:32:46 +02:00
HalfStaticArray<UCHAR, 512> tmp;
UCHAR* temp = tmp.getBuffer(length);
length = blob->BLB_get_data(tdbb, temp, length);
2001-05-23 15:26:42 +02:00
DmlNode* node = NULL;
if (validationExpr)
{
// The set of MET parse functions needs a rework.
// For now, our caller chain is not interested in the returned node.
2013-03-31 20:23:54 +02:00
PAR_validation_blr(tdbb, relation, temp, length, NULL, csb_ptr, 0);
}
else
node = PAR_blr(tdbb, relation, temp, length, NULL, csb_ptr, statementPtr, trigger, 0);
2001-05-23 15:26:42 +02:00
return node;
}
void MET_parse_sys_trigger(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ p a r s e _ s y s _ t r i g g e r
*
**************************************
*
* Functional description
* Parse the blr for a system relation's triggers.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
relation->rel_flags &= ~REL_sys_triggers;
// Release any triggers in case of a rescan
2001-05-23 15:26:42 +02:00
if (relation->rel_pre_store)
MET_release_triggers(tdbb, &relation->rel_pre_store);
if (relation->rel_post_store)
MET_release_triggers(tdbb, &relation->rel_post_store);
if (relation->rel_pre_erase)
MET_release_triggers(tdbb, &relation->rel_pre_erase);
if (relation->rel_post_erase)
MET_release_triggers(tdbb, &relation->rel_post_erase);
if (relation->rel_pre_modify)
MET_release_triggers(tdbb, &relation->rel_pre_modify);
if (relation->rel_post_modify)
MET_release_triggers(tdbb, &relation->rel_post_modify);
// No need to load triggers for ReadOnly databases, since
// INSERT/DELETE/UPDATE statements are not going to be allowed
// hvlad: GTT with ON COMMIT DELETE ROWS clause is writable
2001-05-23 15:26:42 +02:00
if (dbb->readOnly() && !(relation->rel_flags & REL_temp_tran))
2001-05-23 15:26:42 +02:00
return;
relation->rel_flags |= REL_sys_trigs_being_loaded;
AutoCacheRequest request(tdbb, irq_s_triggers2, IRQ_REQUESTS);
2005-05-28 00:45:31 +02:00
FOR (REQUEST_HANDLE request)
2008-12-18 11:57:12 +01:00
TRG IN RDB$TRIGGERS
WITH TRG.RDB$RELATION_NAME = relation->rel_name.c_str()
AND TRG.RDB$SYSTEM_FLAG = 1
{
const FB_UINT64 type = TRG.RDB$TRIGGER_TYPE;
const USHORT trig_flags = TRG.RDB$FLAGS;
const TEXT* name = TRG.RDB$TRIGGER_NAME;
2008-12-18 11:57:12 +01:00
trig_vec** ptr;
2009-01-20 09:33:59 +01:00
switch (type)
{
2001-05-23 15:26:42 +02:00
case 1:
ptr = &relation->rel_pre_store;
break;
case 2:
ptr = &relation->rel_post_store;
break;
case 3:
ptr = &relation->rel_pre_modify;
break;
case 4:
ptr = &relation->rel_post_modify;
break;
case 5:
ptr = &relation->rel_pre_erase;
break;
case 6:
ptr = &relation->rel_post_erase;
break;
default:
ptr = NULL;
break;
}
if (ptr)
{
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), &TRG.RDB$TRIGGER_BLR);
ULONG length = blob->blb_length + 10;
2010-06-12 18:32:46 +02:00
HalfStaticArray<UCHAR, 128> blr;
length = blob->BLB_get_data(tdbb, blr.getBuffer(length), length);
2008-12-18 11:57:12 +01:00
USHORT par_flags = (USHORT) ((trig_flags & TRG_ignore_perm) ? csb_ignore_perm : 0);
if (type & 1)
par_flags |= csb_pre_trigger;
else
par_flags |= csb_post_trigger;
JrdStatement* statement = NULL;
{
2011-05-09 12:15:19 +02:00
Jrd::ContextPoolHolder context(tdbb, attachment->createPool());
PAR_blr(tdbb, relation, blr.begin(), length, NULL, NULL, &statement, true, par_flags);
}
2001-05-23 15:26:42 +02:00
statement->triggerName = name;
2001-05-23 15:26:42 +02:00
statement->flags |= JrdStatement::FLAG_SYS_TRIGGER;
2001-05-23 15:26:42 +02:00
if (trig_flags & TRG_ignore_perm)
statement->flags |= JrdStatement::FLAG_IGNORE_PERM;
2001-05-23 15:26:42 +02:00
save_trigger_data(tdbb, ptr, relation, statement, NULL, NULL, NULL, type, true, 0, "",
"", NULL);
2001-05-23 15:26:42 +02:00
}
}
END_FOR
2001-05-23 15:26:42 +02:00
relation->rel_flags &= ~REL_sys_trigs_being_loaded;
}
void MET_post_existence(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ p o s t _ e x i s t e n c e
*
**************************************
*
* Functional description
* Post an interest in the existence of a relation.
*
**************************************/
SET_TDBB(tdbb);
relation->rel_use_count++;
if (!MET_lookup_relation_id(tdbb, relation->rel_id, false))
{
relation->rel_use_count--;
ERR_post(Arg::Gds(isc_relnotdef) << Arg::Str(relation->rel_name));
}
2001-05-23 15:26:42 +02:00
}
void MET_prepare(thread_db* tdbb, jrd_tra* transaction, USHORT length, const UCHAR* msg)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
2005-05-28 00:45:31 +02:00
* M E T _ p r e p a r e
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Post a transaction description to RDB$TRANSACTIONS.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_s_trans, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
STORE(REQUEST_HANDLE request) X IN RDB$TRANSACTIONS
{
2001-05-23 15:26:42 +02:00
X.RDB$TRANSACTION_ID = transaction->tra_number;
2008-12-18 11:57:12 +01:00
X.RDB$TRANSACTION_STATE = RDB$TRANSACTIONS.RDB$TRANSACTION_STATE.LIMBO;
blb* blob = blb::create(tdbb, attachment->getSysTransaction(), &X.RDB$TRANSACTION_DESCRIPTION);
blob->BLB_put_segment(tdbb, msg, length);
blob->BLB_close(tdbb);
}
END_STORE
2001-05-23 15:26:42 +02:00
}
jrd_prc* MET_procedure(thread_db* tdbb, int id, bool noscan, USHORT flags)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ p r o c e d u r e
*
**************************************
*
* Functional description
* Find or create a procedure block for a given procedure id.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* vector = attachment->att_procedures;
2011-05-11 03:18:28 +02:00
if (!vector)
2011-05-09 12:15:19 +02:00
vector = attachment->att_procedures = vec<jrd_prc*>::newVector(*attachment->att_pool, id + 10);
else if (id >= (int) vector->count())
2001-12-24 03:51:06 +01:00
vector->resize(id + 10);
2001-05-23 15:26:42 +02:00
jrd_prc* procedure = (*vector)[id];
if (procedure && !(procedure->prc_flags & PRC_obsolete))
2001-05-23 15:26:42 +02:00
{
// Make sure PRC_being_scanned and PRC_scanned are not set at the same time
2008-12-18 11:57:12 +01:00
fb_assert(!(procedure->prc_flags & PRC_being_scanned) || !(procedure->prc_flags & PRC_scanned));
2001-05-23 15:26:42 +02:00
/* To avoid scanning recursive procedure's blr recursively let's
make use of PRC_being_scanned bit. Because this bit is set
later in the code, it is not set when we are here first time.
2005-05-28 00:45:31 +02:00
If (in case of rec. procedure) we get here second time it is
2001-05-23 15:26:42 +02:00
already set and we return half baked procedure.
In case of superserver this code is under the rec. mutex
protection, thus the only guy (thread) who can get here and see
PRC_being_scanned bit set is the guy which started procedure scan
and currently holds the mutex.
2005-05-28 00:45:31 +02:00
In case of classic, there is always only one guy and if it
2001-05-23 15:26:42 +02:00
sees PRC_being_scanned bit set it is safe to assume it is here
second time.
If procedure has already been scanned - return. This condition
is for those threads that did not find procedure in cache and
2001-05-23 15:26:42 +02:00
came here to get it from disk. But only one was able to lock
the mutex and do the scanning, others were waiting. As soon as
the first thread releases the mutex another thread gets in and
it would be just unfair to make it do the work again.
2001-05-23 15:26:42 +02:00
*/
2008-12-18 11:57:12 +01:00
if ((procedure->prc_flags & PRC_being_scanned) || (procedure->prc_flags & PRC_scanned))
2001-05-23 15:26:42 +02:00
{
return procedure;
}
}
if (!procedure)
2001-05-23 15:26:42 +02:00
{
procedure = FB_NEW(*dbb->dbb_permanent) jrd_prc(*dbb->dbb_permanent);
2001-05-23 15:26:42 +02:00
}
2001-12-24 03:51:06 +01:00
try {
2001-05-23 15:26:42 +02:00
procedure->prc_flags |= (PRC_being_scanned | flags);
procedure->prc_flags &= ~PRC_obsolete;
2005-05-28 00:45:31 +02:00
procedure->setId(id);
(*vector)[id] = procedure;
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (!procedure->prc_existence_lock)
{
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, 0)
Lock(tdbb, sizeof(SLONG), LCK_prc_exist, procedure, blocking_ast_procedure);
procedure->prc_existence_lock = lock;
lock->lck_key.lck_long = procedure->getId();
}
2001-05-23 15:26:42 +02:00
LCK_lock(tdbb, procedure->prc_existence_lock, LCK_SR, LCK_WAIT);
if (!noscan)
{
bool valid_blr = true;
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_r_procedure, IRQ_REQUESTS);
2007-07-04 18:07:55 +02:00
FOR(REQUEST_HANDLE request)
P IN RDB$PROCEDURES WITH P.RDB$PROCEDURE_ID EQ procedure->getId()
{
if (procedure->getName().toString().length() == 0)
{
procedure->setName(QualifiedName(P.RDB$PROCEDURE_NAME,
(P.RDB$PACKAGE_NAME.NULL ? "" : P.RDB$PACKAGE_NAME)));
}
procedure->setId(P.RDB$PROCEDURE_ID);
if (!P.RDB$SECURITY_CLASS.NULL)
procedure->setSecurityName(P.RDB$SECURITY_CLASS);
else if (!P.RDB$PACKAGE_NAME.NULL)
{
AutoCacheRequest requestHandle(tdbb, irq_l_procedure_pkg_class, IRQ_REQUESTS);
FOR (REQUEST_HANDLE requestHandle)
PKG IN RDB$PACKAGES
WITH PKG.RDB$PACKAGE_NAME EQ P.RDB$PACKAGE_NAME
{
if (!PKG.RDB$SECURITY_CLASS.NULL)
procedure->setSecurityName(PKG.RDB$SECURITY_CLASS);
}
END_FOR
}
procedure->setImplemented(true);
procedure->getInputFields().resize(P.RDB$PROCEDURE_INPUTS);
procedure->getOutputFields().resize(P.RDB$PROCEDURE_OUTPUTS);
procedure->setDefaultCount(0);
2007-07-04 18:07:55 +02:00
AutoCacheRequest request2(tdbb, irq_r_params, IRQ_REQUESTS);
2007-07-04 18:07:55 +02:00
2009-10-30 11:50:59 +01:00
const MetaName packageName(P.RDB$PACKAGE_NAME.NULL ? NULL : P.RDB$PACKAGE_NAME);
FOR (REQUEST_HANDLE request2)
PA IN RDB$PROCEDURE_PARAMETERS
CROSS F IN RDB$FIELDS
WITH F.RDB$FIELD_NAME = PA.RDB$FIELD_SOURCE AND
PA.RDB$PROCEDURE_NAME = P.RDB$PROCEDURE_NAME AND
2009-10-30 11:50:59 +01:00
PA.RDB$PACKAGE_NAME EQUIV NULLIF(packageName.c_str(), '')
{
2009-11-27 07:59:21 +01:00
const SSHORT pa_collation_id_null = PA.RDB$COLLATION_ID.NULL;
const SSHORT pa_collation_id = PA.RDB$COLLATION_ID;
const SSHORT pa_default_value_null = PA.RDB$DEFAULT_VALUE.NULL;
bid pa_default_value = pa_default_value_null ? F.RDB$DEFAULT_VALUE : PA.RDB$DEFAULT_VALUE;
Array<NestConst<Parameter> >& paramArray = PA.RDB$PARAMETER_TYPE ?
procedure->getOutputFields() : procedure->getInputFields();
// should be error if field already exists
2008-12-18 11:57:12 +01:00
Parameter* parameter = FB_NEW(*dbb->dbb_permanent) Parameter(*dbb->dbb_permanent);
parameter->prm_number = PA.RDB$PARAMETER_NUMBER;
paramArray[parameter->prm_number] = parameter;
parameter->prm_name = PA.RDB$PARAMETER_NAME;
parameter->prm_nullable = PA.RDB$NULL_FLAG.NULL || PA.RDB$NULL_FLAG == 0; // ODS_11_1
parameter->prm_mechanism = PA.RDB$PARAMETER_MECHANISM.NULL ? // ODS_11_1
prm_mech_normal : (prm_mech_t) PA.RDB$PARAMETER_MECHANISM;
if (!PA.RDB$FIELD_SOURCE.NULL)
parameter->prm_field_source = PA.RDB$FIELD_SOURCE;
DSC_make_descriptor(&parameter->prm_desc, F.RDB$FIELD_TYPE,
F.RDB$FIELD_SCALE, F.RDB$FIELD_LENGTH,
F.RDB$FIELD_SUB_TYPE, F.RDB$CHARACTER_SET_ID,
(pa_collation_id_null ? F.RDB$COLLATION_ID : pa_collation_id));
2004-01-16 13:59:16 +01:00
if (PA.RDB$PARAMETER_TYPE == 0 &&
(!pa_default_value_null ||
2006-09-01 03:40:22 +02:00
(fb_utils::implicit_domain(F.RDB$FIELD_NAME) && !F.RDB$DEFAULT_VALUE.NULL)))
{
procedure->setDefaultCount(procedure->getDefaultCount() + 1);
2011-05-09 12:15:19 +02:00
MemoryPool* pool = attachment->createPool();
Jrd::ContextPoolHolder context(tdbb, pool);
try
{
parameter->prm_default_value = static_cast<ValueExprNode*>(
MET_parse_blob(tdbb, NULL, &pa_default_value, NULL, NULL, false, false));
}
2010-04-14 00:50:15 +02:00
catch (const Exception&)
{
2010-04-14 11:47:51 +02:00
// Here we lose pools created for previous defaults.
// Probably we should use common pool for defaults and procedure itself.
2005-05-28 00:45:31 +02:00
2011-05-09 12:15:19 +02:00
attachment->deletePool(pool);
throw;
}
2004-01-16 13:59:16 +01:00
}
}
END_FOR
const bool external = !P.RDB$ENGINE_NAME.NULL; // ODS_12_0
Array<NestConst<Parameter> >& paramArray = procedure->getOutputFields();
if (paramArray.hasData() && paramArray[0])
2001-05-23 15:26:42 +02:00
{
Format* format = Format::newFormat(
*dbb->dbb_permanent, procedure->getOutputFields().getCount());
procedure->prc_record_format = format;
ULONG length = FLAG_BYTES(format->fmt_count);
Format::fmt_desc_iterator desc = format->fmt_desc.begin();
Array<NestConst<Parameter> >::iterator ptr, end;
for (ptr = paramArray.begin(), end = paramArray.end(); ptr < end; ++ptr, ++desc)
2001-05-23 15:26:42 +02:00
{
const Parameter* parameter = *ptr;
// check for parameter to be null, this can only happen if the
// parameter numbers get out of sync. This was added to fix bug
// 10534. -Shaunak Mistry 12-May-99
if (parameter)
{
*desc = parameter->prm_desc;
2009-11-27 07:59:21 +01:00
length = MET_align(&(*desc), length);
desc->dsc_address = (UCHAR *) (IPTR) length;
length += desc->dsc_length;
}
2001-05-23 15:26:42 +02:00
}
2011-01-29 18:24:29 +01:00
format->fmt_length = length;
2001-05-23 15:26:42 +02:00
}
procedure->prc_type = P.RDB$PROCEDURE_TYPE.NULL ?
prc_legacy : (prc_t) P.RDB$PROCEDURE_TYPE;
if (external || !P.RDB$PROCEDURE_BLR.NULL)
{
MemoryPool* csb_pool = attachment->createPool();
Jrd::ContextPoolHolder context(tdbb, csb_pool);
CompilerScratch* csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), 5);
if (!P.RDB$DEBUG_INFO.NULL)
DBG_parse_debug_info(tdbb, &P.RDB$DEBUG_INFO, *csb->csb_dbg_info);
try
{
if (external)
{
HalfStaticArray<char, 512> body;
if (!P.RDB$PROCEDURE_SOURCE.NULL)
{
blb* blob = blb::open(tdbb, attachment->getSysTransaction(),
&P.RDB$PROCEDURE_SOURCE);
ULONG len = blob->BLB_get_data(tdbb,
(UCHAR*) body.getBuffer(blob->blb_length + 1), blob->blb_length + 1);
body.begin()[MIN(blob->blb_length, len)] = '\0';
}
else
body.getBuffer(1)[0] = '\0';
2006-11-03 10:42:42 +01:00
dbb->dbb_extManager.makeProcedure(tdbb, csb, procedure, P.RDB$ENGINE_NAME,
(P.RDB$ENTRYPOINT.NULL ? "" : P.RDB$ENTRYPOINT), body.begin());
}
else
procedure->parseBlr(tdbb, csb, &P.RDB$PROCEDURE_BLR);
}
catch (const Exception&)
{
delete csb;
if (procedure->getStatement())
procedure->releaseStatement(tdbb);
else
attachment->deletePool(csb_pool);
2013-03-30 03:31:03 +01:00
ERR_post(Arg::Gds(isc_bad_proc_BLR) << Arg::Str(procedure->getName().toString()));
}
procedure->getStatement()->procedure = procedure;
2013-03-30 03:31:03 +01:00
for (size_t i = 0; i < csb->csb_rpt.getCount(); i++)
{
MessageNode* node = csb->csb_rpt[i].csb_message;
/***
if (node)
{
if (node->messageNumber == 1)
procedure->prc_output_msg = node;
}
***/
2001-05-23 15:26:42 +02:00
}
delete csb;
}
else
{
RefPtr<MsgMetadata> inputMetadata(REF_NO_INCR,
Routine::createMetadata(procedure->getInputFields()));
procedure->setInputFormat(
Routine::createFormat(procedure->getPool(), inputMetadata, false));
RefPtr<MsgMetadata> outputMetadata(REF_NO_INCR,
Routine::createMetadata(procedure->getOutputFields()));
procedure->setOutputFormat(
Routine::createFormat(procedure->getPool(), outputMetadata, true));
procedure->setImplemented(false);
}
if (P.RDB$VALID_BLR.NULL || P.RDB$VALID_BLR == FALSE)
valid_blr = false;
2001-05-23 15:26:42 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
procedure->prc_flags |= PRC_scanned;
2001-05-23 15:26:42 +02:00
if (!dbb->readOnly() && !valid_blr)
{
// if the BLR was marked as invalid but the procedure was compiled,
// mark the BLR as valid
AutoRequest request5;
2006-08-30 06:54:44 +02:00
FOR(REQUEST_HANDLE request5)
P IN RDB$PROCEDURES WITH
P.RDB$PROCEDURE_ID EQ procedure->getId() AND P.RDB$PROCEDURE_BLR NOT MISSING
2010-04-23 03:59:21 +02:00
{
MODIFY P USING
P.RDB$VALID_BLR = TRUE;
P.RDB$VALID_BLR.NULL = FALSE;
END_MODIFY
2010-04-23 03:59:21 +02:00
}
END_FOR
}
} // if !noscan
2005-05-28 00:45:31 +02:00
// Make sure that it is really being scanned
2003-11-04 00:59:24 +01:00
fb_assert(procedure->prc_flags & PRC_being_scanned);
2005-05-28 00:45:31 +02:00
2001-05-23 15:26:42 +02:00
procedure->prc_flags &= ~PRC_being_scanned;
2001-12-24 03:51:06 +01:00
} // try
2010-06-12 18:32:46 +02:00
catch (const Exception&)
2009-11-07 12:58:54 +01:00
{
2001-12-24 03:51:06 +01:00
procedure->prc_flags &= ~(PRC_being_scanned | PRC_scanned);
if (procedure->getExternal())
{
delete procedure->getExternal();
procedure->setExternal(NULL);
}
2001-12-24 03:51:06 +01:00
if (procedure->prc_existence_lock)
{
LCK_release(tdbb, procedure->prc_existence_lock);
2012-07-10 18:08:01 +02:00
delete procedure->prc_existence_lock;
procedure->prc_existence_lock = NULL;
2001-12-24 03:51:06 +01:00
}
throw;
2001-12-24 03:51:06 +01:00
}
2001-05-23 15:26:42 +02:00
return procedure;
}
jrd_rel* MET_relation(thread_db* tdbb, USHORT id)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ r e l a t i o n
*
**************************************
*
* Functional description
* Find or create a relation block for a given relation id.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
2011-05-09 12:15:19 +02:00
Jrd::Attachment* attachment = tdbb->getAttachment();
vec<jrd_rel*>* vector = attachment->att_relations;
MemoryPool* pool = attachment->att_pool;
2001-12-24 03:51:06 +01:00
if (!vector)
2011-05-09 12:15:19 +02:00
vector = attachment->att_relations = vec<jrd_rel*>::newVector(*pool, id + 10);
2001-12-24 03:51:06 +01:00
else if (id >= vector->count())
vector->resize(id + 10);
2001-05-23 15:26:42 +02:00
jrd_rel* relation = (*vector)[id];
2011-05-11 03:18:28 +02:00
if (relation)
2001-05-23 15:26:42 +02:00
return relation;
2009-11-21 04:23:47 +01:00
// From ODS 9 onwards, the first 128 relation IDS have been
// reserved for system relations
const USHORT max_sys_rel = USER_DEF_REL_INIT_ID - 1;
2001-05-23 15:26:42 +02:00
2011-05-09 12:15:19 +02:00
relation = FB_NEW(*pool) jrd_rel(*pool);
(*vector)[id] = relation;
2001-05-23 15:26:42 +02:00
relation->rel_id = id;
2005-07-08 05:25:31 +02:00
{ // Scope block.
Lock* lock = FB_NEW_RPT(*pool, 0)
Lock(tdbb, sizeof(SLONG), LCK_rel_partners, relation, partners_ast_relation);
2005-07-08 05:25:31 +02:00
relation->rel_partners_lock = lock;
lock->lck_key.lck_long = relation->rel_id;
}
{ // Scope block.
Lock* lock = FB_NEW_RPT(*pool, 0)
Lock(tdbb, sizeof(SLONG), LCK_rel_rescan, relation, rescan_ast_relation);
relation->rel_rescan_lock = lock;
lock->lck_key.lck_long = relation->rel_id;
}
// This should check system flag instead.
if (relation->rel_id <= max_sys_rel) {
2001-05-23 15:26:42 +02:00
return relation;
}
2001-05-23 15:26:42 +02:00
2005-07-08 05:25:31 +02:00
{ // Scope block.
Lock* lock = FB_NEW_RPT(*pool, 0)
Lock(tdbb, sizeof(SLONG), LCK_rel_exist, relation, blocking_ast_relation);
2005-07-08 05:25:31 +02:00
relation->rel_existence_lock = lock;
lock->lck_key.lck_long = relation->rel_id;
}
2001-05-23 15:26:42 +02:00
2008-03-04 14:13:19 +01:00
relation->rel_flags |= (REL_check_existence | REL_check_partners);
2001-05-23 15:26:42 +02:00
return relation;
}
void MET_release_existence(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ r e l e a s e _ e x i s t e n c e
*
**************************************
*
* Functional description
* Release interest in relation. If no remaining interest
* and we're blocking the drop of the relation then release
* existence lock and mark deleted.
*
**************************************/
if (relation->rel_use_count) {
relation->rel_use_count--;
}
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (!relation->rel_use_count)
{
if (relation->rel_flags & REL_blocking) {
LCK_re_post(tdbb, relation->rel_existence_lock);
}
2009-11-07 12:58:54 +01:00
if (relation->rel_file)
{
// close external file
EXT_fini(relation, true);
}
}
2001-05-23 15:26:42 +02:00
}
void MET_remove_procedure(thread_db* tdbb, int id, jrd_prc* procedure)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ r e m o v e _ p r o c e d u r e
*
**************************************
*
* Functional description
* Remove a procedure from cache
*
**************************************/
SET_TDBB(tdbb);
2011-05-09 12:15:19 +02:00
Jrd::Attachment* att = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2011-05-09 12:15:19 +02:00
vec<jrd_prc*>* pvector = att->att_procedures;
2011-05-11 03:18:28 +02:00
if (!pvector)
2002-09-28 22:58:40 +02:00
return;
2009-11-07 12:58:54 +01:00
if (!procedure)
{
2009-11-21 04:23:47 +01:00
// If we are in here then dfw.epp/modify_procedure() called us
if (!(procedure = (*pvector)[id]))
2001-05-23 15:26:42 +02:00
return;
}
2005-05-28 00:45:31 +02:00
2009-11-21 04:23:47 +01:00
// MET_procedure locks it. Lets unlock it now to avoid troubles later
if (procedure->prc_existence_lock)
LCK_release(tdbb, procedure->prc_existence_lock);
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Procedure that is being altered may have references
// to it by other procedures via pointer to current meta
// data structure, so don't loose the structure or the pointer.
2008-12-18 11:57:12 +01:00
if ((procedure == (*pvector)[id]) && !(procedure->prc_flags & PRC_being_altered))
(*pvector)[id] = NULL;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// deallocate all structure which were allocated. The procedure
// blr is originally read into a new pool from which all request
// are allocated. That will not be freed up.
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (procedure->prc_existence_lock)
{
2001-12-24 03:51:06 +01:00
delete procedure->prc_existence_lock;
procedure->prc_existence_lock = NULL;
}
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// deallocate input param structures
for (Array<NestConst<Parameter> >::iterator i = procedure->getInputFields().begin();
i != procedure->getInputFields().end(); ++i)
2009-11-07 12:58:54 +01:00
{
if (*i)
delete i->getObject();
2001-05-23 15:26:42 +02:00
}
procedure->getInputFields().clear();
2009-11-21 04:23:47 +01:00
// deallocate output param structures
2001-05-23 15:26:42 +02:00
for (Array<NestConst<Parameter> >::iterator i = procedure->getOutputFields().begin();
i != procedure->getOutputFields().end(); ++i)
2009-11-07 12:58:54 +01:00
{
if (*i)
delete i->getObject();
2001-05-23 15:26:42 +02:00
}
procedure->getOutputFields().clear();
if (!procedure->prc_use_count)
{
if (procedure->prc_record_format)
{
delete procedure->prc_record_format;
procedure->prc_record_format = NULL;
}
}
2001-05-23 15:26:42 +02:00
if (!(procedure->prc_flags & PRC_being_altered) && !procedure->prc_use_count)
2001-05-23 15:26:42 +02:00
{
2001-12-24 03:51:06 +01:00
delete procedure;
2001-05-23 15:26:42 +02:00
}
else
{
2005-05-28 00:45:31 +02:00
// Fully clear procedure block. Some pieces of code check for empty
// procedure name and ID, this is why we do it.
procedure->setName(QualifiedName());
procedure->setSecurityName("");
procedure->setId(0);
procedure->setDefaultCount(0);
2001-05-23 15:26:42 +02:00
}
}
2012-01-09 01:08:58 +01:00
void MET_revoke(thread_db* tdbb, jrd_tra* transaction, const MetaName& relation,
const MetaName& revokee, const string& privilege)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ r e v o k e
*
**************************************
*
* Functional description
* Execute a recursive revoke. This is called only when
* a revoked privilege had the grant option.
*
**************************************/
SET_TDBB(tdbb);
2009-11-21 04:23:47 +01:00
// See if the revokee still has the privilege. If so, there's nothing to do
2001-05-23 15:26:42 +02:00
USHORT count = 0;
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_revoke1, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request TRANSACTION_HANDLE transaction)
FIRST 1 P IN RDB$USER_PRIVILEGES WITH
P.RDB$RELATION_NAME EQ relation.c_str() AND
P.RDB$PRIVILEGE EQ privilege.c_str() AND
P.RDB$USER EQ revokee.c_str()
{
2001-05-23 15:26:42 +02:00
++count;
}
END_FOR
2001-05-23 15:26:42 +02:00
if (count)
return;
request.reset(tdbb, irq_revoke2, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// User lost privilege. Take it away from anybody he/she gave it to.
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request TRANSACTION_HANDLE transaction)
P IN RDB$USER_PRIVILEGES WITH
P.RDB$RELATION_NAME EQ relation.c_str() AND
P.RDB$PRIVILEGE EQ privilege.c_str() AND
P.RDB$GRANTOR EQ revokee.c_str()
{
2001-05-23 15:26:42 +02:00
ERASE P;
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_scan_partners(thread_db* tdbb, jrd_rel* relation)
{
/**************************************
*
* M E T _ s c a n _ p a r t n e r s
*
**************************************
*
* Functional description
2010-01-21 03:51:32 +01:00
* Scan of foreign references on other relations' primary keys and
* scan of primary dependencies on relation's primary key.
*
**************************************/
SET_TDBB(tdbb);
2010-01-21 03:51:32 +01:00
if (relation->rel_flags & REL_check_partners)
{
scan_partners(tdbb, relation);
}
}
void MET_scan_relation(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ s c a n _ r e l a t i o n
*
**************************************
*
* Functional description
* Scan a relation for view RecordSelExpr, computed by expressions, missing
2001-05-23 15:26:42 +02:00
* expressions, and validation expressions.
*
**************************************/
SET_TDBB(tdbb);
trig_vec* triggers[TRIGGER_MAX];
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
Jrd::ContextPoolHolder context(tdbb, dbb->dbb_permanent);
bool dependencies = false;
bool sys_triggers = false;
2001-05-23 15:26:42 +02:00
blb* blob = NULL;
jrd_tra* depTrans = tdbb->getTransaction() ?
tdbb->getTransaction() : attachment->getSysTransaction();
2009-11-21 04:23:47 +01:00
// If anything errors, catch it to reset the scan flag. This will
// make sure that the error will be caught if the operation is tried again.
try {
2008-02-06 01:43:54 +01:00
if (relation->rel_flags & REL_scanned || relation->rel_flags & REL_deleted)
2001-05-23 15:26:42 +02:00
return;
relation->rel_flags |= REL_being_scanned;
dependencies = (relation->rel_flags & REL_get_dependencies) ? true : false;
sys_triggers = (relation->rel_flags & REL_sys_triggers) ? true : false;
2001-05-23 15:26:42 +02:00
relation->rel_flags &= ~(REL_get_dependencies | REL_sys_triggers);
2008-02-06 01:43:54 +01:00
2011-05-11 03:18:28 +02:00
for (USHORT itr = 0; itr < TRIGGER_MAX; ++itr)
triggers[itr] = NULL;
2001-12-24 03:51:06 +01:00
2009-11-21 04:23:47 +01:00
// Since this can be called recursively, find an inactive clone of the request
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_r_fields, IRQ_REQUESTS);
CompilerScratch* csb = NULL;
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
REL IN RDB$RELATIONS WITH REL.RDB$RELATION_ID EQ relation->rel_id
{
2009-11-21 04:23:47 +01:00
// Pick up relation level stuff
2001-05-23 15:26:42 +02:00
relation->rel_current_fmt = REL.RDB$FORMAT;
vec<jrd_fld*>* vector = relation->rel_fields =
2011-05-09 12:15:19 +02:00
vec<jrd_fld*>::newVector(*relation->rel_pool, relation->rel_fields, REL.RDB$FIELD_ID + 1);
2001-12-24 03:51:06 +01:00
if (!REL.RDB$SECURITY_CLASS.NULL) {
relation->rel_security_name = REL.RDB$SECURITY_CLASS;
2001-12-24 03:51:06 +01:00
}
relation->rel_name = REL.RDB$RELATION_NAME;
relation->rel_owner_name = REL.RDB$OWNER_NAME;
if (!REL.RDB$VIEW_BLR.isEmpty())
2001-05-23 15:26:42 +02:00
{
2009-11-21 04:23:47 +01:00
// parse the view blr, getting dependencies on relations, etc. at the same time
2001-05-23 15:26:42 +02:00
DmlNode* rseNode;
2001-05-23 15:26:42 +02:00
if (dependencies)
{
2010-06-12 18:32:46 +02:00
const MetaName depName(REL.RDB$RELATION_NAME);
rseNode = MET_get_dependencies(tdbb, relation, NULL, 0, NULL, &REL.RDB$VIEW_BLR,
NULL, &csb, depName, obj_view, 0, depTrans);
2001-05-23 15:26:42 +02:00
}
else
rseNode = MET_parse_blob(tdbb, relation, &REL.RDB$VIEW_BLR, &csb, NULL, false, false);
if (rseNode)
2001-05-23 15:26:42 +02:00
{
fb_assert(rseNode->kind == DmlNode::KIND_REC_SOURCE);
relation->rel_view_rse = static_cast<RseNode*>(rseNode);
fb_assert(relation->rel_view_rse->type == RseNode::TYPE);
2001-05-23 15:26:42 +02:00
}
else
relation->rel_view_rse = NULL;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// retrieve the view context names
2001-05-23 15:26:42 +02:00
lookup_view_contexts(tdbb, relation);
}
relation->rel_flags |= REL_scanned;
if (REL.RDB$EXTERNAL_FILE[0])
{
2009-04-28 15:08:04 +02:00
EXT_file(relation, REL.RDB$EXTERNAL_FILE); //, &REL.RDB$EXTERNAL_DESCRIPTION);
2001-05-23 15:26:42 +02:00
}
2009-11-27 07:59:21 +01:00
if (!REL.RDB$RELATION_TYPE.NULL)
{
switch (REL.RDB$RELATION_TYPE)
{
case rel_persistent:
break;
case rel_external:
fb_assert(relation->rel_file);
break;
case rel_view:
fb_assert(relation->rel_view_rse);
fb_assert(relation->rel_flags & REL_jrd_view);
relation->rel_flags |= REL_jrd_view;
2009-11-27 07:59:21 +01:00
break;
case rel_virtual:
fb_assert(relation->rel_flags & REL_virtual);
relation->rel_flags |= REL_virtual;
2009-11-27 07:59:21 +01:00
break;
case rel_global_temp_preserve:
fb_assert(relation->rel_flags & REL_temp_conn);
relation->rel_flags |= REL_temp_conn;
2009-11-27 07:59:21 +01:00
break;
case rel_global_temp_delete:
fb_assert(relation->rel_flags & REL_temp_tran);
relation->rel_flags |= REL_temp_tran;
2009-11-27 07:59:21 +01:00
break;
default:
fb_assert(false);
}
}
2009-11-21 04:23:47 +01:00
// Pick up field specific stuff
2001-05-23 15:26:42 +02:00
blob = blb::open(tdbb, attachment->getSysTransaction(), &REL.RDB$RUNTIME);
2010-06-12 18:32:46 +02:00
HalfStaticArray<UCHAR, 256> temp;
2012-02-15 08:13:41 +01:00
UCHAR* const buffer = temp.getBuffer(blob->getMaxSegment() + 1U);
2001-05-23 15:26:42 +02:00
jrd_fld* field = NULL;
ArrayField* array = 0;
USHORT view_context = 0;
USHORT field_id = 0;
2001-05-23 15:26:42 +02:00
for (;;)
{
2012-02-15 08:13:41 +01:00
USHORT length = blob->BLB_get_segment(tdbb, buffer, blob->getMaxSegment());
2001-05-23 15:26:42 +02:00
if (blob->blb_flags & BLB_eof)
{
break;
}
USHORT n;
2001-05-23 15:26:42 +02:00
buffer[length] = 0;
UCHAR* p = (UCHAR*) &n;
const UCHAR* q = buffer + 1;
2001-05-23 15:26:42 +02:00
while (q < buffer + 1 + sizeof(SSHORT))
{
*p++ = *q++;
}
p = buffer + 1;
--length;
2009-04-03 12:07:55 +02:00
switch ((rsr_t) buffer[0])
2009-01-20 09:33:59 +01:00
{
2001-05-23 15:26:42 +02:00
case RSR_field_id:
2008-12-18 11:57:12 +01:00
if (field && field->fld_security_name.length() == 0 && !REL.RDB$DEFAULT_CLASS.NULL)
{
field->fld_security_name = REL.RDB$DEFAULT_CLASS;
}
2001-05-23 15:26:42 +02:00
field_id = n;
field = (*vector)[field_id];
2009-11-07 12:58:54 +01:00
if (field)
{
2006-07-23 13:01:20 +02:00
field->fld_computation = NULL;
field->fld_missing_value = NULL;
field->fld_default_value = NULL;
field->fld_validation = NULL;
field->fld_not_null = NULL;
}
2001-05-23 15:26:42 +02:00
array = NULL;
break;
case RSR_field_name:
2009-11-07 12:58:54 +01:00
if (field)
{
2009-11-21 04:23:47 +01:00
// The field exists. If its name hasn't changed, then
// there's no need to copy anything.
2001-05-23 15:26:42 +02:00
if (field->fld_name == reinterpret_cast<char*>(p))
2001-05-23 15:26:42 +02:00
break;
field->fld_name = reinterpret_cast<char*>(p);
2001-05-23 15:26:42 +02:00
}
2009-11-07 12:58:54 +01:00
else
{
field = FB_NEW(*dbb->dbb_permanent) jrd_fld(*dbb->dbb_permanent);
(*vector)[field_id] = field;
field->fld_name = reinterpret_cast<char*>(p);
2001-05-23 15:26:42 +02:00
}
2002-07-01 18:59:09 +02:00
// CVC: Be paranoid and allow the possible trigger(s) to have a
// not null security class to work on, even if we only take it
// from the relation itself.
2008-12-18 11:57:12 +01:00
if (field->fld_security_name.length() == 0 && !REL.RDB$DEFAULT_CLASS.NULL)
2005-05-22 05:11:41 +02:00
{
field->fld_security_name = REL.RDB$DEFAULT_CLASS;
}
2002-07-01 18:59:09 +02:00
2001-05-23 15:26:42 +02:00
break;
case RSR_view_context:
view_context = n;
break;
case RSR_base_field:
2009-11-07 12:58:54 +01:00
if (dependencies)
{
2006-01-13 10:33:40 +01:00
csb->csb_g_flags |= csb_get_dependencies;
field->fld_source = PAR_make_field(tdbb, csb, view_context, (TEXT*) p);
2010-06-12 18:32:46 +02:00
const MetaName depName(REL.RDB$RELATION_NAME);
store_dependencies(tdbb, csb, 0, depName, obj_view, depTrans);
2006-01-13 10:33:40 +01:00
}
else
field->fld_source = PAR_make_field(tdbb, csb, view_context, (TEXT*) p);
2001-05-23 15:26:42 +02:00
break;
case RSR_computed_blr:
{
DmlNode* nod = dependencies ?
MET_get_dependencies(tdbb, relation, p, length, csb, NULL, NULL, NULL,
field->fld_name, obj_computed, 0, depTrans) :
PAR_blr(tdbb, relation, p, length, csb, NULL, NULL, false, 0);
field->fld_computation = static_cast<ValueExprNode*>(nod);
}
2001-05-23 15:26:42 +02:00
break;
case RSR_missing_value:
field->fld_missing_value = static_cast<ValueExprNode*>(
PAR_blr(tdbb, relation, p, length, csb, NULL, NULL, false, 0));
2001-05-23 15:26:42 +02:00
break;
case RSR_default_value:
field->fld_default_value = static_cast<ValueExprNode*>(
PAR_blr(tdbb, relation, p, length, csb, NULL, NULL, false, 0));
2001-05-23 15:26:42 +02:00
break;
case RSR_validation_blr:
2005-04-25 08:54:45 +02:00
// AB: 2005-04-25 bug SF#1168898
// Ignore validation for VIEWs, because fields (domains) which are
// defined with CHECK constraints and have sub-selects should at least
2005-05-28 00:45:31 +02:00
// be parsed without view-context information. With view-context
2005-04-25 08:54:45 +02:00
// information the context-numbers are wrong.
// Because a VIEW can't have a validation section i ignored the whole call.
2005-05-28 00:45:31 +02:00
if (!csb)
{
2013-03-31 20:23:54 +02:00
field->fld_validation = PAR_validation_blr(tdbb, relation, p, length, csb,
NULL, csb_validation);
}
2001-05-23 15:26:42 +02:00
break;
case RSR_field_not_null:
2013-03-31 20:23:54 +02:00
field->fld_not_null = PAR_validation_blr(tdbb, relation, p, length, csb,
NULL, csb_validation);
2001-05-23 15:26:42 +02:00
break;
case RSR_security_class:
field->fld_security_name = (const TEXT*) p;
2001-05-23 15:26:42 +02:00
break;
case RSR_trigger_name:
MET_load_trigger(tdbb, relation, (const TEXT*) p, triggers);
2001-05-23 15:26:42 +02:00
break;
case RSR_dimensions:
field->fld_array = array = FB_NEW_RPT(*dbb->dbb_permanent, n) ArrayField();
array->arr_desc.iad_dimensions = n;
2001-05-23 15:26:42 +02:00
break;
case RSR_array_desc:
if (array)
2008-02-03 11:41:44 +01:00
memcpy(&array->arr_desc, p, length);
2001-05-23 15:26:42 +02:00
break;
case RSR_field_generator_name:
field->fld_generator_name = (const TEXT*) p;
break;
2009-11-21 04:23:47 +01:00
default: // Shut up compiler warning
break;
2001-05-23 15:26:42 +02:00
}
}
blob->BLB_close(tdbb);
blob = 0;
2009-11-27 07:59:21 +01:00
if (field && field->fld_security_name.length() == 0 && !REL.RDB$DEFAULT_CLASS.NULL)
2001-05-23 15:26:42 +02:00
{
field->fld_security_name = REL.RDB$DEFAULT_CLASS;
2001-05-23 15:26:42 +02:00
}
}
END_FOR
2005-04-28 08:12:08 +02:00
delete csb;
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// release any triggers in case of a rescan, but not if the rescan
// hapenned while system triggers were being loaded.
2001-05-23 15:26:42 +02:00
2009-11-07 12:58:54 +01:00
if (!(relation->rel_flags & REL_sys_trigs_being_loaded))
{
2009-11-21 04:23:47 +01:00
// if we are scanning a system relation during loading the system
// triggers, (during parsing its blr actually), we must not release the
// existing system triggers; because we have already set the
// relation->rel_flag to not have REL_sys_trig, so these
// system triggers will not get loaded again. This fixes bug 8149.
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// We have just loaded the triggers onto the local vector triggers.
// Its now time to place them at their rightful place ie the relation block.
trig_vec* tmp_vector;
2005-05-28 00:45:31 +02:00
2001-05-23 15:26:42 +02:00
tmp_vector = relation->rel_pre_store;
relation->rel_pre_store = triggers[TRIGGER_PRE_STORE];
MET_release_triggers(tdbb, &tmp_vector);
tmp_vector = relation->rel_post_store;
relation->rel_post_store = triggers[TRIGGER_POST_STORE];
MET_release_triggers(tdbb, &tmp_vector);
tmp_vector = relation->rel_pre_erase;
relation->rel_pre_erase = triggers[TRIGGER_PRE_ERASE];
MET_release_triggers(tdbb, &tmp_vector);
tmp_vector = relation->rel_post_erase;
relation->rel_post_erase = triggers[TRIGGER_POST_ERASE];
MET_release_triggers(tdbb, &tmp_vector);
tmp_vector = relation->rel_pre_modify;
relation->rel_pre_modify = triggers[TRIGGER_PRE_MODIFY];
MET_release_triggers(tdbb, &tmp_vector);
tmp_vector = relation->rel_post_modify;
relation->rel_post_modify = triggers[TRIGGER_POST_MODIFY];
MET_release_triggers(tdbb, &tmp_vector);
}
LCK_lock(tdbb, relation->rel_rescan_lock, LCK_SR, LCK_WAIT);
2001-05-23 15:26:42 +02:00
relation->rel_flags &= ~REL_being_scanned;
2008-01-16 10:29:37 +01:00
2001-05-23 15:26:42 +02:00
relation->rel_current_format = NULL;
2001-12-24 03:51:06 +01:00
} // try
2010-06-12 18:32:46 +02:00
catch (const Exception&)
2009-11-07 12:58:54 +01:00
{
2001-12-24 03:51:06 +01:00
relation->rel_flags &= ~(REL_being_scanned | REL_scanned);
if (dependencies) {
relation->rel_flags |= REL_get_dependencies;
}
if (sys_triggers) {
relation->rel_flags |= REL_sys_triggers;
}
if (blob)
blob->BLB_close(tdbb);
2008-01-16 10:29:37 +01:00
throw;
2001-12-24 03:51:06 +01:00
}
2001-05-23 15:26:42 +02:00
}
2010-06-12 18:32:46 +02:00
void MET_trigger_msg(thread_db* tdbb, string& msg, const MetaName& name,
2009-11-21 06:20:18 +01:00
USHORT number)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ t r i g g e r _ m s g
*
**************************************
*
* Functional description
* Look up trigger message using trigger and abort code.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_s_msgs, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
MSG IN RDB$TRIGGER_MESSAGES WITH
MSG.RDB$TRIGGER_NAME EQ name.c_str() AND
2005-05-28 00:45:31 +02:00
MSG.RDB$MESSAGE_NUMBER EQ number
{
msg = MSG.RDB$MESSAGE;
}
END_FOR
2001-05-23 15:26:42 +02:00
msg.rtrim();
2001-05-23 15:26:42 +02:00
}
void MET_update_shadow(thread_db* tdbb, Shadow* shadow, USHORT file_flags)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ u p d a t e _ s h a d o w
*
**************************************
*
* Functional description
* Update the stored file flags for the specified shadow.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2010-04-23 03:59:21 +02:00
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
FIL IN RDB$FILES WITH FIL.RDB$SHADOW_NUMBER EQ shadow->sdw_number
2010-04-23 03:59:21 +02:00
{
2001-05-23 15:26:42 +02:00
MODIFY FIL USING
FIL.RDB$FILE_FLAGS = file_flags;
END_MODIFY
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
}
void MET_update_transaction(thread_db* tdbb, jrd_tra* transaction, const bool do_commit)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* M E T _ u p d a t e _ t r a n s a c t i o n
*
**************************************
*
* Functional description
2003-12-22 11:00:59 +01:00
* Update a record in RDB$TRANSACTIONS. If do_commit is true, this is a
2001-05-23 15:26:42 +02:00
* commit; otherwise it is a ROLLBACK.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
AutoCacheRequest request(tdbb, irq_m_trans, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
2005-05-28 00:45:31 +02:00
X IN RDB$TRANSACTIONS
2008-12-18 11:57:12 +01:00
WITH X.RDB$TRANSACTION_ID EQ transaction->tra_number
{
2003-12-22 11:00:59 +01:00
if (do_commit && (transaction->tra_flags & TRA_prepare2))
2001-05-23 15:26:42 +02:00
ERASE X
else
{
MODIFY X
2009-02-28 12:57:40 +01:00
X.RDB$TRANSACTION_STATE = do_commit ?
2001-05-23 15:26:42 +02:00
RDB$TRANSACTIONS.RDB$TRANSACTION_STATE.COMMITTED :
RDB$TRANSACTIONS.RDB$TRANSACTION_STATE.ROLLED_BACK;
END_MODIFY
2001-05-23 15:26:42 +02:00
}
}
END_FOR
2001-05-23 15:26:42 +02:00
}
static int blocking_ast_dsql_cache(void* ast_object)
{
/**************************************
*
* b l o c k i n g _ a s t _ d s q l _ c a c h e
*
**************************************
*
* Functional description
2008-03-02 08:35:37 +01:00
* Someone is trying to drop an item from the DSQL cache.
* Mark the symbol as obsolete and release the lock.
*
**************************************/
DSqlCacheItem* const item = static_cast<DSqlCacheItem*>(ast_object);
try
{
Database* const dbb = item->lock->lck_dbb;
AsyncContextHolder tdbb(dbb, FB_FUNCTION, item->lock);
item->obsolete = true;
item->locked = false;
LCK_release(tdbb, item->lock);
}
2010-06-12 18:32:46 +02:00
catch (const Exception&)
{} // no-op
return 0;
}
static DSqlCacheItem* get_dsql_cache_item(thread_db* tdbb, int type, const QualifiedName& name)
{
Database* dbb = tdbb->getDatabase();
Jrd::Attachment* attachment = tdbb->getAttachment();
2010-06-12 18:32:46 +02:00
string key((char*) &type, sizeof(type));
int len = name.identifier.length();
key.append((char*) &len, sizeof(len));
2010-01-30 10:58:01 +01:00
key.append(name.identifier.c_str(), len);
len = name.package.length();
key.append((char*) &len, sizeof(len));
2010-01-30 10:58:01 +01:00
key.append(name.package.c_str(), len);
DSqlCacheItem* item = attachment->att_dsql_cache.put(key);
if (item)
{
item->obsolete = false;
item->locked = false;
item->lock = FB_NEW_RPT(*dbb->dbb_permanent, key.length())
Lock(tdbb, key.length(), LCK_dsql_cache, item, blocking_ast_dsql_cache);
memcpy(item->lock->lck_key.lck_string, key.c_str(), key.length());
}
else
{
item = attachment->att_dsql_cache.get(key);
}
return item;
}
2003-12-31 06:36:12 +01:00
static int blocking_ast_procedure(void* ast_object)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* b l o c k i n g _ a s t _ p r o c e d u r e
*
**************************************
*
* Functional description
* Someone is trying to drop a proceedure. If there
* are outstanding interests in the existence of
* the relation then just mark as blocking and return.
* Otherwise, mark the procedure block as questionable
* and release the procedure existence lock.
*
**************************************/
jrd_prc* const procedure = static_cast<jrd_prc*>(ast_object);
2001-05-23 15:26:42 +02:00
try
{
Database* const dbb = procedure->prc_existence_lock->lck_dbb;
AsyncContextHolder tdbb(dbb, FB_FUNCTION, procedure->prc_existence_lock);
2001-05-23 15:26:42 +02:00
2011-05-11 03:18:28 +02:00
if (procedure->prc_existence_lock)
LCK_release(tdbb, procedure->prc_existence_lock);
2011-05-11 03:18:28 +02:00
procedure->prc_flags |= PRC_obsolete;
}
2010-06-12 18:32:46 +02:00
catch (const Exception&)
{} // no-op
2001-05-23 15:26:42 +02:00
2003-05-19 08:57:08 +02:00
return 0;
2001-05-23 15:26:42 +02:00
}
static int blocking_ast_relation(void* ast_object)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* b l o c k i n g _ a s t _ r e l a t i o n
*
**************************************
*
* Functional description
* Someone is trying to drop a relation. If there
* are outstanding interests in the existence of
* the relation then just mark as blocking and return.
* Otherwise, mark the relation block as questionable
* and release the relation existence lock.
*
**************************************/
jrd_rel* const relation = static_cast<jrd_rel*>(ast_object);
2001-05-23 15:26:42 +02:00
try
{
Database* const dbb = relation->rel_existence_lock->lck_dbb;
AsyncContextHolder tdbb(dbb, FB_FUNCTION, relation->rel_existence_lock);
2001-05-23 15:26:42 +02:00
2011-05-11 03:18:28 +02:00
if (relation->rel_use_count)
relation->rel_flags |= REL_blocking;
2009-11-07 12:58:54 +01:00
else
{
relation->rel_flags &= ~REL_blocking;
relation->rel_flags |= REL_check_existence;
if (relation->rel_existence_lock)
LCK_release(tdbb, relation->rel_existence_lock);
}
2001-05-23 15:26:42 +02:00
}
2010-06-12 18:32:46 +02:00
catch (const Exception&)
{} // no-op
2001-05-23 15:26:42 +02:00
2005-05-28 00:45:31 +02:00
return 0;
2001-05-23 15:26:42 +02:00
}
static int partners_ast_relation(void* ast_object)
{
jrd_rel* const relation = static_cast<jrd_rel*>(ast_object);
try
{
Database* const dbb = relation->rel_partners_lock->lck_dbb;
AsyncContextHolder tdbb(dbb, FB_FUNCTION, relation->rel_partners_lock);
LCK_release(tdbb, relation->rel_partners_lock);
relation->rel_flags |= REL_check_partners;
}
2010-06-12 18:32:46 +02:00
catch (const Exception&)
{} // no-op
return 0;
}
static int rescan_ast_relation(void* ast_object)
{
jrd_rel* const relation = static_cast<jrd_rel*>(ast_object);
try
{
Database* const dbb = relation->rel_rescan_lock->lck_dbb;
AsyncContextHolder tdbb(dbb, FB_FUNCTION, relation->rel_rescan_lock);
LCK_release(tdbb, relation->rel_rescan_lock);
relation->rel_flags &= ~REL_scanned;
}
catch (const Firebird::Exception&)
{} // no-op
return 0;
}
static ULONG get_rel_flags_from_FLAGS(USHORT flags)
{
/**************************************
*
* g e t _ r e l _ f l a g s _ f r o m _ F L A G S
*
**************************************
*
* Functional description
* Get rel_flags from RDB$FLAGS
*
**************************************/
ULONG ret = 0;
if (flags & REL_sql) {
ret |= REL_sql_relation;
}
return ret;
}
ULONG MET_get_rel_flags_from_TYPE(USHORT type)
{
/**************************************
*
* M E T _g e t _ r e l _ f l a g s _ f r o m _ T Y P E
*
**************************************
*
* Functional description
* Get rel_flags from RDB$RELATION_TYPE
*
**************************************/
ULONG ret = 0;
switch (type)
{
case rel_persistent:
break;
case rel_external:
break;
case rel_view:
ret |= REL_jrd_view;
break;
case rel_virtual:
ret |= REL_virtual;
break;
case rel_global_temp_preserve:
ret |= REL_temp_conn;
break;
case rel_global_temp_delete:
ret |= REL_temp_tran;
break;
default:
fb_assert(false);
}
return ret;
}
static void get_trigger(thread_db* tdbb, jrd_rel* relation,
bid* blob_id, bid* debug_blob_id, trig_vec** ptr,
const TEXT* name, FB_UINT64 type,
bool sys_trigger, USHORT flags,
2010-06-12 18:32:46 +02:00
const MetaName& engine, const string& entryPoint,
const bid* body)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ t r i g g e r
*
**************************************
*
* Functional description
* Get trigger.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
if (blob_id->isEmpty() && (engine.isEmpty() || entryPoint.isEmpty()))
2001-05-23 15:26:42 +02:00
return;
blb* blrBlob = NULL;
if (!blob_id->isEmpty())
blrBlob = blb::open(tdbb, attachment->getSysTransaction(), blob_id);
save_trigger_data(tdbb, ptr, relation, NULL, blrBlob, debug_blob_id,
name, type, sys_trigger, flags, engine, entryPoint, body);
2001-05-23 15:26:42 +02:00
}
2006-04-05 18:34:18 +02:00
static bool get_type(thread_db* tdbb, USHORT* id, const UCHAR* name, const TEXT* field)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ t y p e
*
**************************************
*
* Functional description
* Resoved a symbolic name in RDB$TYPES. Returned the value
* defined for the name in (*id). Don't touch (*id) if you
* don't find the name.
*
* Return (1) if found, (0) otherwise.
*
**************************************/
UCHAR buffer[MAX_SQL_IDENTIFIER_SIZE]; // BASED ON RDB$TYPE_NAME
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2003-11-04 00:59:24 +01:00
fb_assert(id != NULL);
fb_assert(name != NULL);
fb_assert(field != NULL);
2001-05-23 15:26:42 +02:00
2009-11-21 04:23:47 +01:00
// Force key to uppercase, following C locale rules for uppercase
UCHAR* p;
2001-05-23 15:26:42 +02:00
for (p = buffer; *name && p < buffer + sizeof(buffer) - 1; p++, name++)
{
*p = UPPER7(*name);
}
*p = 0;
2009-11-21 04:23:47 +01:00
// Try for exact name match
2001-05-23 15:26:42 +02:00
bool found = false;
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2010-04-23 03:59:21 +02:00
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
FIRST 1 T IN RDB$TYPES WITH
2002-12-22 14:08:50 +01:00
T.RDB$FIELD_NAME EQ field AND
T.RDB$TYPE_NAME EQ buffer
2010-04-23 03:59:21 +02:00
{
2008-12-18 11:57:12 +01:00
found = true;
2001-05-23 15:26:42 +02:00
*id = T.RDB$TYPE;
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return found;
}
static void lookup_view_contexts( thread_db* tdbb, jrd_rel* view)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o o k u p _ v i e w _ c o n t e x t s
*
**************************************
*
* Functional description
2005-05-28 00:45:31 +02:00
* Lookup view contexts and store in a linked
2001-05-23 15:26:42 +02:00
* list on the relation block.
2005-05-28 00:45:31 +02:00
*
2001-05-23 15:26:42 +02:00
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
Database* dbb = tdbb->getDatabase();
AutoCacheRequest request(tdbb, irq_view_context, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
V IN RDB$VIEW_RELATIONS WITH
V.RDB$VIEW_NAME EQ view->rel_name.c_str()
2005-05-28 00:45:31 +02:00
SORTED BY V.RDB$VIEW_CONTEXT
{
// trim trailing spaces
2008-12-18 11:57:12 +01:00
fb_utils::exact_name_limit(V.RDB$CONTEXT_NAME, sizeof(V.RDB$CONTEXT_NAME));
2008-05-13 03:29:38 +02:00
ViewContext* view_context = FB_NEW(*dbb->dbb_permanent)
ViewContext(*dbb->dbb_permanent,
2009-12-17 16:07:02 +01:00
V.RDB$CONTEXT_NAME, V.RDB$RELATION_NAME, V.RDB$VIEW_CONTEXT,
(V.RDB$CONTEXT_TYPE.NULL ? VCT_TABLE : ViewContextType(V.RDB$CONTEXT_TYPE)));
2005-05-28 00:45:31 +02:00
view->rel_view_contexts.add(view_context);
}
END_FOR
2001-05-23 15:26:42 +02:00
}
2009-11-21 06:20:18 +01:00
static void make_relation_scope_name(const TEXT* rel_name, const USHORT rel_flags,
2010-06-12 18:32:46 +02:00
string& str)
2006-05-24 22:43:07 +02:00
{
/**************************************
*
* m a k e _ r e l a t i o n _ s c o p e _ n a m e
*
**************************************
*
* Functional description
* Make string with relation name and type
2006-05-24 22:43:07 +02:00
* of its temporary scope
*
**************************************/
const char *scope = NULL;
if (rel_flags & REL_temp_conn)
scope = REL_SCOPE_GTT_PRESERVE;
else if (rel_flags & REL_temp_tran)
scope = REL_SCOPE_GTT_DELETE;
else
scope = REL_SCOPE_PERSISTENT;
str.printf(scope, rel_name);
}
2010-11-07 21:26:11 +01:00
// Parses default BLR for a field.
static ValueExprNode* parse_field_default_blr(thread_db* tdbb, bid* blob_id)
2008-01-16 10:29:37 +01:00
{
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2008-01-16 10:29:37 +01:00
2010-11-07 21:26:11 +01:00
CompilerScratch* csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), 5);
2008-01-16 10:29:37 +01:00
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), blob_id);
ULONG length = blob->blb_length + 10;
2010-06-12 18:32:46 +02:00
HalfStaticArray<UCHAR, 512> temp;
2008-01-16 10:29:37 +01:00
length = blob->BLB_get_data(tdbb, temp.getBuffer(length), length);
2008-01-16 10:29:37 +01:00
DmlNode* node = PAR_blr(tdbb, NULL, temp.begin(), length, NULL, &csb, NULL, false, 0);
2008-01-16 10:29:37 +01:00
csb->csb_blr_reader = BlrReader();
2008-01-16 10:29:37 +01:00
delete csb;
return static_cast<ValueExprNode*>(node);
2008-01-16 10:29:37 +01:00
}
2010-11-07 21:26:11 +01:00
// Parses validation BLR for a field.
2013-03-31 20:23:54 +02:00
static BoolExprNode* parse_field_validation_blr(thread_db* tdbb, bid* blob_id, const MetaName name)
2010-11-07 21:26:11 +01:00
{
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
CompilerScratch* csb = CompilerScratch::newCsb(*tdbb->getDefaultPool(), 5, name);
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), blob_id);
ULONG length = blob->blb_length + 10;
2010-11-07 21:26:11 +01:00
HalfStaticArray<UCHAR, 512> temp;
length = blob->BLB_get_data(tdbb, temp.getBuffer(length), length);
2010-11-07 21:26:11 +01:00
2013-03-31 20:23:54 +02:00
BoolExprNode* expr = PAR_validation_blr(tdbb, NULL,
temp.begin(), length, NULL, &csb, 0);
2010-11-07 21:26:11 +01:00
csb->csb_blr_reader = BlrReader();
delete csb;
2013-03-31 20:23:54 +02:00
return expr;
2010-11-07 21:26:11 +01:00
}
2010-06-12 18:32:46 +02:00
void MET_release_trigger(thread_db* tdbb, trig_vec** vector_ptr, const MetaName& name)
{
/***********************************************
*
* M E T _ r e l e a s e _ t r i g g e r
*
***********************************************
*
* Functional description
* Release a specified trigger.
* If trigger are still active let someone
* else do the work.
*
**************************************/
2006-11-11 01:52:31 +01:00
if (!*vector_ptr)
return;
2006-11-11 01:52:31 +01:00
trig_vec& vector = **vector_ptr;
2006-11-11 01:52:31 +01:00
SET_TDBB(tdbb);
2006-11-11 01:52:31 +01:00
for (size_t i = 0; i < vector.getCount(); ++i)
{
2006-11-11 01:52:31 +01:00
if (vector[i].name == name)
{
JrdStatement* stmt = vector[i].statement;
if (stmt)
2006-11-11 01:52:31 +01:00
{
if (stmt->isActive())
2006-11-11 01:52:31 +01:00
break;
stmt->release(tdbb);
2006-11-11 01:52:31 +01:00
}
vector.remove(i);
break;
}
}
}
void MET_release_triggers( thread_db* tdbb, trig_vec** vector_ptr)
2001-05-23 15:26:42 +02:00
{
/***********************************************
*
* M E T _ r e l e a s e _ t r i g g e r s
*
***********************************************
*
* Functional description
* Release a possibly null vector of triggers.
* If triggers are still active let someone
* else do the work.
*
**************************************/
2003-12-22 11:00:59 +01:00
trig_vec* vector = *vector_ptr;
2005-05-28 00:45:31 +02:00
if (!vector)
2004-04-19 17:29:29 +02:00
{
2001-05-23 15:26:42 +02:00
return;
}
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
*vector_ptr = NULL;
2009-06-27 08:23:36 +02:00
for (size_t i = 0; i < vector->getCount(); i++)
2001-05-23 15:26:42 +02:00
{
JrdStatement* stmt = (*vector)[i].statement;
if (stmt && stmt->isActive())
2001-05-23 15:26:42 +02:00
return;
}
2009-06-27 08:23:36 +02:00
for (size_t i = 0; i < vector->getCount(); i++)
2001-05-23 15:26:42 +02:00
{
JrdStatement* stmt = (*vector)[i].statement;
if (stmt)
stmt->release(tdbb);
delete (*vector)[i].extTrigger;
2001-05-23 15:26:42 +02:00
}
2001-12-24 03:51:06 +01:00
delete vector;
2001-05-23 15:26:42 +02:00
}
2008-12-18 11:57:12 +01:00
static bool resolve_charset_and_collation(thread_db* tdbb,
USHORT* id,
const UCHAR* charset,
const UCHAR* collation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* r e s o l v e _ c h a r s e t _ a n d _ c o l l a t i o n
*
**************************************
*
* Functional description
* Given ASCII7 name of charset & collation
* resolve the specification to a Character set id.
* This character set id is also the id of the text_object
* that implements the C locale for the Character set.
*
* Inputs:
* (charset)
* ASCII7z name of character set.
* NULL (implying unspecified) means use the character set
* for defined for (collation).
*
* (collation)
* ASCII7z name of collation.
* NULL means use the default collation for (charset).
*
* Outputs:
2005-05-28 00:45:31 +02:00
* (*id)
* Set to character set specified by this name (low byte)
* Set to collation specified by this name (high byte).
2001-05-23 15:26:42 +02:00
*
* Return:
* true if no errors (and *id is set).
* false if either name not found.
2001-05-23 15:26:42 +02:00
* or if names found, but the collation isn't for the specified
* character set.
*
**************************************/
bool found = false;
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
2003-11-04 00:59:24 +01:00
fb_assert(id != NULL);
2001-05-23 15:26:42 +02:00
AutoRequest handle;
2001-05-23 15:26:42 +02:00
2010-04-23 03:59:21 +02:00
if (!collation)
2001-05-23 15:26:42 +02:00
{
if (charset == NULL)
charset = (const UCHAR*) DEFAULT_CHARACTER_SET_NAME;
2001-05-23 15:26:42 +02:00
2011-05-09 12:15:19 +02:00
if (attachment->att_charset_ids.get((const TEXT*) charset, *id))
return true;
2006-04-05 18:34:18 +02:00
USHORT charset_id = 0;
if (get_type(tdbb, &charset_id, charset, "RDB$CHARACTER_SET_NAME"))
2001-05-23 15:26:42 +02:00
{
2011-05-09 12:15:19 +02:00
attachment->att_charset_ids.put((const TEXT*) charset, charset_id);
2001-05-23 15:26:42 +02:00
*id = charset_id;
return true;
2001-05-23 15:26:42 +02:00
}
2009-11-21 04:23:47 +01:00
// Charset name not found in the alias table - before giving up
// try the character_set table
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE handle)
FIRST 1 CS IN RDB$CHARACTER_SETS
2002-12-22 14:08:50 +01:00
WITH CS.RDB$CHARACTER_SET_NAME EQ charset
2010-04-23 03:59:21 +02:00
{
found = true;
2011-05-09 12:15:19 +02:00
attachment->att_charset_ids.put((const TEXT*) charset, CS.RDB$CHARACTER_SET_ID);
2001-05-23 15:26:42 +02:00
*id = CS.RDB$CHARACTER_SET_ID;
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return found;
}
2008-01-16 10:29:37 +01:00
2010-04-23 03:59:21 +02:00
if (!charset)
2001-05-23 15:26:42 +02:00
{
FOR(REQUEST_HANDLE handle)
FIRST 1 COL IN RDB$COLLATIONS
2002-12-22 14:08:50 +01:00
WITH COL.RDB$COLLATION_NAME EQ collation
2010-04-23 03:59:21 +02:00
{
2008-12-18 11:57:12 +01:00
found = true;
2005-05-28 00:45:31 +02:00
*id = COL.RDB$CHARACTER_SET_ID | (COL.RDB$COLLATION_ID << 8);
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return found;
}
FOR(REQUEST_HANDLE handle)
FIRST 1 CS IN RDB$CHARACTER_SETS CROSS
COL IN RDB$COLLATIONS OVER RDB$CHARACTER_SET_ID CROSS
AL1 IN RDB$TYPES
WITH AL1.RDB$FIELD_NAME EQ "RDB$CHARACTER_SET_NAME"
2002-12-22 14:08:50 +01:00
AND AL1.RDB$TYPE_NAME EQ charset
AND COL.RDB$COLLATION_NAME EQ collation
2005-05-28 00:45:31 +02:00
AND AL1.RDB$TYPE EQ CS.RDB$CHARACTER_SET_ID
2010-04-23 03:59:21 +02:00
{
found = true;
2011-05-09 12:15:19 +02:00
attachment->att_charset_ids.put((const TEXT*) charset, CS.RDB$CHARACTER_SET_ID);
2005-05-28 00:45:31 +02:00
*id = CS.RDB$CHARACTER_SET_ID | (COL.RDB$COLLATION_ID << 8);
2010-04-23 03:59:21 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return found;
}
static void save_trigger_data(thread_db* tdbb, trig_vec** ptr, jrd_rel* relation,
JrdStatement* statement, blb* blrBlob, bid* dbgBlobID,
const TEXT* name, FB_UINT64 type,
bool sys_trigger, USHORT flags,
2010-06-12 18:32:46 +02:00
const MetaName& engine, const string& entryPoint,
const bid* body)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* s a v e _ t r i g g e r _ d a t a
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Save trigger data to passed vector
2001-05-23 15:26:42 +02:00
*
**************************************/
Jrd::Attachment* attachment = tdbb->getAttachment();
trig_vec* vector = *ptr;
2001-12-24 03:51:06 +01:00
if (!vector)
{
2011-06-03 19:39:22 +02:00
MemoryPool* pool = relation ? relation->rel_pool : attachment->att_pool;
vector = FB_NEW(*pool) trig_vec(*pool);
2001-12-24 03:51:06 +01:00
*ptr = vector;
}
2001-05-23 15:26:42 +02:00
2004-04-19 17:29:29 +02:00
Trigger& t = vector->add();
2005-05-28 00:45:31 +02:00
if (blrBlob)
2004-04-19 17:29:29 +02:00
{
const ULONG length = blrBlob->blb_length + 10;
2004-05-12 21:39:17 +02:00
UCHAR* ptr2 = t.blr.getBuffer(length);
t.blr.resize(blrBlob->BLB_get_data(tdbb, ptr2, length));
2003-12-03 09:19:24 +01:00
}
if (dbgBlobID)
t.dbg_blob_id = *dbgBlobID;
if (name)
2004-04-19 17:29:29 +02:00
t.name = name;
if (body)
{
blb* bodyBlob = blb::open(tdbb, attachment->getSysTransaction(), body);
HalfStaticArray<char, 512> temp;
ULONG length = bodyBlob->BLB_get_data(tdbb, (UCHAR*) temp.getBuffer(bodyBlob->blb_length),
bodyBlob->blb_length);
t.extBody.assign(temp.begin(), length);
2003-12-03 09:19:24 +01:00
}
t.type = type;
t.flags = flags;
t.compile_in_progress = false;
t.sys_trigger = sys_trigger;
t.statement = statement;
t.relation = relation;
t.engine = engine;
t.entryPoint = entryPoint;
2001-05-23 15:26:42 +02:00
}
2010-06-12 18:32:46 +02:00
const Trigger* findTrigger(trig_vec* triggers, const MetaName& trig_name)
2006-05-22 00:07:35 +02:00
{
if (triggers)
{
2008-12-18 11:57:12 +01:00
for (trig_vec::iterator t = triggers->begin(); t != triggers->end(); ++t)
2006-05-22 00:07:35 +02:00
{
if (t->name.compare(trig_name) == 0)
return &(*t);
}
}
return NULL;
}
void scan_partners(thread_db* tdbb, jrd_rel* relation)
{
/**************************************
*
* s c a n _ p a r t n e r s
*
**************************************
*
* Functional description
2010-01-21 03:51:32 +01:00
* Scan of foreign references on other relations' primary keys and
* scan of primary dependencies on relation's primary key.
*
**************************************/
Jrd::Attachment* attachment = tdbb->getAttachment();
AutoCacheRequest request(tdbb, irq_foreign1, IRQ_REQUESTS);
frgn* references = &relation->rel_foreign_refs;
int index_number = 0;
if (references->frgn_reference_ids)
{
delete references->frgn_reference_ids;
references->frgn_reference_ids = NULL;
}
if (references->frgn_relations)
{
delete references->frgn_relations;
references->frgn_relations = NULL;
}
if (references->frgn_indexes)
{
delete references->frgn_indexes;
references->frgn_indexes = NULL;
}
FOR(REQUEST_HANDLE request)
IDX IN RDB$INDICES CROSS
RC IN RDB$RELATION_CONSTRAINTS
OVER RDB$INDEX_NAME CROSS
IND IN RDB$INDICES WITH
RC.RDB$CONSTRAINT_TYPE EQ FOREIGN_KEY AND
IDX.RDB$RELATION_NAME EQ relation->rel_name.c_str() AND
IND.RDB$INDEX_NAME EQ IDX.RDB$FOREIGN_KEY AND
IND.RDB$UNIQUE_FLAG = 1
{
const jrd_rel* partner_relation = MET_lookup_relation(tdbb, IND.RDB$RELATION_NAME);
if (partner_relation && !IDX.RDB$INDEX_INACTIVE && !IND.RDB$INDEX_INACTIVE)
{
// This seems a good candidate for vcl.
references->frgn_reference_ids =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, references->frgn_reference_ids,
index_number + 1);
(*references->frgn_reference_ids)[index_number] = IDX.RDB$INDEX_ID - 1;
references->frgn_relations =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, references->frgn_relations,
index_number + 1);
(*references->frgn_relations)[index_number] = partner_relation->rel_id;
references->frgn_indexes =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, references->frgn_indexes,
index_number + 1);
(*references->frgn_indexes)[index_number] = IND.RDB$INDEX_ID - 1;
index_number++;
}
}
END_FOR
// Prepare for rescan of primary dependencies on relation's primary key and stale vectors.
request.reset(tdbb, irq_foreign2, IRQ_REQUESTS);
prim* dependencies = &relation->rel_primary_dpnds;
index_number = 0;
if (dependencies->prim_reference_ids)
{
delete dependencies->prim_reference_ids;
dependencies->prim_reference_ids = NULL;
}
if (dependencies->prim_relations)
{
delete dependencies->prim_relations;
dependencies->prim_relations = NULL;
}
if (dependencies->prim_indexes)
{
delete dependencies->prim_indexes;
dependencies->prim_indexes = NULL;
}
FOR(REQUEST_HANDLE request)
IDX IN RDB$INDICES CROSS
IND IN RDB$INDICES WITH
IDX.RDB$UNIQUE_FLAG = 1 AND
IDX.RDB$RELATION_NAME EQ relation->rel_name.c_str() AND
IND.RDB$FOREIGN_KEY EQ IDX.RDB$INDEX_NAME
{
const jrd_rel* partner_relation = MET_lookup_relation(tdbb, IND.RDB$RELATION_NAME);
if (partner_relation && !IDX.RDB$INDEX_INACTIVE && !IND.RDB$INDEX_INACTIVE)
{
dependencies->prim_reference_ids =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, dependencies->prim_reference_ids,
index_number + 1);
(*dependencies->prim_reference_ids)[index_number] = IDX.RDB$INDEX_ID - 1;
dependencies->prim_relations =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, dependencies->prim_relations,
index_number + 1);
(*dependencies->prim_relations)[index_number] = partner_relation->rel_id;
dependencies->prim_indexes =
2011-05-09 12:15:19 +02:00
vec<int>::newVector(*relation->rel_pool, dependencies->prim_indexes,
index_number + 1);
(*dependencies->prim_indexes)[index_number] = IND.RDB$INDEX_ID - 1;
index_number++;
}
}
END_FOR
LCK_lock(tdbb, relation->rel_partners_lock, LCK_SR, LCK_WAIT);
relation->rel_flags &= ~REL_check_partners;
}
static void store_dependencies(thread_db* tdbb,
CompilerScratch* csb,
const jrd_rel* dep_rel,
2010-06-12 18:32:46 +02:00
const MetaName& object_name,
int dependency_type,
jrd_tra* transaction)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* s t o r e _ d e p e n d e n c i e s
*
**************************************
*
* Functional description
* Store records in RDB$DEPENDENCIES
2005-05-28 00:45:31 +02:00
* corresponding to the objects found during
2001-05-23 15:26:42 +02:00
* compilation of blr for a trigger, view, etc.
*
**************************************/
2010-06-12 18:32:46 +02:00
MetaName name;
2001-05-23 15:26:42 +02:00
SET_TDBB(tdbb);
2006-05-25 10:40:23 +02:00
const Trigger* t = 0;
const bool checkTableScope =
2006-05-22 00:07:35 +02:00
(dependency_type == obj_computed) ||
2008-12-18 11:57:12 +01:00
(dependency_type == obj_trigger) && (dep_rel != 0) &&
(
2006-05-22 00:07:35 +02:00
(t = findTrigger(dep_rel->rel_pre_erase, object_name)) ||
(t = findTrigger(dep_rel->rel_pre_modify, object_name)) ||
(t = findTrigger(dep_rel->rel_pre_store, object_name)) ||
(t = findTrigger(dep_rel->rel_post_erase, object_name)) ||
(t = findTrigger(dep_rel->rel_post_modify, object_name)) ||
(t = findTrigger(dep_rel->rel_post_store, object_name))
2006-05-22 00:07:35 +02:00
) && t && (t->sys_trigger);
while (csb->csb_dependencies.hasData())
2001-05-23 15:26:42 +02:00
{
CompilerScratch::Dependency dependency = csb->csb_dependencies.pop();
if (!dependency.relation && !dependency.function && !dependency.procedure &&
!dependency.name && !dependency.number)
{
2001-05-23 15:26:42 +02:00
continue;
}
2005-05-28 00:45:31 +02:00
int dpdo_type = dependency.objType;
jrd_rel* relation = NULL;
const jrd_prc* procedure = NULL;
2010-06-22 02:53:35 +02:00
const MetaName* dpdo_name = NULL;
MetaName packageName;
SubtypeInfo info;
2009-01-20 09:33:59 +01:00
switch (dpdo_type)
{
case obj_relation:
relation = dependency.relation;
dpdo_name = &relation->rel_name;
2006-05-22 00:07:35 +02:00
2008-12-18 11:57:12 +01:00
fb_assert(dep_rel || !checkTableScope);
2006-05-22 00:07:35 +02:00
if (checkTableScope &&
2006-06-01 06:45:48 +02:00
( (dep_rel->rel_flags & (REL_temp_tran | REL_temp_conn)) !=
(relation->rel_flags & (REL_temp_tran | REL_temp_conn)) ))
2006-05-22 00:07:35 +02:00
{
if ( !( // master is ON COMMIT PRESERVE, detail is ON COMMIT DELETE
(dep_rel->rel_flags & REL_temp_tran) && (relation->rel_flags & REL_temp_conn) ||
// computed field of a view
(dependency_type == obj_computed) && (dep_rel->rel_view_rse != NULL)
))
2006-05-22 00:07:35 +02:00
{
2010-06-12 18:32:46 +02:00
string sMaster, sChild;
2006-05-24 22:43:07 +02:00
make_relation_scope_name(relation->rel_name.c_str(),
2006-05-24 22:43:07 +02:00
relation->rel_flags, sMaster);
make_relation_scope_name(dep_rel->rel_name.c_str(),
2006-05-24 22:43:07 +02:00
dep_rel->rel_flags, sChild);
2006-05-22 00:07:35 +02:00
ERR_post(Arg::Gds(isc_no_meta_update) <<
Arg::Gds(isc_met_wrong_gtt_scope) << Arg::Str(sChild) <<
Arg::Str(sMaster));
2006-05-22 00:07:35 +02:00
}
}
MET_scan_relation(tdbb, relation);
if (relation->rel_view_rse) {
dpdo_type = obj_view;
}
2008-12-18 11:57:12 +01:00
break;
case obj_procedure:
procedure = dependency.procedure;
dpdo_name = &procedure->getName().identifier;
packageName = procedure->getName().package;
2008-12-18 11:57:12 +01:00
break;
case obj_collation:
{
const USHORT number = dependency.number;
MET_get_char_coll_subtype_info(tdbb, number, &info);
dpdo_name = &info.collationName;
}
2006-08-13 08:46:16 +02:00
break;
2008-12-18 11:57:12 +01:00
case obj_exception:
{
const SLONG number = dependency.number;
MET_lookup_exception(tdbb, number, name, NULL);
dpdo_name = &name;
}
break;
2008-12-18 11:57:12 +01:00
case obj_field:
dpdo_name = dependency.name;
break;
2008-12-18 11:57:12 +01:00
// CVC: Here I'm going to track those pesky things named generators and UDFs.
case obj_generator:
{
const SLONG number = dependency.number;
MET_lookup_generator_id (tdbb, number, name);
dpdo_name = &name;
}
break;
2008-12-18 11:57:12 +01:00
case obj_udf:
{
const Function* const udf = dependency.function;
dpdo_name = &udf->getName().identifier;
packageName = udf->getName().package;
}
break;
case obj_index:
name = *dependency.name;
dpdo_name = &name;
break;
}
2002-07-01 18:59:09 +02:00
2010-06-12 18:32:46 +02:00
MetaName field_name;
if (dependency.subNumber || dependency.subName)
2001-05-23 15:26:42 +02:00
{
if (dependency.subNumber)
2001-05-23 15:26:42 +02:00
{
const SSHORT fld_id = (SSHORT) dependency.subNumber;
2001-05-23 15:26:42 +02:00
if (relation)
{
const jrd_fld* field = MET_get_field(relation, fld_id);
if (field)
2001-05-23 15:26:42 +02:00
field_name = field->fld_name;
}
else if (procedure)
{
const Parameter* param = procedure->getOutputFields()[fld_id];
// CVC: Setting the field var here didn't make sense alone,
// so I thought the missing code was to try to extract
// the field name that's in this case an output var from a proc.
if (param)
field_name = param->prm_name;
2001-05-23 15:26:42 +02:00
}
}
else
field_name = *dependency.subName;
2001-05-23 15:26:42 +02:00
}
if (field_name.hasData())
2001-05-23 15:26:42 +02:00
{
AutoCacheRequest request(tdbb, irq_c_deps_f, IRQ_REQUESTS);
bool found = false;
fb_assert(dpdo_name);
FOR(REQUEST_HANDLE request TRANSACTION_HANDLE transaction) X IN RDB$DEPENDENCIES WITH
X.RDB$DEPENDENT_NAME = object_name.c_str() AND
X.RDB$DEPENDED_ON_NAME = dpdo_name->c_str() AND
2001-05-23 15:26:42 +02:00
X.RDB$DEPENDED_ON_TYPE = dpdo_type AND
X.RDB$FIELD_NAME = field_name.c_str() AND
2001-05-23 15:26:42 +02:00
X.RDB$DEPENDENT_TYPE = dependency_type
{
found = true;
}
END_FOR
2006-07-09 07:08:23 +02:00
if (found)
2001-05-23 15:26:42 +02:00
continue;
}
else
{
AutoCacheRequest request(tdbb, irq_c_deps, IRQ_REQUESTS);
bool found = false;
FOR(REQUEST_HANDLE request TRANSACTION_HANDLE transaction) X IN RDB$DEPENDENCIES WITH
X.RDB$DEPENDENT_NAME = object_name.c_str() AND
X.RDB$DEPENDED_ON_NAME = dpdo_name->c_str() AND
2001-05-23 15:26:42 +02:00
X.RDB$DEPENDED_ON_TYPE = dpdo_type AND
X.RDB$FIELD_NAME MISSING AND
X.RDB$DEPENDENT_TYPE = dependency_type AND
X.RDB$PACKAGE_NAME EQUIV NULLIF(packageName.c_str(), '')
{
found = true;
}
END_FOR
2001-05-23 15:26:42 +02:00
if (found)
2001-05-23 15:26:42 +02:00
continue;
}
AutoCacheRequest request(tdbb, irq_s_deps, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
fb_assert(dpdo_name);
STORE(REQUEST_HANDLE request TRANSACTION_HANDLE transaction) DEP IN RDB$DEPENDENCIES
{
strcpy(DEP.RDB$DEPENDENT_NAME, object_name.c_str());
2001-05-23 15:26:42 +02:00
DEP.RDB$DEPENDED_ON_TYPE = dpdo_type;
strcpy(DEP.RDB$DEPENDED_ON_NAME, dpdo_name->c_str());
if (field_name.hasData())
2001-05-23 15:26:42 +02:00
{
DEP.RDB$FIELD_NAME.NULL = FALSE;
strcpy(DEP.RDB$FIELD_NAME, field_name.c_str());
2001-05-23 15:26:42 +02:00
}
else
2001-05-23 15:26:42 +02:00
DEP.RDB$FIELD_NAME.NULL = TRUE;
if (packageName.hasData())
{
DEP.RDB$PACKAGE_NAME.NULL = FALSE;
strcpy(DEP.RDB$PACKAGE_NAME, packageName.c_str());
}
else
DEP.RDB$PACKAGE_NAME.NULL = TRUE;
2001-05-23 15:26:42 +02:00
DEP.RDB$DEPENDENT_TYPE = dependency_type;
}
END_STORE
2001-05-23 15:26:42 +02:00
}
}
2010-06-12 18:32:46 +02:00
static bool verify_TRG_ignore_perm(thread_db* tdbb, const MetaName& trig_name)
2001-05-23 15:26:42 +02:00
{
/*****************************************************
*
* v e r i f y _ T R G _ i g n o r e _ p e r m
*
*****************************************************
*
* Functional description
* Return true if this trigger can go through without any permission
2001-05-23 15:26:42 +02:00
* checks. Currently, the only class of triggers that can go
2005-05-28 00:45:31 +02:00
* through without permission checks are
2001-05-23 15:26:42 +02:00
* (a) two system triggers (RDB$TRIGGERS_34 and RDB$TRIGGERS_35)
2005-05-28 00:45:31 +02:00
* (b) those defined for referential integrity actions such as,
2001-05-23 15:26:42 +02:00
* set null, set default, and cascade.
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
2001-05-23 15:26:42 +02:00
// See if this is a system trigger, with the flag set as TRG_ignore_perm
if (INI_get_trig_flags(trig_name.c_str()) & TRG_ignore_perm)
2001-05-23 15:26:42 +02:00
{
return true;
2001-05-23 15:26:42 +02:00
}
// See if this is a RI trigger
AutoCacheRequest request(tdbb, irq_c_trg_perm, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request)
CHK IN RDB$CHECK_CONSTRAINTS CROSS
REF IN RDB$REF_CONSTRAINTS WITH
CHK.RDB$TRIGGER_NAME EQ trig_name.c_str() AND
2001-05-23 15:26:42 +02:00
REF.RDB$CONSTRAINT_NAME = CHK.RDB$CONSTRAINT_NAME
{
fb_utils::exact_name_limit(REF.RDB$UPDATE_RULE, sizeof(REF.RDB$UPDATE_RULE));
fb_utils::exact_name_limit(REF.RDB$DELETE_RULE, sizeof(REF.RDB$DELETE_RULE));
2001-05-23 15:26:42 +02:00
if (!strcmp(REF.RDB$UPDATE_RULE, RI_ACTION_CASCADE) ||
!strcmp(REF.RDB$UPDATE_RULE, RI_ACTION_NULL) ||
!strcmp(REF.RDB$UPDATE_RULE, RI_ACTION_DEFAULT) ||
!strcmp(REF.RDB$DELETE_RULE, RI_ACTION_CASCADE) ||
!strcmp(REF.RDB$DELETE_RULE, RI_ACTION_NULL) ||
!strcmp(REF.RDB$DELETE_RULE, RI_ACTION_DEFAULT))
{
return true;
2001-05-23 15:26:42 +02:00
}
return false;
2001-05-23 15:26:42 +02:00
}
END_FOR
2001-05-23 15:26:42 +02:00
return false;
}
2013-11-14 17:16:24 +01:00
int MET_get_linger(Jrd::thread_db* tdbb)
{
/**************************************
*
* M E T _ g e t _ l i n g e r
*
**************************************
*
* Functional description
* Return linger value for current database
*
**************************************/
SET_TDBB(tdbb);
Jrd::Attachment* attachment = tdbb->getAttachment();
int rc = 0;
AutoCacheRequest request(tdbb, irq_linger, IRQ_REQUESTS);
FOR(REQUEST_HANDLE request)
DAT IN RDB$DATABASE
{
if (!DAT.RDB$LINGER.NULL)
{
rc = DAT.RDB$LINGER;
}
}
END_FOR
return rc;
}