2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-10-08 10:42:48 +02:00
|
|
|
* MODULE: exe.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: Statement execution
|
|
|
|
*
|
|
|
|
* 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-06-30 11:58:20 +02:00
|
|
|
*
|
|
|
|
* 2001.6.21 Claudio Valderrama: Allow inserting strings into blob fields.
|
|
|
|
* 2001.6.28 Claudio Valderrama: Move code to cleanup_rpb() as directed
|
|
|
|
* by Ann Harrison and cleanup of new record in store() routine.
|
|
|
|
* 2001.10.11 Claudio Valderrama: Fix SF Bug #436462: From now, we only
|
|
|
|
* count real store, modify and delete operations either in an external
|
|
|
|
* file or in a table. Counting on a view caused up to three operations
|
|
|
|
* being reported instead of one.
|
|
|
|
* 2001.12.03 Claudio Valderrama: new visit to the same issue: views need
|
|
|
|
* to count virtual operations, not real I/O on the underlying tables.
|
2002-09-28 16:04:35 +02:00
|
|
|
* 2002.09.28 Dmitry Yemanov: Reworked internal_info stuff, enhanced
|
|
|
|
* exception handling in SPs/triggers,
|
|
|
|
* implemented ROWS_AFFECTED system variable
|
2002-10-29 04:17:45 +01:00
|
|
|
*
|
2002-10-29 21:20:44 +01:00
|
|
|
* 2002.10.21 Nickolay Samofatov: Added support for explicit pessimistic locks
|
2002-10-29 04:17:45 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
|
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
|
2002-10-29 21:20:44 +01:00
|
|
|
* 2002.10.29 Nickolay Samofatov: Added support for savepoints
|
2002-10-31 06:06:02 +01:00
|
|
|
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
|
2003-11-02 12:55:17 +01:00
|
|
|
* 2003.10.05 Dmitry Yemanov: Added support for explicit cursors in PSQL
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2002-09-18 14:50:13 +02:00
|
|
|
#if TIME_WITH_SYS_TIME
|
|
|
|
# include <sys/time.h>
|
|
|
|
# include <time.h>
|
|
|
|
#else
|
|
|
|
# if HAVE_SYS_TIME_H
|
|
|
|
# include <sys/time.h>
|
|
|
|
# else
|
|
|
|
# include <time.h>
|
|
|
|
# endif
|
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include "../jrd/ibsetjmp.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/val.h"
|
|
|
|
#include "../jrd/exe.h"
|
|
|
|
#include "../jrd/tra.h"
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/btr.h"
|
|
|
|
#include "../jrd/rse.h"
|
|
|
|
#include "../jrd/lck.h"
|
|
|
|
#include "../jrd/intl.h"
|
|
|
|
#include "../jrd/rng.h"
|
|
|
|
#include "../jrd/sbm.h"
|
|
|
|
#include "../jrd/blb.h"
|
|
|
|
#include "../jrd/blr.h"
|
|
|
|
#include "../jrd/all_proto.h"
|
|
|
|
#include "../jrd/bookmark.h"
|
|
|
|
#include "../jrd/blb_proto.h"
|
|
|
|
#include "../jrd/btr_proto.h"
|
|
|
|
#include "../jrd/cmp_proto.h"
|
|
|
|
#include "../jrd/dfw_proto.h"
|
|
|
|
#include "../jrd/dpm_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/evl_proto.h"
|
|
|
|
#include "../jrd/exe_proto.h"
|
|
|
|
#include "../jrd/ext_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/idx_proto.h"
|
2005-05-28 00:45:31 +02:00
|
|
|
#include "../jrd/intl_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/jrd_proto.h"
|
|
|
|
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#include "../jrd/mov_proto.h"
|
|
|
|
#include "../jrd/opt_proto.h"
|
|
|
|
#include "../jrd/par_proto.h"
|
|
|
|
#include "../jrd/rlck_proto.h"
|
2004-03-20 16:33:30 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/rse_proto.h"
|
|
|
|
#include "../jrd/rng_proto.h"
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../jrd/thd.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/tra_proto.h"
|
|
|
|
#include "../jrd/vio_proto.h"
|
|
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
#include "../jrd/execute_statement.h"
|
2003-04-22 10:45:52 +02:00
|
|
|
#include "../dsql/dsql_proto.h"
|
2003-04-25 16:51:04 +02:00
|
|
|
#include "../jrd/rpb_chain.h"
|
2004-03-19 07:14:53 +01:00
|
|
|
|
2003-03-01 20:19:23 +01:00
|
|
|
|
2004-03-20 16:33:30 +01:00
|
|
|
using namespace Jrd;
|
|
|
|
|
2005-06-24 14:56:34 +02:00
|
|
|
// AffectedRows class implementation
|
|
|
|
|
|
|
|
AffectedRows::AffectedRows()
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AffectedRows::clear()
|
|
|
|
{
|
|
|
|
writeFlag = false;
|
|
|
|
fetchedRows = modifiedRows = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AffectedRows::bumpFetched()
|
|
|
|
{
|
|
|
|
fetchedRows++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AffectedRows::bumpModified(bool increment)
|
|
|
|
{
|
|
|
|
if (increment) {
|
|
|
|
modifiedRows++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
writeFlag = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AffectedRows::isReadOnly() const
|
|
|
|
{
|
|
|
|
return !writeFlag;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AffectedRows::hasCursor() const
|
|
|
|
{
|
|
|
|
return (fetchedRows > 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int AffectedRows::getCount() const
|
|
|
|
{
|
|
|
|
return writeFlag ? modifiedRows : fetchedRows;
|
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
// StatusXcp class implementation
|
2003-11-02 12:55:17 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
StatusXcp::StatusXcp()
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
void StatusXcp::clear()
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
status[0] = isc_arg_gds;
|
|
|
|
status[1] = FB_SUCCESS;
|
|
|
|
status[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
void StatusXcp::init(const ISC_STATUS* vector)
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
memcpy(status, vector, sizeof(ISC_STATUS_ARRAY));
|
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
void StatusXcp::copyTo(ISC_STATUS* vector) const
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
memcpy(vector, status, sizeof(ISC_STATUS_ARRAY));
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
bool StatusXcp::success() const
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
2003-11-02 13:05:38 +01:00
|
|
|
return (status[1] == FB_SUCCESS);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
SLONG StatusXcp::as_gdscode() const
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
return status[1];
|
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
SLONG StatusXcp::as_sqlcode() const
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
|
|
|
return gds__sqlcode(status);
|
|
|
|
}
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
static void cleanup_rpb(thread_db*, record_param*);
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* erase(thread_db*, jrd_nod*, SSHORT);
|
|
|
|
static void execute_looper(thread_db*, jrd_req*, jrd_tra*, enum jrd_req::req_s);
|
|
|
|
static void exec_sql(thread_db*, jrd_req*, DSC *);
|
|
|
|
static void execute_procedure(thread_db*, jrd_nod*);
|
2004-03-18 06:56:06 +01:00
|
|
|
static jrd_req* execute_triggers(thread_db*, trig_vec**, Record*, Record*,
|
|
|
|
enum jrd_req::req_ta);
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* looper(thread_db*, jrd_req*, jrd_nod*);
|
|
|
|
static jrd_nod* modify(thread_db*, jrd_nod*, SSHORT);
|
|
|
|
static jrd_nod* receive_msg(thread_db*, jrd_nod*);
|
|
|
|
static void release_blobs(thread_db*, jrd_req*);
|
2004-01-03 11:59:52 +01:00
|
|
|
static void release_proc_save_points(jrd_req*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* seek_rse(thread_db*, jrd_req*, jrd_nod*);
|
2004-03-28 11:10:30 +02:00
|
|
|
static void seek_rsb(thread_db*, jrd_req*, RecordSource*, USHORT, SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* selct(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* send_msg(thread_db*, jrd_nod*);
|
|
|
|
static void set_error(thread_db*, const xcp_repeat*, jrd_nod*);
|
|
|
|
static jrd_nod* stall(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* store(thread_db*, jrd_nod*, SSHORT);
|
2004-03-28 11:10:30 +02:00
|
|
|
static bool test_and_fixup_error(thread_db*, const PsqlException*, jrd_req*);
|
2004-03-11 06:04:26 +01:00
|
|
|
static void trigger_failure(thread_db*, jrd_req*);
|
|
|
|
static void validate(thread_db*, jrd_nod*);
|
2005-01-04 14:09:11 +01:00
|
|
|
inline void verb_cleanup(thread_db*, jrd_tra*);
|
2004-03-18 06:56:06 +01:00
|
|
|
inline void PreModifyEraseTriggers(thread_db*, trig_vec**, SSHORT, record_param*,
|
|
|
|
Record*, jrd_req::req_ta);
|
2005-01-04 14:09:11 +01:00
|
|
|
static void stuff_stack_trace(const jrd_req*);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* find(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* find_dbkey(thread_db*, jrd_nod*);
|
2004-03-18 06:56:06 +01:00
|
|
|
static Lock* implicit_record_lock(jrd_tra*, record_param*);
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* release_bookmark(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* set_bookmark(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* set_index(thread_db*, jrd_nod*);
|
|
|
|
static jrd_nod* stream(thread_db*, jrd_nod*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* macro definitions */
|
|
|
|
|
2004-05-03 23:43:56 +02:00
|
|
|
const char* NULL_STRING = "*** null ***";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#if (defined SUPERSERVER) && (defined WIN_NT || defined SOLARIS_MT)
|
2004-05-03 23:43:56 +02:00
|
|
|
const int MAX_CLONES = 750;
|
|
|
|
#elif defined (HP10) && defined (SUPERSERVER)
|
|
|
|
const int MAX_CLONES = 110;
|
|
|
|
#else
|
|
|
|
const int MAX_CLONES = 1000;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-05-03 23:43:56 +02:00
|
|
|
const int ALL_TRIGS = 0;
|
|
|
|
const int PRE_TRIG = 1;
|
|
|
|
const int POST_TRIG = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-07 15:30:38 +01:00
|
|
|
const size_t MAX_STACK_TRACE = 2048;
|
2004-08-06 17:26:55 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* this constant defines how many records are locked
|
|
|
|
before we check whether record locking has been
|
|
|
|
turned off for a given relation; if we set the
|
|
|
|
constant to a low number, we will do too much
|
|
|
|
locking in the case where record locking is always
|
|
|
|
turned on; too high and we will do too much record
|
|
|
|
locking in the case where someone is only occasionally
|
|
|
|
locking a record */
|
|
|
|
|
2004-05-03 23:43:56 +02:00
|
|
|
const int RECORD_LOCK_CHECK_INTERVAL = 10;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
#ifdef PC_ENGINE
|
2004-03-18 06:56:06 +01:00
|
|
|
// TMN: RAII class for Lock. Unlocks the Lock on destruction.
|
2001-12-28 07:31:38 +01:00
|
|
|
class LCK_RAII_wrapper
|
|
|
|
{
|
|
|
|
LCK_RAII_wrapper() : l(0) {}
|
|
|
|
~LCK_RAII_wrapper() {
|
|
|
|
if (l) {
|
|
|
|
RLCK_unlock_record_implicit(l, 0);
|
|
|
|
}
|
|
|
|
}
|
2004-03-18 06:56:06 +01:00
|
|
|
void assign(Lock* lock) { l = lock; }
|
2001-12-28 07:31:38 +01:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* l;
|
2001-12-28 07:31:38 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
LCK_RAII_wrapper(const LCK_RAII_wrapper&); // no impl.
|
|
|
|
void operator=(const LCK_RAII_wrapper&); // no impl.
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_assignment(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ a s s i g n m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Perform an assignment
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DEV_BLKCHK(node, type_nod);
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2003-01-15 13:08:59 +01:00
|
|
|
/* Get descriptors of receiving and sending fields/parameters, variables, etc. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-10-08 10:42:48 +02:00
|
|
|
const dsc* missing = NULL;
|
2001-12-24 03:51:06 +01:00
|
|
|
if (node->nod_arg[e_asgn_missing]) {
|
2001-05-23 15:26:42 +02:00
|
|
|
missing = EVL_expr(tdbb, node->nod_arg[e_asgn_missing]);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* to = node->nod_arg[e_asgn_to];
|
2001-12-24 03:51:06 +01:00
|
|
|
DSC* to_desc = EVL_assign_to(tdbb, to);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
request->req_flags &= ~req_null;
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
dsc* from_desc = EVL_expr(tdbb, node->nod_arg[e_asgn_from]);
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2004-11-25 01:47:20 +01:00
|
|
|
// NS: If we are assigning to NULL, we finished.
|
|
|
|
// This functionality is currently used to allow calling UDF routines
|
|
|
|
// without assigning resulting value anywhere.
|
|
|
|
if (!to_desc) {
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
SSHORT null = (request->req_flags & req_null) ? -1 : 0;
|
|
|
|
|
|
|
|
if (!null && missing && MOV_compare(missing, from_desc) == 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
null = -1;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If the value is non-missing, move/convert it. Otherwise fill the
|
|
|
|
field with appropriate nulls. */
|
2004-01-13 10:52:19 +01:00
|
|
|
dsc temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!null)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* if necessary and appropriate, use the indicator variable */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (to->nod_type == nod_argument && to->nod_arg[e_arg_indicator])
|
|
|
|
{
|
2004-01-13 10:52:19 +01:00
|
|
|
dsc* indicator = EVL_assign_to(tdbb, to->nod_arg[e_arg_indicator]);
|
2001-12-24 03:51:06 +01:00
|
|
|
temp.dsc_dtype = dtype_short;
|
|
|
|
temp.dsc_length = sizeof(SSHORT);
|
|
|
|
temp.dsc_scale = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
temp.dsc_sub_type = 0;
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
SSHORT len;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((from_desc->dsc_dtype <= dtype_varying) &&
|
|
|
|
(to_desc->dsc_dtype <= dtype_varying) &&
|
|
|
|
(TEXT_LEN(from_desc) > TEXT_LEN(to_desc)))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
len = TEXT_LEN(from_desc);
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
len = 0;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-10-08 10:42:48 +02:00
|
|
|
temp.dsc_address = (UCHAR *) &len;
|
2001-05-23 15:26:42 +02:00
|
|
|
MOV_move(&temp, indicator);
|
|
|
|
|
|
|
|
if (len) {
|
|
|
|
temp = *from_desc;
|
|
|
|
temp.dsc_length = TEXT_LEN(to_desc);
|
2001-12-24 03:51:06 +01:00
|
|
|
if (temp.dsc_dtype == dtype_cstring) {
|
2001-05-23 15:26:42 +02:00
|
|
|
temp.dsc_length += 1;
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else if (temp.dsc_dtype == dtype_varying) {
|
2001-05-23 15:26:42 +02:00
|
|
|
temp.dsc_length += 2;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
from_desc = &temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef VMS
|
|
|
|
if (DTYPE_IS_BLOB(to_desc->dsc_dtype))
|
|
|
|
#else
|
|
|
|
if (DTYPE_IS_BLOB(to_desc->dsc_dtype)
|
|
|
|
&& to_desc->dsc_dtype != dtype_d_float)
|
|
|
|
#endif
|
2002-06-30 11:58:20 +02:00
|
|
|
{
|
|
|
|
/* CVC: This is a case that has hurt me for years and I'm going to solve it.
|
|
|
|
It should be possible to copy a string to a blob, even if the charset is
|
|
|
|
lost as a result of this experimental implementation. */
|
2004-11-01 09:13:44 +01:00
|
|
|
if (from_desc->dsc_dtype <= dtype_varying
|
|
|
|
&& to_desc->dsc_dtype != dtype_array)
|
|
|
|
{
|
2002-06-30 11:58:20 +02:00
|
|
|
BLB_move_from_string(tdbb, from_desc, to_desc, to);
|
2004-11-01 09:13:44 +01:00
|
|
|
}
|
2002-06-30 11:58:20 +02:00
|
|
|
else
|
|
|
|
BLB_move(tdbb, from_desc, to_desc, to);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
else if (!DSC_EQUIV(from_desc, to_desc))
|
|
|
|
MOV_move(from_desc, to_desc);
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
else if (from_desc->dsc_dtype == dtype_short) {
|
2001-05-23 15:26:42 +02:00
|
|
|
*((SSHORT *) to_desc->dsc_address) =
|
|
|
|
*((SSHORT *) from_desc->dsc_address);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
else if (from_desc->dsc_dtype == dtype_long) {
|
2001-05-23 15:26:42 +02:00
|
|
|
*((SLONG *) to_desc->dsc_address) =
|
|
|
|
*((SLONG *) from_desc->dsc_address);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
else if (from_desc->dsc_dtype == dtype_int64) {
|
2001-05-23 15:26:42 +02:00
|
|
|
*((SINT64 *) to_desc->dsc_address) =
|
|
|
|
*((SINT64 *) from_desc->dsc_address);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
else if (((U_IPTR) from_desc->dsc_address & (ALIGNMENT - 1)) ||
|
|
|
|
((U_IPTR) to_desc->dsc_address & (ALIGNMENT - 1)))
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FAST(from_desc->dsc_address, to_desc->dsc_address,
|
|
|
|
from_desc->dsc_length);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FASTER(from_desc->dsc_address, to_desc->dsc_address,
|
|
|
|
from_desc->dsc_length);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
to_desc->dsc_flags &= ~DSC_null;
|
|
|
|
}
|
|
|
|
else if (node->nod_arg[e_asgn_missing2] &&
|
2001-12-24 03:51:06 +01:00
|
|
|
(missing = EVL_expr(tdbb, node->nod_arg[e_asgn_missing2])))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
MOV_move(missing, to_desc);
|
|
|
|
to_desc->dsc_flags |= DSC_null;
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
else
|
|
|
|
{
|
2003-10-08 10:42:48 +02:00
|
|
|
USHORT l = to_desc->dsc_length;
|
|
|
|
UCHAR* p = to_desc->dsc_address;
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (to_desc->dsc_dtype) {
|
|
|
|
case dtype_text:
|
2005-11-08 06:59:06 +01:00
|
|
|
// YYY - not necessarily the right thing to do
|
|
|
|
// for text formats that don't have trailing spaces
|
2001-12-24 03:51:06 +01:00
|
|
|
if (l) {
|
2005-11-08 06:59:06 +01:00
|
|
|
const CHARSET_ID chid = DSC_GET_CHARSET(to_desc);
|
|
|
|
/*
|
|
|
|
CVC: I don't know if we have to check for dynamic-127 charset here.
|
|
|
|
If that is needed, the line above should be replaced by the ccmmented code here.
|
|
|
|
CHARSET_ID chid = INTL_TTYPE(to_desc);
|
|
|
|
if (chid == ttype_dynamic)
|
|
|
|
chid = INTL_charset(tdbb, chid);
|
|
|
|
*/
|
|
|
|
const char pad = chid == ttype_binary ? '\0' : ' ';
|
2001-12-24 03:51:06 +01:00
|
|
|
do {
|
2005-11-08 06:59:06 +01:00
|
|
|
*p++ = pad;
|
2001-12-24 03:51:06 +01:00
|
|
|
} while (--l);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_cstring:
|
|
|
|
*p = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_varying:
|
|
|
|
*(SSHORT *) p = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-01-13 10:52:19 +01:00
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = 0;
|
2004-01-13 10:52:19 +01:00
|
|
|
} while (--l);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
to_desc->dsc_flags |= DSC_null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle the null flag as appropriate for fields and message arguments. */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (to->nod_type == nod_field)
|
|
|
|
{
|
2004-01-21 08:18:30 +01:00
|
|
|
const SSHORT id = (USHORT)(IPTR) to->nod_arg[e_fld_id];
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record =
|
|
|
|
request->req_rpb[(int) (IPTR) to->nod_arg[e_fld_stream]].rpb_record;
|
2001-12-24 03:51:06 +01:00
|
|
|
if (null) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_NULL(record, id);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
CLEAR_NULL(record, id);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
else if (to->nod_type == nod_argument && to->nod_arg[e_arg_flag])
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
to_desc = EVL_assign_to(tdbb, to->nod_arg[e_arg_flag]);
|
|
|
|
|
|
|
|
/* If the null flag is a string with an effective length of one,
|
|
|
|
then -1 will not fit. Therefore, store 1 instead. */
|
|
|
|
|
|
|
|
if (null &&
|
|
|
|
to_desc->dsc_dtype <= dtype_varying &&
|
|
|
|
to_desc->dsc_length <=
|
2001-12-24 03:51:06 +01:00
|
|
|
((to_desc->dsc_dtype == dtype_text) ? 1 :
|
|
|
|
((to_desc->dsc_dtype == dtype_cstring) ? 2 :
|
|
|
|
3)))
|
|
|
|
{
|
|
|
|
null = 1;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
temp.dsc_dtype = dtype_short;
|
|
|
|
temp.dsc_length = sizeof(SSHORT);
|
|
|
|
temp.dsc_scale = 0;
|
|
|
|
temp.dsc_sub_type = 0;
|
|
|
|
temp.dsc_address = (UCHAR *) & null;
|
|
|
|
MOV_move(&temp, to_desc);
|
|
|
|
if (null && to->nod_arg[e_arg_indicator]) {
|
|
|
|
to_desc = EVL_assign_to(tdbb, to->nod_arg[e_arg_indicator]);
|
|
|
|
MOV_move(&temp, to_desc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-28 11:10:30 +02:00
|
|
|
bool EXE_crack(thread_db* tdbb, RecordSource* rsb, USHORT flags)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ c r a c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check whether stream is on a crack, BOF
|
|
|
|
* or EOF, according to the flags passed.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DEV_BLKCHK(rsb, type_rsb);
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* correct boolean rsbs to point to the "real" rsb */
|
|
|
|
|
|
|
|
if (rsb->rsb_type == rsb_boolean)
|
|
|
|
rsb = rsb->rsb_next;
|
2003-12-31 06:36:12 +01:00
|
|
|
irsb* impure = (IRSB) ((UCHAR *) request + rsb->rsb_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-09 10:51:02 +01:00
|
|
|
/* if any of the passed flags are set, return true */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-09 10:51:02 +01:00
|
|
|
return (impure->irsb_flags & flags);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
jrd_req* EXE_find_request(thread_db* tdbb, jrd_req* request, bool validate)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ f i n d _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Find an inactive incarnation of a trigger request. If necessary,
|
|
|
|
* clone it.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
#ifdef ANY_THREADING
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* I found a core file from my test runs that came from a NULL request -
|
|
|
|
* but have no idea what test was running. Let's bugcheck so we can
|
|
|
|
* figure it out
|
|
|
|
*/
|
|
|
|
if (!request)
|
|
|
|
BUGCHECK /* REQUEST */ (167); /* msg 167 invalid SEND request */
|
|
|
|
|
|
|
|
THD_MUTEX_LOCK(dbb->dbb_mutexes + DBB_MUTX_clone);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* clone = NULL;
|
|
|
|
USHORT count = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(request->req_flags & req_in_use))
|
|
|
|
clone = request;
|
|
|
|
else {
|
|
|
|
if (request->req_attachment == tdbb->tdbb_attachment)
|
|
|
|
count++;
|
|
|
|
|
|
|
|
/* Request exists and is in use. Search clones for one in use by
|
|
|
|
this attachment. If not found, return first inactive request. */
|
|
|
|
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<jrd_req*>* vector = request->req_sub_requests;
|
2003-12-31 06:36:12 +01:00
|
|
|
const USHORT clones = (vector) ? (vector->count() - 1) : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
USHORT n;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (n = 1; n <= clones; n++) {
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* next = CMP_clone_request(tdbb, request, n, validate);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (next->req_attachment == tdbb->tdbb_attachment) {
|
|
|
|
if (!(next->req_flags & req_in_use)) {
|
|
|
|
clone = next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
else if (!(next->req_flags & req_in_use) && !clone)
|
|
|
|
clone = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count > MAX_CLONES) {
|
|
|
|
THD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_clone);
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_max_clones_exceeded, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (!clone)
|
|
|
|
clone = CMP_clone_request(tdbb, request, n, validate);
|
|
|
|
}
|
|
|
|
clone->req_attachment = tdbb->tdbb_attachment;
|
|
|
|
clone->req_flags |= req_in_use;
|
|
|
|
THD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_clone);
|
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-28 11:10:30 +02:00
|
|
|
void EXE_mark_crack(thread_db* tdbb, RecordSource* rsb, USHORT flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ m a r k _ c r a c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Mark a stream as being at a crack,
|
|
|
|
* plus report the fact in the status
|
|
|
|
* vector.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(rsb, type_rsb);
|
|
|
|
|
|
|
|
/* correct boolean rsbs to point to the "real" rsb */
|
|
|
|
|
|
|
|
if (rsb->rsb_type == rsb_boolean)
|
|
|
|
rsb = rsb->rsb_next;
|
|
|
|
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, flag);
|
|
|
|
|
|
|
|
if (flag == irsb_eof)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_warning(isc_stream_eof, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (flag == irsb_bof)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_warning(isc_stream_bof, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (flag & irsb_crack)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_warning(isc_stream_crack, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_receive(thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request,
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT msg,
|
|
|
|
USHORT length,
|
|
|
|
UCHAR* buffer)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ r e c e i v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Move a message from JRD to the host program. This corresponds to
|
2004-01-13 10:52:19 +01:00
|
|
|
* a JRD BLR/jrd_nod* send.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
2004-10-03 14:10:19 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
if (--tdbb->tdbb_quantum < 0)
|
|
|
|
JRD_reschedule(tdbb, 0, true);
|
|
|
|
#endif
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_tra* transaction = request->req_transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!(request->req_flags & req_active)) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (request->req_flags & req_proc_fetch)
|
|
|
|
{
|
|
|
|
/* request->req_proc_sav_point stores all the request savepoints.
|
|
|
|
When going to continue execution put request save point list
|
|
|
|
into transaction->tra_save_point so that it is used in looper.
|
|
|
|
When we come back to EXE_receive() restore
|
|
|
|
transaction->tra_save_point and merge all work done under
|
|
|
|
stored procedure savepoints into the current transaction
|
|
|
|
savepoint, which is the savepoint for fetch. */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const save_sav_point = transaction->tra_save_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->tra_save_point = request->req_proc_sav_point;
|
|
|
|
request->req_proc_sav_point = save_sav_point;
|
|
|
|
|
|
|
|
if (!transaction->tra_save_point) {
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_start_save_point(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2002-04-03 01:12:03 +02:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (request->req_message->nod_type == nod_stall
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
|| request->req_flags & req_fetch_required
|
|
|
|
#endif
|
|
|
|
)
|
2002-11-21 00:18:16 +01:00
|
|
|
execute_looper(tdbb, request, transaction, jrd_req::req_sync);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!(request->req_flags & req_active) ||
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation != jrd_req::req_send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
const jrd_nod* message = request->req_message;
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) message->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
if (msg != (USHORT)(IPTR) message->nod_arg[e_msg_number])
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
if (length != format->fmt_length) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_port_len,
|
|
|
|
isc_arg_number, (SLONG) length,
|
|
|
|
isc_arg_number, (SLONG) format->fmt_length, 0);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if ((U_IPTR) buffer & (ALIGNMENT - 1))
|
|
|
|
MOVE_FAST((SCHAR *) request + message->nod_impure, buffer, length);
|
|
|
|
else
|
|
|
|
MOVE_FASTER((SCHAR *) request + message->nod_impure, buffer, length);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
execute_looper(tdbb, request, transaction, jrd_req::req_proceed);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-04-03 01:12:03 +02:00
|
|
|
} //try
|
2003-02-14 10:26:55 +01:00
|
|
|
catch (const std::exception&)
|
2002-04-03 01:12:03 +02:00
|
|
|
{
|
|
|
|
if (request->req_flags & req_proc_fetch)
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const save_sav_point = transaction->tra_save_point;
|
2002-04-03 01:12:03 +02:00
|
|
|
transaction->tra_save_point = request->req_proc_sav_point;
|
|
|
|
request->req_proc_sav_point = save_sav_point;
|
|
|
|
release_proc_save_points(request);
|
|
|
|
}
|
2002-04-18 05:54:36 +02:00
|
|
|
throw;
|
2002-04-03 01:12:03 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (request->req_flags & req_proc_fetch) {
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const save_sav_point = transaction->tra_save_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->tra_save_point = request->req_proc_sav_point;
|
|
|
|
request->req_proc_sav_point = save_sav_point;
|
|
|
|
VIO_merge_proc_sav_points(tdbb, transaction,
|
|
|
|
&request->req_proc_sav_point);
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_seek(thread_db* tdbb, jrd_req* request, USHORT direction, ULONG offset)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ s e e k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Seek a given request in a particular direction
|
|
|
|
* for offset records.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
|
|
|
/* loop through all RSEs in the request,
|
|
|
|
and describe the rsb tree for that rsb;
|
|
|
|
go backwards because items were popped
|
|
|
|
off the stack backwards */
|
|
|
|
|
|
|
|
/* find the top-level rsb in the request and seek it */
|
|
|
|
|
2004-01-10 02:48:46 +01:00
|
|
|
for (SLONG i = request->req_fors.getCount() - 1; i >= 0; i--) {
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = request->req_fors[i];
|
2003-12-31 06:36:12 +01:00
|
|
|
if (rsb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
seek_rsb(tdbb, request, rsb, direction, offset);
|
|
|
|
break;
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_send(thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request,
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT msg,
|
|
|
|
USHORT length,
|
2004-11-10 05:26:45 +01:00
|
|
|
const UCHAR* buffer)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ s e n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Send a message from the host program to the engine.
|
|
|
|
* This corresponds to a blr_receive or blr_select statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
2004-10-03 14:10:19 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
if (--tdbb->tdbb_quantum < 0)
|
|
|
|
JRD_reschedule(tdbb, 0, true);
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(request->req_flags & req_active))
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_nod* message;
|
|
|
|
jrd_nod* node;
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
/* look for an asynchronous send message--if such
|
|
|
|
a message was defined, we allow the user to send
|
|
|
|
us a message at any time during request execution */
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod* save_next = NULL, save_message = NULL;
|
2003-12-31 06:36:12 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((message = request->req_async_message) &&
|
|
|
|
(node = message->nod_arg[e_send_message]) &&
|
2003-12-31 06:36:12 +01:00
|
|
|
(msg == (USHORT)(ULONG) node->nod_arg[e_msg_number]))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* save the current state of the request so we can go
|
|
|
|
back to what was interrupted */
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
const USHORT save_operation = request->req_operation;
|
2001-05-23 15:26:42 +02:00
|
|
|
save_message = request->req_message;
|
|
|
|
save_next = request->req_next;
|
|
|
|
|
|
|
|
request->req_operation = req_receive;
|
|
|
|
request->req_message = node;
|
|
|
|
request->req_next = message->nod_arg[e_send_statement];
|
|
|
|
|
|
|
|
/* indicate that we are processing an asynchronous message */
|
|
|
|
|
|
|
|
request->req_flags |= req_async_processing;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
#endif
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation != jrd_req::req_receive)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
node = request->req_message;
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_tra* transaction = request->req_transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (node->nod_type == nod_message)
|
|
|
|
message = node;
|
2003-12-31 06:36:12 +01:00
|
|
|
else if (node->nod_type == nod_select) {
|
|
|
|
jrd_nod** ptr = node->nod_arg;
|
|
|
|
for (const jrd_nod* const* const end = ptr + node->nod_count; ptr < end;
|
|
|
|
ptr++)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
message = (*ptr)->nod_arg[e_send_message];
|
2004-01-21 08:18:30 +01:00
|
|
|
if ((USHORT)(IPTR) message->nod_arg[e_msg_number] == msg) {
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_next = *ptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
BUGCHECK(167); /* msg 167 invalid SEND request */
|
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) message->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
if (msg != (USHORT)(IPTR) message->nod_arg[e_msg_number])
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
if (length != format->fmt_length) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_port_len,
|
|
|
|
isc_arg_number, (SLONG) length,
|
|
|
|
isc_arg_number, (SLONG) format->fmt_length, 0);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if ((U_IPTR) buffer & (ALIGNMENT - 1))
|
|
|
|
MOVE_FAST(buffer, (SCHAR *) request + message->nod_impure, length);
|
|
|
|
else
|
|
|
|
MOVE_FASTER(buffer, (SCHAR *) request + message->nod_impure, length);
|
|
|
|
|
2005-05-28 00:45:31 +02:00
|
|
|
for (USHORT i = 0; i < format->fmt_count; ++i)
|
|
|
|
{
|
|
|
|
const DSC* desc = &format->fmt_desc[i];
|
|
|
|
|
|
|
|
// ASF: I'll not test for dtype_cstring because usage is only internal
|
|
|
|
if (desc->dsc_dtype == dtype_text || desc->dsc_dtype == dtype_varying)
|
|
|
|
{
|
2005-12-02 17:48:47 +01:00
|
|
|
const UCHAR* p = (UCHAR*)request + message->nod_impure + (ULONG)(IPTR)desc->dsc_address;
|
2005-05-28 00:45:31 +02:00
|
|
|
USHORT len;
|
|
|
|
|
|
|
|
switch (desc->dsc_dtype)
|
|
|
|
{
|
|
|
|
case dtype_text:
|
|
|
|
len = desc->dsc_length;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_varying:
|
|
|
|
len = reinterpret_cast<const vary*>(p)->vary_length;
|
|
|
|
p += sizeof(USHORT);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-06-14 05:16:54 +02:00
|
|
|
CharSet* charSet = INTL_charset_lookup(tdbb, DSC_GET_CHARSET(desc));
|
2005-05-28 00:45:31 +02:00
|
|
|
|
2005-06-14 05:16:54 +02:00
|
|
|
if (!charSet->wellFormed(len, p))
|
2005-05-28 00:45:31 +02:00
|
|
|
ERR_post(isc_malformed_string, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
execute_looper(tdbb, request, transaction, jrd_req::req_proceed);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
if (save_next) {
|
|
|
|
/* if the message was sent asynchronously, restore all the
|
|
|
|
previous values so that whatever we were trying to do when
|
|
|
|
the message came in is what we do next */
|
|
|
|
|
|
|
|
request->req_operation = save_operation;
|
|
|
|
request->req_message = save_message;
|
|
|
|
request->req_next = save_next;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_start(thread_db* tdbb, jrd_req* request, jrd_tra* transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start an execution running.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
BLKCHK(request, type_req);
|
|
|
|
BLKCHK(transaction, type_tra);
|
|
|
|
|
|
|
|
if (request->req_flags & req_active)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, isc_arg_gds, isc_reqinuse, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_prepared)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_no_trans, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Post resources to transaction block. In particular, the interest locks
|
|
|
|
on relations/indices are copied to the transaction, which is very
|
|
|
|
important for (short-lived) dynamically compiled requests. This will
|
|
|
|
provide transaction stability by preventing a relation from being
|
|
|
|
dropped after it has been referenced from an active transaction. */
|
|
|
|
|
|
|
|
TRA_post_resources(tdbb, transaction, request->req_resources);
|
|
|
|
|
2004-12-07 02:19:55 +01:00
|
|
|
TRA_attach_request(transaction, request);
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags &= REQ_FLAGS_INIT_MASK;
|
|
|
|
request->req_flags |= req_active;
|
|
|
|
request->req_flags &= ~req_reserved;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* set up to count records affected by request */
|
|
|
|
|
|
|
|
request->req_flags |= req_count_records;
|
|
|
|
request->req_records_selected = 0;
|
|
|
|
request->req_records_updated = 0;
|
|
|
|
request->req_records_inserted = 0;
|
|
|
|
request->req_records_deleted = 0;
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: set up to count virtual operations on SQL views. */
|
|
|
|
|
|
|
|
request->req_view_flags = 0;
|
|
|
|
request->req_top_view_store = NULL;
|
|
|
|
request->req_top_view_modify = NULL;
|
|
|
|
request->req_top_view_erase = NULL;
|
|
|
|
|
2003-02-12 20:12:15 +01:00
|
|
|
// Store request start time for timestamp work
|
2004-10-30 21:41:54 +02:00
|
|
|
request->req_timestamp.validate();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-23 21:17:30 +01:00
|
|
|
// Set all invariants to not computed.
|
|
|
|
jrd_nod **ptr, **end;
|
|
|
|
for (ptr = request->req_invariants.begin(),
|
|
|
|
end = request->req_invariants.end(); ptr < end;
|
|
|
|
++ptr)
|
2003-02-12 20:12:15 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_value* impure = (impure_value*) ((SCHAR *) request + (*ptr)->nod_impure);
|
2003-11-23 21:17:30 +01:00
|
|
|
impure->vlu_flags = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-02-12 20:12:15 +01:00
|
|
|
// Start a save point if not in middle of one
|
2002-11-03 18:29:51 +01:00
|
|
|
if (transaction && (transaction != dbb->dbb_sys_trans)) {
|
|
|
|
VIO_start_save_point(tdbb, transaction);
|
|
|
|
}
|
2003-02-12 20:12:15 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
START_CHECK_FOR_EXCEPTIONS(NULL);
|
|
|
|
#endif
|
2003-02-12 20:12:15 +01:00
|
|
|
// TODO:
|
|
|
|
// 1. Try to fix the problem with MSVC C++ runtime library, making
|
|
|
|
// even C++ exceptions that are implemented in terms of Win32 SEH
|
|
|
|
// getting catched by the SEH handler below.
|
|
|
|
// 2. Check if it really is correct that only Win32 catches CPU
|
|
|
|
// exceptions (such as SEH) here. Shouldn't any platform capable
|
|
|
|
// of handling signals use this stuff?
|
|
|
|
// (see jrd/ibsetjmp.h for implementation of these macros)
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
looper(tdbb, request, request->req_top_node);
|
2003-02-12 20:12:15 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
END_CHECK_FOR_EXCEPTIONS(NULL);
|
|
|
|
#endif
|
|
|
|
|
2003-02-12 20:12:15 +01:00
|
|
|
// If any requested modify/delete/insert ops have completed, forget them
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-12 20:12:15 +01:00
|
|
|
if (transaction &&
|
|
|
|
(transaction != dbb->dbb_sys_trans) &&
|
|
|
|
transaction->tra_save_point &&
|
|
|
|
!(transaction->tra_save_point->sav_flags & SAV_user) &&
|
|
|
|
!transaction->tra_save_point->sav_verb_count)
|
|
|
|
{
|
|
|
|
// Forget about any undo for this verb
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void EXE_unwind(thread_db* tdbb, jrd_req* request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X E _ u n w i n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2005-01-21 11:45:01 +01:00
|
|
|
* Unwind a request, maybe active, maybe not. This is particularly
|
2001-05-23 15:26:42 +02:00
|
|
|
* simple since nothing really needs to be done.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
if (request->req_flags & req_active) {
|
2004-01-10 02:48:46 +01:00
|
|
|
if (request->req_fors.getCount()) {
|
2004-08-30 20:11:08 +02:00
|
|
|
Jrd::ContextPoolHolder context(tdbb, request->req_pool);
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* old_request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_request = request;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* old_transaction = tdbb->tdbb_transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_transaction = request->req_transaction;
|
2003-12-31 06:36:12 +01:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource** ptr = request->req_fors.begin();
|
|
|
|
for (const RecordSource* const* const end =
|
2004-01-10 02:48:46 +01:00
|
|
|
request->req_fors.end(); ptr < end; ptr++)
|
2003-11-02 12:55:17 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*ptr)
|
2004-01-10 02:48:46 +01:00
|
|
|
RSE_close(tdbb, *ptr);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_request = old_request;
|
2003-07-14 15:21:07 +02:00
|
|
|
tdbb->tdbb_transaction = old_transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
release_blobs(tdbb, request);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request->req_proc_sav_point && (request->req_flags & req_proc_fetch))
|
|
|
|
release_proc_save_points(request);
|
|
|
|
|
2004-12-07 02:19:55 +01:00
|
|
|
TRA_detach_request(request);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags &= ~(req_active | req_proc_fetch | req_reserved);
|
|
|
|
request->req_flags |= req_abort | req_stall;
|
2004-10-30 21:41:54 +02:00
|
|
|
request->req_timestamp.invalidate();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: Moved to its own routine, originally in store(). */
|
2004-03-18 06:56:06 +01:00
|
|
|
static void cleanup_rpb(thread_db* tdbb, record_param* rpb)
|
2002-06-30 11:58:20 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c l e a n u p _ r p b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Perform cleaning of rpb, zeroing unassigned fields and
|
|
|
|
* the impure tail of varying fields that we don't want to carry
|
|
|
|
* when the RLE algorithm is applied.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record = rpb->rpb_record;
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = record->rec_format;
|
2002-06-30 11:58:20 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb); /* Is it necessary? */
|
|
|
|
|
|
|
|
/*
|
|
|
|
Starting from the format, walk through its
|
|
|
|
array of descriptors. If the descriptor has
|
|
|
|
no address, its a computed field and we shouldn't
|
|
|
|
try to fix it. Get a pointer to the actual data
|
|
|
|
and see if that field is null by indexing into
|
|
|
|
the null flags between the record header and the
|
|
|
|
record data.
|
|
|
|
*/
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
for (USHORT n = 0; n < format->fmt_count; n++)
|
2002-06-30 11:58:20 +02:00
|
|
|
{
|
2003-12-31 06:36:12 +01:00
|
|
|
const dsc* desc = &format->fmt_desc[n];
|
2002-06-30 11:58:20 +02:00
|
|
|
if (!desc->dsc_address)
|
|
|
|
continue;
|
2004-01-21 08:18:30 +01:00
|
|
|
UCHAR* p = record->rec_data + (IPTR) desc->dsc_address;
|
2002-06-30 11:58:20 +02:00
|
|
|
if (TEST_NULL(record, n))
|
|
|
|
{
|
2003-12-31 06:36:12 +01:00
|
|
|
USHORT length = desc->dsc_length;
|
|
|
|
if (length) {
|
|
|
|
do {
|
|
|
|
*p++ = 0;
|
|
|
|
} while (--length);
|
|
|
|
}
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
|
|
|
else if (desc->dsc_dtype == dtype_varying)
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
vary* varying = reinterpret_cast<vary*>(p);
|
2003-12-31 06:36:12 +01:00
|
|
|
USHORT length = desc->dsc_length - sizeof(USHORT);
|
2004-02-20 07:43:27 +01:00
|
|
|
if (length > varying->vary_length)
|
2002-06-30 11:58:20 +02:00
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
p = reinterpret_cast<UCHAR*>(varying->vary_string + varying->vary_length);
|
|
|
|
length -= varying->vary_length;
|
2003-12-31 06:36:12 +01:00
|
|
|
do {
|
|
|
|
*p++ = 0;
|
|
|
|
} while (--length);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
inline void PreModifyEraseTriggers(thread_db* tdbb,
|
2003-05-05 13:47:45 +02:00
|
|
|
trig_vec** trigs,
|
|
|
|
SSHORT which_trig,
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb,
|
|
|
|
Record* rec,
|
2003-05-05 13:47:45 +02:00
|
|
|
jrd_req::req_ta op)
|
|
|
|
{
|
|
|
|
/******************************************************
|
|
|
|
*
|
|
|
|
* P r e M o d i f y E r a s e T r i g g e r s
|
|
|
|
*
|
|
|
|
******************************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Perform operation's pre-triggers,
|
|
|
|
* storing active rpb in chain.
|
|
|
|
*
|
|
|
|
******************************************************/
|
2005-04-15 16:33:55 +02:00
|
|
|
if (! tdbb->tdbb_transaction->tra_rpblist) {
|
|
|
|
tdbb->tdbb_transaction->tra_rpblist =
|
|
|
|
FB_NEW(*tdbb->tdbb_transaction->tra_pool)
|
|
|
|
traRpbList(*tdbb->tdbb_transaction->tra_pool);
|
|
|
|
}
|
|
|
|
const int rpblevel =
|
|
|
|
tdbb->tdbb_transaction->tra_rpblist->PushRpb(rpb);
|
|
|
|
jrd_req* trigger = NULL;
|
2003-05-05 13:47:45 +02:00
|
|
|
if ((*trigs) && (which_trig != POST_TRIG)) {
|
2005-04-15 16:33:55 +02:00
|
|
|
trigger = execute_triggers(tdbb, trigs, rpb->rpb_record, rec, op);
|
|
|
|
}
|
|
|
|
tdbb->tdbb_transaction->tra_rpblist->PopRpb(rpb, rpblevel);
|
|
|
|
if (trigger) {
|
|
|
|
trigger_failure(tdbb, trigger);
|
2003-05-05 13:47:45 +02:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* erase(thread_db* tdbb, jrd_nod* node, SSHORT which_trig)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e r a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Perform erase operation.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
|
|
|
jrd_tra* transaction = request->req_transaction;
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb = &request->req_rpb[(int) (IPTR) node->nod_arg[e_erase_stream]];
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-06-07 12:52:38 +02:00
|
|
|
if (rpb->rpb_number.isBof()) {
|
2005-04-25 22:31:52 +02:00
|
|
|
ERR_post(isc_no_cur_rec, 0);
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef PC_ENGINE
|
|
|
|
/* for navigational streams, retrieve the rsb */
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = NULL;
|
2003-12-31 06:36:12 +01:00
|
|
|
irsb* impure = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (node->nod_arg[e_erase_rsb]) {
|
2004-03-28 11:10:30 +02:00
|
|
|
rsb = *(RecordSource**) node->nod_arg[e_erase_rsb];
|
2001-05-23 15:26:42 +02:00
|
|
|
impure = (IRSB) ((UCHAR *) request + rsb->rsb_impure);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(false);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!node->nod_arg[e_erase_statement])
|
|
|
|
break;
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = MET_current(tdbb, rpb->rpb_relation);
|
2004-08-16 14:28:43 +02:00
|
|
|
Record* record = VIO_record(tdbb, rpb, format, tdbb->getDefaultPool());
|
2001-05-23 15:26:42 +02:00
|
|
|
rpb->rpb_address = record->rec_data;
|
|
|
|
rpb->rpb_length = format->fmt_length;
|
|
|
|
rpb->rpb_format_number = format->fmt_version;
|
|
|
|
return node->nod_arg[e_erase_statement];
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
/* if we are on a crack in a navigational stream, erase
|
|
|
|
is not a valid operation */
|
|
|
|
|
|
|
|
if (rsb && EXE_crack(tdbb, rsb, irsb_bof | irsb_eof | irsb_crack)) {
|
|
|
|
EXE_mark_crack(tdbb, rsb, impure->irsb_flags);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2004-03-18 06:56:06 +01:00
|
|
|
RLCK_reserve_relation(tdbb, transaction, relation, true, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If the stream was sorted, the various fields in the rpb are
|
|
|
|
probably junk. Just to make sure that everything is cool,
|
|
|
|
refetch and release the record. */
|
|
|
|
|
|
|
|
if (rpb->rpb_stream_flags & RPB_s_refetch) {
|
2005-03-27 17:01:56 +02:00
|
|
|
VIO_refetch_record(tdbb, rpb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
rpb->rpb_stream_flags &= ~RPB_s_refetch;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
/* set up to do record locking; in case of a consistency
|
|
|
|
mode transaction, we already have an exclusive lock on
|
|
|
|
the table, so don't bother */
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
LCK_RAII_wrapper implicit_lock;
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!(transaction->tra_flags & TRA_degree3))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* check whether record locking is turned on */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* record_locking = RLCK_record_locking(relation);
|
2001-12-24 03:51:06 +01:00
|
|
|
if (record_locking->lck_physical != LCK_PR)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* get an implicit lock on the record */
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
implicit_lock.assign(implicit_record_lock(transaction, rpb));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* set up to catch any errors so that we can
|
|
|
|
release the implicit lock */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
|
|
|
|
|
|
|
/* Handle pre-operation trigger */
|
2003-05-05 13:47:45 +02:00
|
|
|
PreModifyEraseTriggers(tdbb, &relation->rel_pre_erase,
|
2005-04-15 16:33:55 +02:00
|
|
|
which_trig, rpb, NULL,
|
|
|
|
jrd_req::req_trigger_delete);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (relation->rel_file)
|
2004-02-20 07:43:27 +01:00
|
|
|
EXT_erase(rpb, reinterpret_cast<int*>(transaction));
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (!relation->rel_view_rse)
|
2002-11-18 21:27:24 +01:00
|
|
|
VIO_erase(tdbb, rpb, transaction);
|
2005-12-05 12:06:46 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Handle post operation trigger */
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* trigger;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (relation->rel_post_erase &&
|
|
|
|
which_trig != PRE_TRIG &&
|
|
|
|
(trigger = execute_triggers(tdbb, &relation->rel_post_erase,
|
2005-12-05 12:06:46 +01:00
|
|
|
rpb->rpb_record, NULL,
|
2003-01-15 13:08:59 +01:00
|
|
|
jrd_req::req_trigger_delete)))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_delete_count, relation, true);
|
2001-12-24 03:51:06 +01:00
|
|
|
trigger_failure(tdbb, trigger);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* call IDX_erase (which checks constraints) after all post erase triggers
|
|
|
|
have fired. This is required for cascading referential integrity, which
|
|
|
|
can be implemented as post_erase triggers */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!relation->rel_file & !relation->rel_view_rse)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* bad_relation = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT bad_index;
|
|
|
|
|
2004-10-09 05:52:59 +02:00
|
|
|
const IDX_E error_code =
|
2001-12-24 03:51:06 +01:00
|
|
|
IDX_erase(tdbb, rpb, transaction, &bad_relation, &bad_index);
|
|
|
|
|
|
|
|
if (error_code) {
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_delete_count, relation, true);
|
2001-12-24 03:51:06 +01:00
|
|
|
ERR_duplicate_error(error_code, bad_relation, bad_index);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: Increment the counter only if we called VIO/EXT_erase() and
|
|
|
|
we were successful. */
|
|
|
|
if (!(request->req_view_flags & req_first_erase_return)) {
|
|
|
|
request->req_view_flags |= req_first_erase_return;
|
|
|
|
if (relation->rel_view_rse) {
|
|
|
|
request->req_top_view_erase = relation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (relation == request->req_top_view_erase) {
|
|
|
|
if (which_trig == ALL_TRIGS || which_trig == POST_TRIG) {
|
|
|
|
request->req_records_deleted++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (relation->rel_file || !relation->rel_view_rse) {
|
|
|
|
request->req_records_deleted++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2001-05-23 15:26:42 +02:00
|
|
|
--transaction->tra_save_point->sav_verb_count;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* if the stream is navigational, it is now positioned on a crack */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (rsb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RSE_MARK_CRACK(tdbb, rsb, irsb_crack);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void execute_looper(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request,
|
|
|
|
jrd_tra* transaction, enum jrd_req::req_s next_state)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x e c u t e _ l o o p e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wrapper around looper. This will execute
|
|
|
|
* looper with the save point mechanism.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Start a save point */
|
|
|
|
|
|
|
|
if (!(request->req_flags & req_proc_fetch) && request->req_transaction)
|
|
|
|
if (transaction && (transaction != dbb->dbb_sys_trans))
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_start_save_point(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
request->req_flags &= ~req_stall;
|
|
|
|
request->req_operation = next_state;
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
looper(tdbb, request, request->req_next);
|
|
|
|
|
|
|
|
/* If any requested modify/delete/insert ops have completed, forget them */
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
if (!(request->req_flags & req_proc_fetch) && request->req_transaction) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (transaction && (transaction != dbb->dbb_sys_trans) &&
|
|
|
|
transaction->tra_save_point &&
|
2003-12-31 06:36:12 +01:00
|
|
|
!transaction->tra_save_point->sav_verb_count)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Forget about any undo for this verb */
|
|
|
|
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void exec_sql(thread_db* tdbb, jrd_req* request, DSC* dsc)
|
2002-04-04 15:53:20 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x e c _ s q l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a string as SQL operator.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-20 16:33:30 +01:00
|
|
|
SET_TDBB(tdbb);
|
2002-04-04 15:53:20 +02:00
|
|
|
|
2004-03-20 16:33:30 +01:00
|
|
|
if (tdbb->tdbb_transaction->tra_callback_count >= MAX_CALLBACKS) {
|
|
|
|
ERR_post(isc_exec_sql_max_call_exceeded, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::string SqlStatementText;
|
|
|
|
ExecuteStatement::getString(SqlStatementText, dsc, request);
|
|
|
|
|
|
|
|
ISC_STATUS_ARRAY local;
|
2002-04-04 15:53:20 +02:00
|
|
|
memset(local, 0, sizeof(local));
|
2003-12-31 06:36:12 +01:00
|
|
|
ISC_STATUS* status = local;
|
2002-04-04 15:53:20 +02:00
|
|
|
|
2004-03-20 16:33:30 +01:00
|
|
|
tdbb->tdbb_transaction->tra_callback_count++;
|
|
|
|
callback_execute_immediate(status,
|
|
|
|
tdbb->tdbb_attachment,
|
|
|
|
tdbb->tdbb_transaction,
|
|
|
|
SqlStatementText);
|
|
|
|
tdbb->tdbb_transaction->tra_callback_count--;
|
2002-04-04 15:53:20 +02:00
|
|
|
|
|
|
|
if (status[1]) {
|
|
|
|
memcpy(tdbb->tdbb_status_vector, status, sizeof(local));
|
|
|
|
ERR_punt();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void execute_procedure(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x e c u t e _ p r o c e d u r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a stored procedure. Begin by
|
|
|
|
* assigning the input parameters. End
|
|
|
|
* by assigning the output parameters.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_nod* temp = node->nod_arg[e_esp_inputs];
|
|
|
|
if (temp) {
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod** ptr;
|
|
|
|
jrd_nod** end;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (ptr = temp->nod_arg, end = ptr + temp->nod_count; ptr < end;
|
|
|
|
ptr++)
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_assignment(tdbb, *ptr);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
USHORT in_msg_length;
|
|
|
|
SCHAR* in_msg;
|
|
|
|
jrd_nod* in_message = node->nod_arg[e_esp_in_msg];
|
|
|
|
if (in_message) {
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) in_message->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
in_msg_length = format->fmt_length;
|
|
|
|
in_msg = (SCHAR *) request + in_message->nod_impure;
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
|
|
|
|
USHORT out_msg_length;
|
|
|
|
SCHAR* out_msg;
|
|
|
|
jrd_nod* out_message = node->nod_arg[e_esp_out_msg];
|
|
|
|
if (out_message) {
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) out_message->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
out_msg_length = format->fmt_length;
|
|
|
|
out_msg = (SCHAR *) request + out_message->nod_impure;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_prc* procedure = (jrd_prc*) node->nod_arg[e_esp_procedure];
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* proc_request = EXE_find_request(tdbb, procedure->prc_request, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::Array<char> temp_buffer;
|
2003-12-31 06:36:12 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!out_message) {
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = (Format*) procedure->prc_output_msg->nod_arg[e_msg_format];
|
2001-05-23 15:26:42 +02:00
|
|
|
out_msg_length = format->fmt_length;
|
2005-05-17 20:20:07 +02:00
|
|
|
out_msg = temp_buffer.getBuffer(out_msg_length + DOUBLE_ALIGN - 1);
|
|
|
|
out_msg = (SCHAR *) FB_ALIGN((U_IPTR) out_msg, DOUBLE_ALIGN);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Catch errors so we can unwind cleanly */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2004-08-30 20:11:08 +02:00
|
|
|
// Save the old pool
|
|
|
|
Jrd::ContextPoolHolder context(tdbb, proc_request->req_pool);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
jrd_tra* transaction = request->req_transaction;
|
|
|
|
const SLONG save_point_number = transaction->tra_save_point->sav_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
proc_request->req_timestamp = request->req_timestamp;
|
|
|
|
EXE_start(tdbb, proc_request, transaction);
|
|
|
|
if (in_message) {
|
|
|
|
EXE_send(tdbb, proc_request, 0, in_msg_length,
|
2004-11-10 05:26:45 +01:00
|
|
|
reinterpret_cast<const UCHAR*>(in_msg));
|
2004-08-30 20:11:08 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
EXE_receive(tdbb, proc_request, 1, out_msg_length,
|
2004-02-20 07:43:27 +01:00
|
|
|
reinterpret_cast<UCHAR*>(out_msg));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Clean up all savepoints started during execution of the
|
|
|
|
procedure */
|
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
if (transaction != tdbb->tdbb_database->dbb_sys_trans) {
|
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
|
|
|
save_point && save_point_number < save_point->sav_number;
|
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
} // try
|
2003-02-13 14:33:57 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
tdbb->tdbb_request = request;
|
|
|
|
EXE_unwind(tdbb, proc_request);
|
2003-05-14 10:24:49 +02:00
|
|
|
proc_request->req_attachment = NULL;
|
2001-12-24 03:51:06 +01:00
|
|
|
proc_request->req_flags &= ~(req_in_use | req_proc_fetch);
|
2004-10-30 21:41:54 +02:00
|
|
|
proc_request->req_timestamp.invalidate();
|
2004-03-01 04:35:23 +01:00
|
|
|
throw;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_unwind(tdbb, proc_request);
|
|
|
|
tdbb->tdbb_request = request;
|
2001-12-28 07:31:38 +01:00
|
|
|
|
|
|
|
temp = node->nod_arg[e_esp_outputs];
|
|
|
|
if (temp) {
|
2004-01-13 10:52:19 +01:00
|
|
|
jrd_nod** ptr;
|
|
|
|
jrd_nod** end;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (ptr = temp->nod_arg, end = ptr + temp->nod_count; ptr < end;
|
|
|
|
ptr++)
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_assignment(tdbb, *ptr);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2001-12-28 07:31:38 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
proc_request->req_attachment = NULL;
|
|
|
|
proc_request->req_flags &= ~(req_in_use | req_proc_fetch);
|
2004-10-30 21:41:54 +02:00
|
|
|
proc_request->req_timestamp.invalidate();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_req* execute_triggers(thread_db* tdbb,
|
2004-02-20 07:43:27 +01:00
|
|
|
trig_vec** triggers,
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* old_rec,
|
|
|
|
Record* new_rec,
|
2003-08-22 12:56:55 +02:00
|
|
|
enum jrd_req::req_ta trigger_action)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x e c u t e _ t r i g g e r s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute group of triggers. Return pointer to failing trigger
|
|
|
|
* if any blow up.
|
|
|
|
*
|
|
|
|
**************************************/
|
2001-12-28 07:31:38 +01:00
|
|
|
if (!*triggers) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction = tdbb->tdbb_request->req_transaction;
|
2004-02-20 07:43:27 +01:00
|
|
|
trig_vec* vector = *triggers;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* result = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-12-05 12:06:46 +01:00
|
|
|
Record* null_rec = NULL;
|
|
|
|
|
|
|
|
if (!old_rec || !new_rec) {
|
|
|
|
const Record* record = old_rec ? old_rec : new_rec;
|
|
|
|
fb_assert(record && record->rec_format);
|
|
|
|
// copy the record
|
|
|
|
null_rec = FB_NEW_RPT(record->rec_pool, record->rec_length)
|
|
|
|
Record(record->rec_pool);
|
|
|
|
memcpy(null_rec, record, sizeof(Record));
|
|
|
|
// zero the record buffer
|
|
|
|
memset(null_rec->rec_data, 0, record->rec_length);
|
|
|
|
// initialize all fields to missing
|
|
|
|
const SSHORT n = (record->rec_format->fmt_count + 7) >> 3;
|
|
|
|
memset(null_rec->rec_data, 0xFF, n);
|
|
|
|
}
|
|
|
|
|
|
|
|
jrd_req* trigger = NULL;
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
try
|
|
|
|
{
|
2002-09-19 18:02:58 +02:00
|
|
|
for (trig_vec::iterator ptr = vector->begin(); ptr != vector->end(); ++ptr)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2002-09-19 18:02:58 +02:00
|
|
|
ptr->compile(tdbb);
|
2003-11-09 10:51:02 +01:00
|
|
|
trigger = EXE_find_request(tdbb, ptr->request, false);
|
2005-12-05 12:06:46 +01:00
|
|
|
trigger->req_rpb[0].rpb_record = old_rec ? old_rec : null_rec;
|
|
|
|
trigger->req_rpb[1].rpb_record = new_rec ? new_rec : null_rec;
|
2001-12-28 07:31:38 +01:00
|
|
|
trigger->req_timestamp = tdbb->tdbb_request->req_timestamp;
|
2003-01-15 13:08:59 +01:00
|
|
|
trigger->req_trigger_action = trigger_action;
|
2001-12-28 07:31:38 +01:00
|
|
|
EXE_start(tdbb, trigger, transaction);
|
|
|
|
trigger->req_attachment = NULL;
|
|
|
|
trigger->req_flags &= ~req_in_use;
|
2004-10-30 21:41:54 +02:00
|
|
|
trigger->req_timestamp.invalidate();
|
2002-11-21 00:18:16 +01:00
|
|
|
if (trigger->req_operation == jrd_req::req_unwind) {
|
2001-12-28 07:31:38 +01:00
|
|
|
result = trigger;
|
|
|
|
break;
|
|
|
|
}
|
2003-08-28 15:16:03 +02:00
|
|
|
trigger = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2005-12-05 12:06:46 +01:00
|
|
|
delete null_rec;
|
2001-12-28 07:31:38 +01:00
|
|
|
if (vector != *triggers) {
|
2002-09-19 18:02:58 +02:00
|
|
|
MET_release_triggers(tdbb, &vector);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
return result;
|
|
|
|
}
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2005-12-05 12:06:46 +01:00
|
|
|
delete null_rec;
|
2001-12-24 03:51:06 +01:00
|
|
|
if (vector != *triggers) {
|
2002-09-19 18:02:58 +02:00
|
|
|
MET_release_triggers(tdbb, &vector);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
if (!trigger) {
|
2002-09-19 18:02:58 +02:00
|
|
|
throw; // trigger probally fails to compile
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
return trigger;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* find(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f i n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Find the given key value in a stream.
|
|
|
|
* Assume that the stream is open.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = *((RecordSource**) node->nod_arg[e_find_rsb]);
|
2001-12-28 07:31:38 +01:00
|
|
|
|
2004-08-17 14:28:57 +02:00
|
|
|
const dsc* desc = EVL_expr(tdbb, node->nod_arg[e_find_operator]);
|
|
|
|
|
|
|
|
const USHORT blr_operator = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
(USHORT) MOV_get_long(desc, 0) : MAX_USHORT;
|
|
|
|
|
2004-10-14 21:09:19 +02:00
|
|
|
if (blr_operator != blr_equiv &&
|
|
|
|
blr_operator != blr_eql &&
|
2004-02-02 12:02:12 +01:00
|
|
|
blr_operator != blr_leq &&
|
|
|
|
blr_operator != blr_lss &&
|
|
|
|
blr_operator != blr_geq &&
|
|
|
|
blr_operator != blr_gtr)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_invalid_operator, 0);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-08-17 14:28:57 +02:00
|
|
|
desc = EVL_expr(tdbb, node->nod_arg[e_find_direction]);
|
|
|
|
|
|
|
|
const USHORT direction = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
(USHORT) MOV_get_long(desc, 0) : MAX_USHORT;
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
if (direction != blr_backward &&
|
|
|
|
direction != blr_forward &&
|
|
|
|
direction != blr_backward_starting &&
|
|
|
|
direction != blr_forward_starting)
|
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_invalid_direction, 0);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* try to find the record; the position is defined to be on a crack
|
|
|
|
regardless of whether we are at BOF or EOF; also be sure to perpetuate
|
|
|
|
the forced crack (bug #7024) */
|
|
|
|
|
2004-08-17 14:28:57 +02:00
|
|
|
if (!RSE_find_record(tdbb, rsb, blr_operator, direction,
|
|
|
|
node->nod_arg[e_find_args]))
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
|
|
|
if (EXE_crack(tdbb, rsb, irsb_bof | irsb_eof | irsb_crack))
|
|
|
|
{
|
|
|
|
if (EXE_crack(tdbb, rsb, irsb_forced_crack)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_mark_crack(tdbb, rsb, irsb_crack | irsb_forced_crack);
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else if (EXE_crack(tdbb, rsb, irsb_bof)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_mark_crack(tdbb, rsb, irsb_bof);
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else if (EXE_crack(tdbb, rsb, irsb_eof)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_mark_crack(tdbb, rsb, irsb_eof);
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_mark_crack(tdbb, rsb, irsb_crack);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* find_dbkey(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f i n d _ d b k e y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Find the given dbkey in a navigational stream,
|
|
|
|
* resetting the position of the stream to that record.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = *((RecordSource**) node->nod_arg[e_find_dbkey_rsb]);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
if (!RSE_find_dbkey(tdbb,
|
|
|
|
rsb,
|
|
|
|
node->nod_arg[e_find_dbkey_dbkey],
|
|
|
|
node->nod_arg[e_find_dbkey_version]))
|
|
|
|
{
|
|
|
|
EXE_mark_crack(tdbb, rsb, irsb_crack);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-18 06:56:06 +01:00
|
|
|
static Lock* implicit_record_lock(jrd_tra* transaction, record_param* rpb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i m p l i c i t _ r e c o r d _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* An update to a record is being attempted and
|
|
|
|
* record locking has been initiated. Take out
|
|
|
|
* an implicit record lock to prevent updating
|
|
|
|
* a record that someone has explicitly locked.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
DEV_BLKCHK(transaction, type_tra);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* record_locking = relation->rel_record_locking;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* occasionally we should check whether we really still need to
|
|
|
|
do record locking; this is defined as RECORD_LOCK_CHECK_INTERVAL--
|
|
|
|
if we can get a PR on the record locking lock there is no need
|
|
|
|
to do implicit locking anymore */
|
|
|
|
|
|
|
|
if ((record_locking->lck_physical == LCK_none) &&
|
|
|
|
!(relation->rel_lock_total % RECORD_LOCK_CHECK_INTERVAL) &&
|
2004-10-07 11:15:32 +02:00
|
|
|
LCK_lock_non_blocking(tdbb, record_locking, LCK_PR, LCK_NO_WAIT))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* lock = RLCK_lock_record_implicit(transaction, rpb, LCK_SW, 0, 0);
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!lock) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_record_lock, 0);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-08-06 17:26:55 +02:00
|
|
|
static void stuff_stack_trace(const jrd_req* request)
|
|
|
|
{
|
|
|
|
Firebird::string sTrace;
|
|
|
|
bool isEmpty = true;
|
|
|
|
|
2004-10-04 10:15:00 +02:00
|
|
|
for (const jrd_req* req = request; req; req = req->req_caller)
|
2004-08-06 17:26:55 +02:00
|
|
|
{
|
|
|
|
Firebird::string name;
|
|
|
|
|
|
|
|
if (req->req_trg_name) {
|
|
|
|
name = "At trigger '";
|
|
|
|
name += req->req_trg_name;
|
|
|
|
}
|
|
|
|
else if (req->req_procedure) {
|
|
|
|
name = "At procedure '";
|
2005-05-12 20:28:04 +02:00
|
|
|
name += req->req_procedure->prc_name.c_str();
|
2004-08-06 17:26:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (! name.isEmpty())
|
|
|
|
{
|
|
|
|
name.trim();
|
|
|
|
|
2004-10-04 10:15:00 +02:00
|
|
|
if (sTrace.length() + name.length() > MAX_STACK_TRACE)
|
2004-08-06 17:26:55 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
if (isEmpty) {
|
|
|
|
isEmpty = false;
|
|
|
|
sTrace += name + "'";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sTrace += "\n" + name + "'";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isEmpty)
|
2004-11-18 14:58:46 +01:00
|
|
|
ERR_post_nothrow(isc_stack_trace, isc_arg_string, ERR_cstring(sTrace), 0);
|
2004-08-06 17:26:55 +02:00
|
|
|
}
|
|
|
|
|
2004-08-21 11:29:46 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* looper(thread_db* tdbb, jrd_req* request, jrd_nod* in_node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l o o p e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Cycle thru request execution tree. Return next node for
|
|
|
|
* execution on stall or request complete.
|
|
|
|
*
|
|
|
|
**************************************/
|
2001-12-28 07:31:38 +01:00
|
|
|
SSHORT which_erase_trig = 0;
|
|
|
|
SSHORT which_sto_trig = 0;
|
|
|
|
SSHORT which_mod_trig = 0;
|
2004-03-01 04:35:23 +01:00
|
|
|
jrd_nod* top_node = 0;
|
|
|
|
jrd_nod* prev_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_tra* transaction = request->req_transaction;
|
|
|
|
if (!transaction) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_no_trans, 0);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(in_node, type_nod);
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
// Save the old pool and request to restore on exit
|
2004-08-16 14:28:43 +02:00
|
|
|
JrdMemoryPool* old_pool = tdbb->getDefaultPool();
|
2004-08-30 20:11:08 +02:00
|
|
|
Jrd::ContextPoolHolder context(tdbb, request->req_pool);
|
2001-12-28 07:31:38 +01:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* old_request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_request = request;
|
|
|
|
tdbb->tdbb_transaction = transaction;
|
2004-06-22 22:13:10 +02:00
|
|
|
fb_assert(request->req_caller == NULL);
|
|
|
|
request->req_caller = old_request;
|
2001-12-28 07:31:38 +01:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
const SLONG save_point_number = (transaction->tra_save_point) ?
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->tra_save_point->sav_number : 0;
|
|
|
|
|
2004-03-01 04:35:23 +01:00
|
|
|
jrd_nod* node = in_node;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.clear();
|
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
// Catch errors so we can unwind cleanly
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-06-28 11:32:46 +02:00
|
|
|
bool error_pending = false;
|
|
|
|
bool catch_disabled = false;
|
2004-08-06 17:26:55 +02:00
|
|
|
tdbb->tdbb_flags &= ~TDBB_stack_trace_done;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
// Execute stuff until we drop
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
while (node && !(request->req_flags & req_stall))
|
|
|
|
{
|
2002-04-03 01:12:03 +02:00
|
|
|
try {
|
|
|
|
|
2003-03-24 15:41:42 +01:00
|
|
|
#ifdef SUPERSERVER
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate &&
|
2005-06-12 08:27:12 +02:00
|
|
|
(--tdbb->tdbb_quantum < 0))
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2003-12-22 11:00:59 +01:00
|
|
|
JRD_reschedule(tdbb, 0, true);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#endif
|
2003-04-24 07:46:49 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#if defined(DEBUG_GDS_ALLOC) && FALSE
|
2001-12-28 07:31:38 +01:00
|
|
|
int node_type = node->nod_type;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (node->nod_type) {
|
|
|
|
case nod_asn_list:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2004-03-01 04:35:23 +01:00
|
|
|
jrd_nod** ptr = node->nod_arg;
|
2004-01-13 10:52:19 +01:00
|
|
|
for (const jrd_nod* const* const end = ptr + node->nod_count;
|
2001-05-23 15:26:42 +02:00
|
|
|
ptr < end; ptr++)
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_assignment(tdbb, *ptr);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_assignment:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_assignment(tdbb, node);
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_dcl_variable:
|
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_value* variable = (impure_value*) ((SCHAR *) request + node->nod_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
variable->vlu_desc = *(DSC *) (node->nod_arg + e_dcl_desc);
|
|
|
|
variable->vlu_desc.dsc_flags = 0;
|
|
|
|
variable->vlu_desc.dsc_address =
|
|
|
|
(UCHAR *) & variable->vlu_misc;
|
|
|
|
if (variable->vlu_desc.dsc_dtype <= dtype_varying
|
2003-11-02 12:55:17 +01:00
|
|
|
&& !variable->vlu_string)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
variable->vlu_string =
|
2004-08-16 14:28:43 +02:00
|
|
|
FB_NEW_RPT(*tdbb->getDefaultPool(),
|
2005-05-12 20:28:04 +02:00
|
|
|
variable->vlu_desc.dsc_length) VaryingString();
|
2001-05-23 15:26:42 +02:00
|
|
|
variable->vlu_string->str_length =
|
|
|
|
variable->vlu_desc.dsc_length;
|
|
|
|
variable->vlu_desc.dsc_address =
|
|
|
|
variable->vlu_string->str_data;
|
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_erase:
|
2002-11-21 00:18:16 +01:00
|
|
|
if ((request->req_operation == jrd_req::req_return) &&
|
2001-12-28 07:31:38 +01:00
|
|
|
(node->nod_arg[e_erase_sub_erase]))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!top_node) {
|
|
|
|
top_node = node;
|
|
|
|
which_erase_trig = PRE_TRIG;
|
|
|
|
}
|
|
|
|
prev_node = node;
|
|
|
|
node = erase(tdbb, node, which_erase_trig);
|
|
|
|
if (which_erase_trig == PRE_TRIG) {
|
|
|
|
node = prev_node->nod_arg[e_erase_sub_erase];
|
|
|
|
node->nod_parent = prev_node;
|
|
|
|
}
|
|
|
|
if (top_node == prev_node && which_erase_trig == POST_TRIG) {
|
|
|
|
top_node = NULL;
|
|
|
|
which_erase_trig = ALL_TRIGS;
|
|
|
|
}
|
|
|
|
else
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
prev_node = node;
|
|
|
|
node = erase(tdbb, node, ALL_TRIGS);
|
|
|
|
if (!(prev_node->nod_arg[e_erase_sub_erase]) &&
|
|
|
|
which_erase_trig == PRE_TRIG)
|
2001-12-28 07:31:38 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
which_erase_trig = POST_TRIG;
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
2002-10-29 21:20:44 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case nod_exec_proc:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_unwind) {
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
execute_procedure(tdbb, node);
|
|
|
|
node = node->nod_parent;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_for:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.clear();
|
2004-03-28 11:10:30 +02:00
|
|
|
RSE_open(tdbb, (RecordSource*) node->nod_arg[e_for_rsb]);
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (node->nod_arg[e_for_stall]) {
|
|
|
|
node = node->nod_arg[e_for_stall];
|
|
|
|
break;
|
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_sync:
|
2004-03-28 11:10:30 +02:00
|
|
|
if (RSE_get_record(tdbb, (RecordSource*) node->nod_arg[e_for_rsb],
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
RSE_get_next))
|
|
|
|
#else
|
|
|
|
RSE_get_forward))
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
node = node->nod_arg[e_for_statement];
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
default:
|
2004-03-28 11:10:30 +02:00
|
|
|
RSE_close(tdbb, (RecordSource*) node->nod_arg[e_for_rsb]);
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
case nod_dcl_cursor:
|
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2003-11-05 10:02:33 +01:00
|
|
|
const USHORT number = (USHORT) (IPTR) node->nod_arg[e_dcl_cursor_number];
|
2003-11-02 12:55:17 +01:00
|
|
|
// set up the cursors vector
|
2005-12-02 08:35:34 +01:00
|
|
|
request->req_cursors = vec<RecordSource*>::newVector(*request->req_pool,
|
2003-11-02 12:55:17 +01:00
|
|
|
request->req_cursors, number + 1);
|
2004-03-28 11:10:30 +02:00
|
|
|
// store RecordSource in the vector
|
2005-12-02 08:35:34 +01:00
|
|
|
(*request->req_cursors)[number] = (RecordSource*) node->nod_arg[e_dcl_cursor_rsb];
|
2003-11-02 12:55:17 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_cursor_stmt:
|
|
|
|
{
|
2003-11-05 10:02:33 +01:00
|
|
|
const UCHAR op = (UCHAR) (IPTR) node->nod_arg[e_cursor_stmt_op];
|
|
|
|
const USHORT number = (USHORT) (IPTR) node->nod_arg[e_cursor_stmt_number];
|
2004-03-28 11:10:30 +02:00
|
|
|
// get RecordSource and the impure area
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(request->req_cursors && number < request->req_cursors->count());
|
2005-12-02 08:35:34 +01:00
|
|
|
RecordSource* rsb = (*request->req_cursors)[number];
|
2003-11-02 12:55:17 +01:00
|
|
|
IRSB impure = (IRSB) ((UCHAR*) tdbb->tdbb_request + rsb->rsb_impure);
|
|
|
|
switch (op) {
|
|
|
|
case blr_cursor_open:
|
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
|
|
|
// check cursor state
|
|
|
|
if (impure->irsb_flags & irsb_open) {
|
2004-11-17 16:23:27 +01:00
|
|
|
ERR_post(isc_cursor_already_open, 0);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
|
|
|
// open cursor
|
|
|
|
RSE_open(tdbb, rsb);
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
case blr_cursor_close:
|
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
|
|
|
// check cursor state
|
|
|
|
if (!(impure->irsb_flags & irsb_open)) {
|
2004-11-17 16:23:27 +01:00
|
|
|
ERR_post(isc_cursor_not_open, 0);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
|
|
|
// close cursor
|
|
|
|
RSE_close(tdbb, rsb);
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
case blr_cursor_fetch:
|
|
|
|
switch (request->req_operation) {
|
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
// check cursor state
|
|
|
|
if (!(impure->irsb_flags & irsb_open)) {
|
2004-11-17 16:23:27 +01:00
|
|
|
ERR_post(isc_cursor_not_open, 0);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.clear();
|
2003-11-02 12:55:17 +01:00
|
|
|
// perform preliminary navigation, if specified
|
|
|
|
if (node->nod_arg[e_cursor_stmt_seek]) {
|
|
|
|
node = node->nod_arg[e_cursor_stmt_seek];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case jrd_req::req_return:
|
2005-06-24 14:56:34 +02:00
|
|
|
if (!request->req_records_affected.hasCursor()) {
|
2003-11-02 12:55:17 +01:00
|
|
|
// fetch one record
|
|
|
|
if (RSE_get_record(tdbb, rsb,
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
RSE_get_next))
|
|
|
|
#else
|
|
|
|
RSE_get_forward))
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
node = node->nod_arg[e_cursor_stmt_into];
|
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case nod_abort:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2002-09-28 16:04:35 +02:00
|
|
|
{
|
2005-05-22 08:13:54 +02:00
|
|
|
PsqlException* xcp_node = reinterpret_cast<PsqlException*>(node->nod_arg[e_xcp_desc]);
|
|
|
|
if (xcp_node)
|
|
|
|
{
|
|
|
|
/* PsqlException is defined,
|
|
|
|
so throw an exception */
|
|
|
|
set_error(tdbb, &xcp_node->xcp_rpt[0], node->nod_arg[e_xcp_msg]);
|
|
|
|
}
|
|
|
|
else if (!request->req_last_xcp.success())
|
|
|
|
{
|
|
|
|
/* PsqlException is undefined, but there was a known exception before,
|
|
|
|
so re-initiate it */
|
|
|
|
set_error(tdbb, NULL, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* PsqlException is undefined and there weren't any exceptions before,
|
|
|
|
so just do nothing */
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
}
|
2002-09-28 16:04:35 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
2003-06-10 15:40:19 +02:00
|
|
|
|
2002-10-29 21:20:44 +01:00
|
|
|
case nod_user_savepoint:
|
2003-06-26 12:44:16 +02:00
|
|
|
switch (request->req_operation) {
|
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2002-10-29 21:20:44 +01:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
const UCHAR operation = (UCHAR) (IPTR) node->nod_arg[e_sav_operation];
|
|
|
|
const TEXT* node_savepoint_name = (TEXT*) node->nod_arg[e_sav_name];
|
2003-04-13 15:22:23 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
// Skip the savepoint created by EXE_start
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* savepoint = transaction->tra_save_point->sav_next;
|
|
|
|
Savepoint* previous = transaction->tra_save_point;
|
2003-04-13 15:22:23 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
// Find savepoint
|
|
|
|
bool found = false;
|
|
|
|
while (true) {
|
|
|
|
if (!savepoint || !(savepoint->sav_flags & SAV_user))
|
|
|
|
break;
|
2003-04-13 15:22:23 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
if (!strcmp(node_savepoint_name, savepoint->sav_name)) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
previous = savepoint;
|
|
|
|
savepoint = savepoint->sav_next;
|
|
|
|
}
|
|
|
|
if (!found && operation != blr_savepoint_set) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_invalid_savepoint,
|
2004-11-06 07:05:38 +01:00
|
|
|
isc_arg_string, ERR_cstring(node_savepoint_name), 0);
|
2003-06-10 15:40:19 +02:00
|
|
|
}
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
switch (operation)
|
|
|
|
{
|
|
|
|
case blr_savepoint_set:
|
2003-06-26 12:44:16 +02:00
|
|
|
// Release the savepoint
|
|
|
|
if (found) {
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const current = transaction->tra_save_point;
|
2003-06-26 12:44:16 +02:00
|
|
|
transaction->tra_save_point = savepoint;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2005-01-26 19:04:37 +01:00
|
|
|
previous->sav_next = transaction->tra_save_point;
|
2003-06-26 12:44:16 +02:00
|
|
|
transaction->tra_save_point = current;
|
|
|
|
}
|
2003-06-10 15:40:19 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
// Use the savepoint created by EXE_start
|
|
|
|
transaction->tra_save_point->sav_flags |= SAV_user;
|
|
|
|
strcpy(transaction->tra_save_point->sav_name, node_savepoint_name);
|
2004-02-02 12:02:12 +01:00
|
|
|
break;
|
|
|
|
case blr_savepoint_release_single:
|
|
|
|
{
|
2003-06-26 12:44:16 +02:00
|
|
|
// Release the savepoint
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const current = transaction->tra_save_point;
|
2003-06-10 15:40:19 +02:00
|
|
|
transaction->tra_save_point = savepoint;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2005-01-26 19:04:37 +01:00
|
|
|
previous->sav_next = transaction->tra_save_point;
|
2003-06-10 15:40:19 +02:00
|
|
|
transaction->tra_save_point = current;
|
2004-02-02 12:02:12 +01:00
|
|
|
break;
|
2003-06-10 15:40:19 +02:00
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
case blr_savepoint_release:
|
|
|
|
{
|
|
|
|
const SLONG sav_number = savepoint->sav_number;
|
2003-06-10 15:40:19 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
// Release the savepoint and all subsequent ones
|
|
|
|
while (transaction->tra_save_point &&
|
|
|
|
transaction->tra_save_point->sav_number >= sav_number)
|
|
|
|
{
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2003-06-26 12:44:16 +02:00
|
|
|
}
|
2003-06-10 15:40:19 +02:00
|
|
|
|
2003-06-26 12:44:16 +02:00
|
|
|
// Restore the savepoint initially created by EXE_start
|
|
|
|
VIO_start_save_point(tdbb, transaction);
|
2004-02-02 12:02:12 +01:00
|
|
|
break;
|
2003-06-10 15:40:19 +02:00
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
case blr_savepoint_undo:
|
|
|
|
{
|
|
|
|
const SLONG sav_number = savepoint->sav_number;
|
2003-06-26 12:44:16 +02:00
|
|
|
|
|
|
|
// Undo the savepoint
|
|
|
|
while (transaction->tra_save_point &&
|
|
|
|
transaction->tra_save_point->sav_number >= sav_number)
|
|
|
|
{
|
|
|
|
transaction->tra_save_point->sav_verb_count++;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2003-06-26 12:44:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now set the savepoint again to allow to return to it later
|
|
|
|
VIO_start_save_point(tdbb, transaction);
|
|
|
|
transaction->tra_save_point->sav_flags |= SAV_user;
|
|
|
|
strcpy(transaction->tra_save_point->sav_name, node_savepoint_name);
|
2004-02-02 12:02:12 +01:00
|
|
|
break;
|
2003-06-26 12:44:16 +02:00
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
default:
|
2003-06-26 12:44:16 +02:00
|
|
|
BUGCHECK(232);
|
2004-02-02 12:02:12 +01:00
|
|
|
break;
|
2003-06-10 15:40:19 +02:00
|
|
|
}
|
|
|
|
}
|
2003-06-26 12:44:16 +02:00
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
request->req_operation = jrd_req::req_return;
|
2002-11-03 18:29:51 +01:00
|
|
|
}
|
2002-10-29 21:20:44 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case nod_start_savepoint:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Start a save point */
|
|
|
|
|
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_start_save_point(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_end_savepoint:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
case jrd_req::req_unwind:
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If any requested modify/delete/insert
|
|
|
|
ops have completed, forget them */
|
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
|
|
|
/* If an error is still pending when the savepoint is
|
|
|
|
supposed to end, then the application didn't handle the
|
|
|
|
error and the savepoint should be undone. */
|
2001-12-29 12:41:29 +01:00
|
|
|
if (error_pending) {
|
2001-05-23 15:26:42 +02:00
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_handler:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_arg[0];
|
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_unwind:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!request->req_label)
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_block:
|
2004-03-18 06:56:06 +01:00
|
|
|
switch (request->req_operation)
|
|
|
|
{
|
|
|
|
SLONG count;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_start_save_point(tdbb, transaction);
|
2004-03-18 06:56:06 +01:00
|
|
|
const Savepoint* save_point = transaction->tra_save_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
count = save_point->sav_number;
|
|
|
|
MOVE_FAST(&count,
|
|
|
|
(SCHAR *) request + node->nod_impure,
|
|
|
|
sizeof(SLONG));
|
|
|
|
}
|
|
|
|
node = node->nod_arg[e_blk_action];
|
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_unwind:
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2001-12-29 12:41:29 +01:00
|
|
|
if (request->req_flags & req_leave)
|
|
|
|
{
|
2003-08-30 18:43:16 +02:00
|
|
|
// Although the req_operation is set to req_unwind,
|
|
|
|
// it's not an error case if req_leave bit is set.
|
|
|
|
// req_leave bit indicates that we hit an EXIT or
|
|
|
|
// BREAK/LEAVE statement in the SP/trigger code.
|
|
|
|
// Do not perform the error handling stuff.
|
2005-01-04 14:09:11 +01:00
|
|
|
|
2003-08-30 18:43:16 +02:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
|
|
|
MOVE_FAST((SCHAR *) request + node->nod_impure,
|
|
|
|
&count, sizeof(SLONG));
|
2004-03-18 06:56:06 +01:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
2003-08-30 18:43:16 +02:00
|
|
|
save_point && count <= save_point->sav_number;
|
|
|
|
save_point = transaction->tra_save_point)
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2003-08-30 18:43:16 +02:00
|
|
|
}
|
2005-01-04 14:09:11 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
}
|
2001-12-29 12:41:29 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FAST((SCHAR *) request + node->nod_impure,
|
|
|
|
&count, sizeof(SLONG));
|
|
|
|
/* Since there occurred an error (req_unwind), undo all savepoints
|
|
|
|
up to, but not including, the savepoint of this block. The
|
|
|
|
savepoint of this block will be dealt with below. */
|
2004-03-18 06:56:06 +01:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
save_point && count < save_point->sav_number;
|
2004-01-13 10:52:19 +01:00
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2001-12-29 12:41:29 +01:00
|
|
|
|
2004-03-01 04:35:23 +01:00
|
|
|
jrd_nod* handlers = node->nod_arg[e_blk_handlers];
|
2001-12-29 12:41:29 +01:00
|
|
|
if (handlers)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
2004-03-01 04:35:23 +01:00
|
|
|
jrd_nod** ptr = handlers->nod_arg;
|
2004-01-13 10:52:19 +01:00
|
|
|
for (const jrd_nod* const* const end = ptr + handlers->nod_count;
|
|
|
|
ptr < end; ptr++)
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
const PsqlException* xcp_node =
|
|
|
|
reinterpret_cast<PsqlException*>((*ptr)->nod_arg[e_err_conditions]);
|
2002-09-28 16:04:35 +02:00
|
|
|
if (test_and_fixup_error(tdbb, xcp_node, request))
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = (*ptr)->nod_arg[e_err_action];
|
2003-06-28 11:32:46 +02:00
|
|
|
error_pending = false;
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* On entering looper old_request etc. are saved.
|
|
|
|
On recursive calling we will loose the actual old
|
|
|
|
request for that invocation of looper. Avoid this. */
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
{
|
|
|
|
Jrd::ContextPoolHolder contextLooper(tdbb, old_pool);
|
|
|
|
tdbb->tdbb_request = old_request;
|
|
|
|
fb_assert(request->req_caller == old_request);
|
|
|
|
request->req_caller = NULL;
|
|
|
|
|
|
|
|
/* Save the previous state of req_error_handler
|
|
|
|
bit. We need to restore it later. This is
|
|
|
|
necessary if the error handler is deeply
|
|
|
|
nested. */
|
|
|
|
|
|
|
|
const ULONG prev_req_error_handler =
|
|
|
|
request->req_flags & req_error_handler;
|
|
|
|
request->req_flags |= req_error_handler;
|
|
|
|
node = looper(tdbb, request, node);
|
2005-01-04 14:09:11 +01:00
|
|
|
request->req_flags &= ~req_error_handler;
|
2004-08-30 20:11:08 +02:00
|
|
|
request->req_flags |= prev_req_error_handler;
|
|
|
|
|
|
|
|
/* Note: Previously the above call
|
|
|
|
"node = looper (tdbb, request, node);"
|
|
|
|
never returned back till the node tree
|
|
|
|
was executed completely. Now that the looper
|
|
|
|
has changed its behaviour such that it
|
|
|
|
returns back after handling error. This
|
|
|
|
makes it necessary that the jmpbuf be reset
|
|
|
|
so that looper can proceede with the
|
|
|
|
processing of execution tree. If this is
|
|
|
|
not done then anymore errors will take the
|
|
|
|
engine out of looper there by abruptly
|
|
|
|
terminating the processing. */
|
|
|
|
|
|
|
|
catch_disabled = false;
|
|
|
|
tdbb->tdbb_request = request;
|
|
|
|
fb_assert(request->req_caller == NULL);
|
|
|
|
request->req_caller = old_request;
|
|
|
|
}
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* The error is dealt with by the application, cleanup
|
|
|
|
this block's savepoint. */
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
2005-08-18 15:56:24 +02:00
|
|
|
save_point && count <= save_point->sav_number;
|
2001-12-29 12:41:29 +01:00
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If the application didn't have an error handler, then
|
|
|
|
the error will still be pending. Undo the block by
|
|
|
|
using its savepoint. */
|
|
|
|
|
|
|
|
if (error_pending && transaction != dbb->dbb_sys_trans) {
|
2005-08-18 15:56:24 +02:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
|
|
|
save_point && count <= save_point->sav_number;
|
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
|
|
|
verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
|
|
|
MOVE_FAST((SCHAR *) request + node->nod_impure,
|
|
|
|
&count, sizeof(SLONG));
|
2004-03-18 06:56:06 +01:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
save_point && count <= save_point->sav_number;
|
|
|
|
save_point = transaction->tra_save_point)
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_error_handler:
|
2005-01-04 14:09:11 +01:00
|
|
|
if (request->req_flags & req_error_handler && !error_pending)
|
|
|
|
{
|
|
|
|
fb_assert(request->req_caller == old_request);
|
|
|
|
request->req_caller = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
node = node->nod_parent;
|
2005-01-04 14:09:11 +01:00
|
|
|
if (request->req_operation == jrd_req::req_unwind) {
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
2005-01-04 14:09:11 +01:00
|
|
|
}
|
2003-11-02 12:55:17 +01:00
|
|
|
request->req_last_xcp.clear();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_label:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_arg[e_lbl_statement];
|
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_unwind:
|
2004-01-21 08:18:30 +01:00
|
|
|
if ((request->req_label == (USHORT)(IPTR) node->nod_arg[e_lbl_label]) &&
|
2004-03-18 06:56:06 +01:00
|
|
|
(request->req_flags & (req_leave | req_error_handler)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags &= ~req_leave;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_leave:
|
|
|
|
request->req_flags |= req_leave;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_unwind;
|
2004-01-21 08:18:30 +01:00
|
|
|
request->req_label = (USHORT)(IPTR) node->nod_arg[0];
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_list:
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_state* impure = (impure_state*) ((SCHAR *) request + node->nod_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
impure->sta_state = 0;
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
|
|
|
case jrd_req::req_sync:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (impure->sta_state < node->nod_count) {
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_arg[impure->sta_state++];
|
|
|
|
break;
|
|
|
|
}
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case nod_loop:
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_arg[0];
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_if:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (EVL_boolean(tdbb, node->nod_arg[e_if_boolean])) {
|
|
|
|
node = node->nod_arg[e_if_true];
|
2003-11-23 18:39:38 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (node->nod_arg[e_if_false]) {
|
|
|
|
node = node->nod_arg[e_if_false];
|
2003-11-23 18:39:38 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_modify:
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_state* impure = (impure_state*) ((SCHAR *) request + node->nod_impure);
|
2002-11-21 00:18:16 +01:00
|
|
|
if ((request->req_operation == jrd_req::req_return) &&
|
2001-05-23 15:26:42 +02:00
|
|
|
(!impure->sta_state) && (node->nod_arg[e_mod_sub_mod])) {
|
|
|
|
if (!top_node) {
|
|
|
|
top_node = node;
|
|
|
|
which_mod_trig = PRE_TRIG;
|
|
|
|
}
|
|
|
|
prev_node = node;
|
|
|
|
node = modify(tdbb, node, which_mod_trig);
|
|
|
|
if (which_mod_trig == PRE_TRIG) {
|
|
|
|
node = prev_node->nod_arg[e_mod_sub_mod];
|
|
|
|
node->nod_parent = prev_node;
|
|
|
|
}
|
|
|
|
if (top_node == prev_node && which_mod_trig == POST_TRIG) {
|
|
|
|
top_node = NULL;
|
|
|
|
which_mod_trig = ALL_TRIGS;
|
|
|
|
}
|
2001-12-28 07:31:38 +01:00
|
|
|
else {
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
prev_node = node;
|
|
|
|
node = modify(tdbb, node, ALL_TRIGS);
|
|
|
|
if (!(prev_node->nod_arg[e_mod_sub_mod]) &&
|
2001-12-28 07:31:38 +01:00
|
|
|
which_mod_trig == PRE_TRIG)
|
|
|
|
{
|
|
|
|
which_mod_trig = POST_TRIG;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case nod_nop:
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_receive:
|
|
|
|
node = receive_msg(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
2002-04-04 15:53:20 +02:00
|
|
|
case nod_exec_sql:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_unwind) {
|
2002-04-04 15:53:20 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
exec_sql(tdbb, request, EVL_expr(tdbb, node->nod_arg[0]));
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
|
|
|
request->req_operation = jrd_req::req_return;
|
2002-04-04 15:53:20 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
2003-03-01 20:19:23 +01:00
|
|
|
case nod_exec_into:
|
|
|
|
{
|
2004-10-23 02:54:26 +02:00
|
|
|
ExecuteStatement* impure =
|
|
|
|
(ExecuteStatement*)
|
|
|
|
((SCHAR *) request + node->nod_impure);
|
|
|
|
switch (request->req_operation) {
|
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
impure->Open(tdbb, node->nod_arg[0], node->nod_count - 2,
|
|
|
|
(!node->nod_arg[1]));
|
|
|
|
case jrd_req::req_return:
|
|
|
|
case jrd_req::req_sync:
|
|
|
|
if (impure->Fetch(tdbb, &node->nod_arg[2])) {
|
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
|
|
|
node = node->nod_arg[1];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
request->req_operation = jrd_req::req_return;
|
|
|
|
default:
|
|
|
|
// if have active opened request - close it
|
|
|
|
impure->Close(tdbb);
|
|
|
|
node = node->nod_parent;
|
2003-03-01 20:19:23 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
case nod_post:
|
2003-01-15 13:08:59 +01:00
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
DeferredWork* work = DFW_post_work(transaction, dfw_post_event,
|
2003-01-15 13:08:59 +01:00
|
|
|
EVL_expr(tdbb, node->nod_arg[0]), 0);
|
|
|
|
if (node->nod_arg[1])
|
|
|
|
DFW_post_work_arg(transaction, work,
|
|
|
|
EVL_expr(tdbb, node->nod_arg[1]), 0);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
// for an autocommit transaction, events can be posted
|
|
|
|
// without any updates
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_autocommit)
|
|
|
|
transaction->tra_flags |= TRA_perform_autocommit;
|
|
|
|
case nod_message:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate)
|
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_stall:
|
|
|
|
node = stall(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_select:
|
|
|
|
node = selct(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_send:
|
|
|
|
node = send_msg(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_store:
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_state* impure = (impure_state*) ((SCHAR *) request + node->nod_impure);
|
2002-11-21 00:18:16 +01:00
|
|
|
if ((request->req_operation == jrd_req::req_return) &&
|
2004-01-13 10:52:19 +01:00
|
|
|
(!impure->sta_state) && (node->nod_arg[e_sto_sub_store]))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!top_node) {
|
|
|
|
top_node = node;
|
|
|
|
which_sto_trig = PRE_TRIG;
|
|
|
|
}
|
|
|
|
prev_node = node;
|
|
|
|
node = store(tdbb, node, which_sto_trig);
|
|
|
|
if (which_sto_trig == PRE_TRIG) {
|
|
|
|
node = prev_node->nod_arg[e_sto_sub_store];
|
|
|
|
node->nod_parent = prev_node;
|
|
|
|
}
|
|
|
|
if (top_node == prev_node && which_sto_trig == POST_TRIG) {
|
|
|
|
top_node = NULL;
|
|
|
|
which_sto_trig = ALL_TRIGS;
|
|
|
|
}
|
|
|
|
else
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
prev_node = node;
|
|
|
|
node = store(tdbb, node, ALL_TRIGS);
|
|
|
|
if (!(prev_node->nod_arg[e_sto_sub_store]) &&
|
|
|
|
which_sto_trig == PRE_TRIG)
|
|
|
|
which_sto_trig = POST_TRIG;
|
|
|
|
}
|
|
|
|
break;
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
case nod_seek:
|
|
|
|
node = seek_rse(tdbb, request, node);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
case nod_stream:
|
|
|
|
node = stream(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_find:
|
|
|
|
node = find(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_find_dbkey:
|
|
|
|
case nod_find_dbkey_version:
|
|
|
|
node = find_dbkey(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_set_index:
|
|
|
|
node = set_index(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_set_bookmark:
|
|
|
|
node = set_bookmark(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_release_bookmark:
|
|
|
|
node = release_bookmark(tdbb, node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_end_range:
|
|
|
|
node = RNG_end(node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_delete_range:
|
|
|
|
node = RNG_delete(node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_delete_ranges:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RNG_delete_ranges(request);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_range_relation:
|
|
|
|
node = RNG_add_relation(node);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_release_lock:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2001-05-23 15:26:42 +02:00
|
|
|
DSC *desc;
|
|
|
|
|
|
|
|
desc = EVL_expr(tdbb, node->nod_arg[e_rellock_lock]);
|
2002-09-17 07:58:40 +02:00
|
|
|
#if SIZEOF_VOID_P != 8
|
2004-03-18 06:56:06 +01:00
|
|
|
RLCK_release_lock(*(Lock**) desc->dsc_address);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
2004-01-13 10:52:19 +01:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* lock = NULL;
|
2004-01-13 10:52:19 +01:00
|
|
|
const ULONG slot = *(ULONG *) desc->dsc_address;
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<Lock*>* vector = attachment->att_lck_quick_ref;
|
|
|
|
if (vector && slot < vector->count()) {
|
|
|
|
lock = (*vector)[slot];
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
RLCK_release_lock(lock);
|
2005-12-02 08:35:34 +01:00
|
|
|
(*vector)[slot] = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_release_locks:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RLCK_release_locks(request->req_attachment);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_force_crack:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2004-03-28 11:10:30 +02:00
|
|
|
RSE_MARK_CRACK(tdbb, *(RecordSource**) node->nod_arg[1],
|
2001-05-23 15:26:42 +02:00
|
|
|
irsb_crack | irsb_forced_crack);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_reset_stream:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RSE_reset_position(tdbb,
|
2004-03-28 11:10:30 +02:00
|
|
|
*(RecordSource**) node->nod_arg[e_reset_from_rsb],
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_rpb +
|
2003-08-09 22:58:34 +02:00
|
|
|
(USHORT)(ULONG) node->nod_arg[e_reset_to_stream]);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
case nod_set_generator:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2004-01-13 10:52:19 +01:00
|
|
|
dsc* desc = EVL_expr(tdbb, node->nod_arg[e_gen_value]);
|
2004-07-07 05:47:12 +02:00
|
|
|
DPM_gen_id(tdbb, (IPTR) node->nod_arg[e_gen_id], true,
|
2001-05-23 15:26:42 +02:00
|
|
|
MOV_get_int64(desc, 0));
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case nod_set_generator2:
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2004-01-13 10:52:19 +01:00
|
|
|
dsc* desc = EVL_expr(tdbb, node->nod_arg[e_gen_value]);
|
2004-07-07 05:47:12 +02:00
|
|
|
DPM_gen_id(tdbb, (IPTR) node->nod_arg[e_gen_id], true,
|
2001-05-23 15:26:42 +02:00
|
|
|
MOV_get_int64(desc, 0));
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BUGCHECK(168); /* msg 168 looper: action not yet implemented */
|
|
|
|
}
|
2001-12-29 12:41:29 +01:00
|
|
|
|
2002-04-03 01:12:03 +02:00
|
|
|
} // try
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
2005-01-04 14:09:11 +01:00
|
|
|
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
|
2005-01-04 14:09:11 +01:00
|
|
|
|
|
|
|
// Skip this handling for errors coming from the nested looper calls,
|
|
|
|
// as they're already handled properly. The only need is to undo
|
|
|
|
// our own savepoints.
|
2003-06-28 11:32:46 +02:00
|
|
|
if (catch_disabled) {
|
2005-01-04 14:09:11 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
|
|
|
((save_point) && (save_point_number <= save_point->sav_number));
|
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
|
|
|
verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ERR_punt();
|
2002-04-03 01:12:03 +02:00
|
|
|
}
|
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
// If the database is already bug-checked, then get out
|
2002-04-03 01:12:03 +02:00
|
|
|
if (dbb->dbb_flags & DBB_bugcheck) {
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::status_exception::raise(tdbb->tdbb_status_vector);
|
2002-04-03 01:12:03 +02:00
|
|
|
}
|
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
// Since an error happened, the current savepoint needs to be undone
|
2005-07-25 07:13:01 +02:00
|
|
|
if (transaction != dbb->dbb_sys_trans &&
|
|
|
|
transaction->tra_save_point)
|
|
|
|
{
|
2002-04-03 01:12:03 +02:00
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2002-04-03 01:12:03 +02:00
|
|
|
}
|
|
|
|
|
2003-06-28 11:32:46 +02:00
|
|
|
error_pending = true;
|
2005-01-04 14:09:11 +01:00
|
|
|
catch_disabled = true;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_unwind;
|
2002-04-03 01:12:03 +02:00
|
|
|
request->req_label = 0;
|
2004-08-06 17:26:55 +02:00
|
|
|
|
|
|
|
if (! (tdbb->tdbb_flags & TDBB_stack_trace_done) ) {
|
|
|
|
stuff_stack_trace(request);
|
|
|
|
tdbb->tdbb_flags |= TDBB_stack_trace_done;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-04-03 01:12:03 +02:00
|
|
|
} // while()
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
// If there is no node, assume we have finished processing the
|
|
|
|
// request unless we are in the middle of processing an
|
|
|
|
// asynchronous message
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!node
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
&& !(request->req_flags & req_async_processing)
|
|
|
|
#endif
|
2001-12-29 12:41:29 +01:00
|
|
|
)
|
|
|
|
{
|
2005-01-04 14:09:11 +01:00
|
|
|
// Close active cursors
|
2003-11-02 12:55:17 +01:00
|
|
|
if (request->req_cursors) {
|
2005-12-02 08:35:34 +01:00
|
|
|
for (vec<RecordSource*>::iterator ptr = request->req_cursors->begin(),
|
2003-11-02 12:55:17 +01:00
|
|
|
end = request->req_cursors->end(); ptr < end; ptr++)
|
|
|
|
{
|
|
|
|
if (*ptr)
|
2005-12-02 08:35:34 +01:00
|
|
|
RSE_close(tdbb, *ptr);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags &= ~(req_active | req_reserved);
|
2004-10-30 21:41:54 +02:00
|
|
|
request->req_timestamp.invalidate();
|
2001-05-23 15:26:42 +02:00
|
|
|
release_blobs(tdbb, request);
|
|
|
|
}
|
|
|
|
|
|
|
|
request->req_next = node;
|
|
|
|
tdbb->tdbb_transaction = (tdbb->tdbb_request = old_request) ?
|
|
|
|
old_request->req_transaction : NULL;
|
2004-06-22 22:13:10 +02:00
|
|
|
fb_assert(request->req_caller == old_request);
|
|
|
|
request->req_caller = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
// In the case of a pending error condition (one which did not
|
2001-12-29 12:41:29 +01:00
|
|
|
// result in a exception to the top of looper), we need to
|
|
|
|
// delete the last savepoint
|
2005-01-04 14:09:11 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (error_pending) {
|
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2004-03-18 06:56:06 +01:00
|
|
|
for (const Savepoint* save_point = transaction->tra_save_point;
|
2003-12-31 06:36:12 +01:00
|
|
|
((save_point) && (save_point_number <= save_point->sav_number));
|
|
|
|
save_point = transaction->tra_save_point)
|
|
|
|
{
|
2002-11-03 18:29:51 +01:00
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
2005-01-04 14:09:11 +01:00
|
|
|
verb_cleanup(tdbb, transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ERR_punt();
|
|
|
|
}
|
|
|
|
|
2002-04-03 01:12:03 +02:00
|
|
|
// if the request was aborted, assume that we have already
|
|
|
|
// longjmp'ed to the top of looper, and therefore that the
|
|
|
|
// last savepoint has already been deleted
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2002-04-03 01:12:03 +02:00
|
|
|
if (request->req_flags & req_abort) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_req_sync, 0);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* modify(thread_db* tdbb, jrd_nod* node, SSHORT which_trig)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* m o d i f y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a MODIFY statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
|
|
|
jrd_tra* transaction = request->req_transaction;
|
2004-03-28 11:10:30 +02:00
|
|
|
impure_state* impure = (impure_state*) ((SCHAR *) request + node->nod_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
const SSHORT org_stream = (USHORT)(IPTR) node->nod_arg[e_mod_org_stream];
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* org_rpb = &request->req_rpb[org_stream];
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_rel* relation = org_rpb->rpb_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-06-07 12:52:38 +02:00
|
|
|
if (org_rpb->rpb_number.isBof()) {
|
2005-04-25 22:31:52 +02:00
|
|
|
ERR_post(isc_no_cur_rec, 0);
|
|
|
|
}
|
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
const SSHORT new_stream = (USHORT)(IPTR) node->nod_arg[e_mod_new_stream];
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* new_rpb = &request->req_rpb[new_stream];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
/* for navigational streams, retrieve the rsb */
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = NULL;
|
2003-12-31 06:36:12 +01:00
|
|
|
IRSB irsb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (node->nod_arg[e_mod_rsb]) {
|
2004-03-28 11:10:30 +02:00
|
|
|
rsb = *(RecordSource**) node->nod_arg[e_mod_rsb];
|
2001-05-23 15:26:42 +02:00
|
|
|
irsb = (IRSB) ((UCHAR *) request + rsb->rsb_impure);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if we are on a crack in a navigational stream, modify is an illegal operation */
|
|
|
|
|
|
|
|
if (rsb && EXE_crack(tdbb, rsb, irsb_bof | irsb_eof | irsb_crack)) {
|
|
|
|
EXE_mark_crack(tdbb, rsb, irsb->irsb_flags);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2005-04-18 05:22:31 +02:00
|
|
|
/* If the stream was sorted, the various fields in the rpb are
|
|
|
|
probably junk. Just to make sure that everything is cool,
|
|
|
|
refetch and release the record. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-04-18 05:22:31 +02:00
|
|
|
if (org_rpb->rpb_stream_flags & RPB_s_refetch) {
|
|
|
|
VIO_refetch_record(tdbb, org_rpb, transaction);
|
|
|
|
org_rpb->rpb_stream_flags &= ~RPB_s_refetch;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(false);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (impure->sta_state) {
|
|
|
|
impure->sta_state = 0;
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* org_record = org_rpb->rpb_record;
|
|
|
|
const Record* new_record = new_rpb->rpb_record;
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FASTER(new_record->rec_data, org_record->rec_data,
|
|
|
|
new_record->rec_length);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_arg[e_mod_statement];
|
|
|
|
}
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: This call made here to clear the record in each NULL field and
|
|
|
|
varchar field whose tail may contain garbage. */
|
|
|
|
cleanup_rpb(tdbb, new_rpb);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef PC_ENGINE
|
|
|
|
/* check to see if record locking has been initiated in this database;
|
|
|
|
if so then lock the record for shared write so that normal processing
|
|
|
|
will be able to read or write the record but not when an explicit
|
|
|
|
lock has been taken out */
|
|
|
|
|
2001-12-28 07:31:38 +01:00
|
|
|
LCK_RAII_wrapper implicit_lock;
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
if (!(transaction->tra_flags & TRA_degree3))
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
const Lock* record_locking = RLCK_record_locking(relation);
|
2001-12-24 03:51:06 +01:00
|
|
|
if (record_locking->lck_physical != LCK_PR)
|
|
|
|
{
|
2001-12-28 07:31:38 +01:00
|
|
|
implicit_lock.assign(implicit_record_lock(transaction, org_rpb));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
|
|
|
|
2003-05-05 13:47:45 +02:00
|
|
|
PreModifyEraseTriggers(tdbb, &relation->rel_pre_modify,
|
2005-04-15 16:33:55 +02:00
|
|
|
which_trig, org_rpb, new_rpb->rpb_record,
|
|
|
|
jrd_req::req_trigger_update);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (node->nod_arg[e_mod_validate]) {
|
2001-05-23 15:26:42 +02:00
|
|
|
validate(tdbb, node->nod_arg[e_mod_validate]);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (relation->rel_file)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
EXT_modify(org_rpb, new_rpb,
|
2001-12-24 03:51:06 +01:00
|
|
|
reinterpret_cast<int*>(transaction));
|
|
|
|
}
|
|
|
|
else if (!relation->rel_view_rse)
|
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
USHORT bad_index;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* bad_relation = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-18 21:27:24 +01:00
|
|
|
VIO_modify(tdbb, org_rpb, new_rpb, transaction);
|
2005-03-27 17:01:56 +02:00
|
|
|
const IDX_E error_code =
|
|
|
|
IDX_modify(tdbb, org_rpb, new_rpb, transaction,
|
|
|
|
&bad_relation, &bad_index);
|
2002-10-29 21:20:44 +01:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (error_code) {
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_update_count, bad_relation, true);
|
2001-12-24 03:51:06 +01:00
|
|
|
ERR_duplicate_error(error_code, bad_relation, bad_index);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* trigger;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (relation->rel_post_modify &&
|
|
|
|
which_trig != PRE_TRIG &&
|
|
|
|
(trigger = execute_triggers(tdbb, &relation->rel_post_modify,
|
2005-12-05 12:06:46 +01:00
|
|
|
org_rpb->rpb_record, new_rpb->rpb_record,
|
2003-01-15 13:08:59 +01:00
|
|
|
jrd_req::req_trigger_update)))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_update_count, relation, true);
|
2001-12-24 03:51:06 +01:00
|
|
|
trigger_failure(tdbb, trigger);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* now call IDX_modify_check_constrints after all post modify triggers
|
|
|
|
have fired. This is required for cascading referential integrity,
|
|
|
|
which can be implemented as post_erase triggers */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!relation->rel_file && !relation->rel_view_rse)
|
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
USHORT bad_index;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* bad_relation = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-03-27 17:01:56 +02:00
|
|
|
const IDX_E error_code =
|
|
|
|
IDX_modify_check_constraints(tdbb, org_rpb, new_rpb, transaction,
|
|
|
|
&bad_relation, &bad_index);
|
|
|
|
|
2003-05-04 17:39:42 +02:00
|
|
|
if (error_code) {
|
|
|
|
VIO_bump_count(tdbb, DBB_update_count, relation, true);
|
|
|
|
ERR_duplicate_error(error_code, bad_relation, bad_index);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2001-05-23 15:26:42 +02:00
|
|
|
--transaction->tra_save_point->sav_verb_count;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* if the stream is navigational, we must position the stream on the new
|
|
|
|
record version, but first set the record number */
|
|
|
|
|
|
|
|
new_rpb->rpb_number = org_rpb->rpb_number;
|
2001-12-28 07:31:38 +01:00
|
|
|
if (rsb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RSE_reset_position(tdbb, rsb, new_rpb);
|
2001-12-28 07:31:38 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: Increment the counter only if we called VIO/EXT_modify() and
|
|
|
|
we were successful. */
|
|
|
|
if (!(request->req_view_flags & req_first_modify_return)) {
|
|
|
|
request->req_view_flags |= req_first_modify_return;
|
|
|
|
if (relation->rel_view_rse) {
|
|
|
|
request->req_top_view_modify = relation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (relation == request->req_top_view_modify) {
|
|
|
|
if (which_trig == ALL_TRIGS || which_trig == POST_TRIG) {
|
|
|
|
request->req_records_updated++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (relation->rel_file || !relation->rel_view_rse) {
|
|
|
|
request->req_records_updated++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (which_trig != PRE_TRIG) {
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* org_record = org_rpb->rpb_record;
|
2001-05-23 15:26:42 +02:00
|
|
|
org_rpb->rpb_record = new_rpb->rpb_record;
|
|
|
|
new_rpb->rpb_record = org_record;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
impure->sta_state = 0;
|
2004-03-18 06:56:06 +01:00
|
|
|
RLCK_reserve_relation(tdbb, transaction, relation, true, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Fall thru on evaluate to set up for modify before executing sub-statement.
|
|
|
|
This involves finding the appropriate format, making sure a record block
|
|
|
|
exists for the stream and is big enough, and copying fields from the
|
|
|
|
original record to the new record. */
|
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* new_format = MET_current(tdbb, new_rpb->rpb_relation);
|
2004-08-16 14:28:43 +02:00
|
|
|
Record* new_record = VIO_record(tdbb, new_rpb, new_format, tdbb->getDefaultPool());
|
2001-05-23 15:26:42 +02:00
|
|
|
new_rpb->rpb_address = new_record->rec_data;
|
|
|
|
new_rpb->rpb_length = new_format->fmt_length;
|
|
|
|
new_rpb->rpb_format_number = new_format->fmt_version;
|
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* org_format;
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* org_record = org_rpb->rpb_record;
|
2003-12-31 06:36:12 +01:00
|
|
|
if (!org_record) {
|
2001-05-23 15:26:42 +02:00
|
|
|
org_record =
|
2004-08-16 14:28:43 +02:00
|
|
|
VIO_record(tdbb, org_rpb, new_format, tdbb->getDefaultPool());
|
2001-05-23 15:26:42 +02:00
|
|
|
org_format = org_record->rec_format;
|
|
|
|
org_rpb->rpb_address = org_record->rec_data;
|
|
|
|
org_rpb->rpb_length = org_format->fmt_length;
|
|
|
|
org_rpb->rpb_format_number = org_format->fmt_version;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
org_format = org_record->rec_format;
|
|
|
|
|
|
|
|
/* Copy the original record to the new record. If the format hasn't changed,
|
|
|
|
this is a simple move. If the format has changed, each field must be
|
|
|
|
fetched and moved separately, remembering to set the missing flag. */
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
if (new_format->fmt_version == org_format->fmt_version) {
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FASTER(org_record->rec_data, new_rpb->rpb_address,
|
|
|
|
new_rpb->rpb_length);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
DSC org_desc, new_desc;
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
for (SSHORT i = 0; i < new_format->fmt_count; i++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* In order to "map a null to a default" value (in EVL_field()),
|
|
|
|
* the relation block is referenced.
|
|
|
|
* Reference: Bug 10116, 10424
|
|
|
|
*/
|
|
|
|
CLEAR_NULL(new_record, i);
|
2005-06-04 15:39:30 +02:00
|
|
|
if (EVL_field(new_rpb->rpb_relation, new_record, i, &new_desc)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (EVL_field
|
|
|
|
(org_rpb->rpb_relation, org_record, i,
|
2003-12-31 06:36:12 +01:00
|
|
|
&org_desc))
|
|
|
|
{
|
|
|
|
MOV_move(&org_desc, &new_desc);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
SET_NULL(new_record, i);
|
|
|
|
if (new_desc.dsc_dtype) {
|
2003-12-31 06:36:12 +01:00
|
|
|
UCHAR* p = new_desc.dsc_address;
|
|
|
|
USHORT n = new_desc.dsc_length;
|
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = 0;
|
2003-12-31 06:36:12 +01:00
|
|
|
} while (--n);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
} /* if (org_record) */
|
|
|
|
} /* if (new_record) */
|
|
|
|
} /* for (fmt_count) */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (node->nod_arg[e_mod_map_view]) {
|
|
|
|
impure->sta_state = 1;
|
|
|
|
return node->nod_arg[e_mod_map_view];
|
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_arg[e_mod_statement];
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* receive_msg(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e c e i v e _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a RECEIVE statement. This can be entered either
|
|
|
|
* with "req_evaluate" (ordinary receive statement) or
|
|
|
|
* "req_proceed" (select statement). In the latter case,
|
|
|
|
* the statement isn't every formalled evaluated.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
|
|
|
request->req_operation = jrd_req::req_receive;
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_message = node->nod_arg[e_send_message];
|
|
|
|
request->req_flags |= req_stall;
|
|
|
|
return node;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_proceed:
|
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
return (node->nod_arg[e_send_statement]);
|
|
|
|
|
|
|
|
default:
|
|
|
|
return (node->nod_parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void release_blobs(thread_db* tdbb, jrd_req* request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ b l o b s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release temporary blobs assigned by this request.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(request, type_req);
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_tra* transaction = request->req_transaction;
|
|
|
|
if (transaction) {
|
2001-05-23 15:26:42 +02:00
|
|
|
DEV_BLKCHK(transaction, type_tra);
|
|
|
|
|
2004-06-22 22:13:10 +02:00
|
|
|
/* Release blobs bound to this request */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-22 22:13:10 +02:00
|
|
|
if (request->req_blobs.getFirst()) do {
|
2004-08-21 11:29:46 +02:00
|
|
|
const ULONG blob_temp_id = request->req_blobs.current();
|
2004-06-22 22:13:10 +02:00
|
|
|
if (transaction->tra_blobs.locate(blob_temp_id)) {
|
|
|
|
BlobIndex *current = &transaction->tra_blobs.current();
|
|
|
|
if (current->bli_materialized)
|
|
|
|
transaction->tra_blobs.fastRemove();
|
|
|
|
else {
|
|
|
|
// Blob was created by request, is accounted for internal needs,
|
|
|
|
// but is not materialized. Get rid of it.
|
|
|
|
BLB_cancel(tdbb, current->bli_blob_object);
|
|
|
|
// Since the routine above modifies req_blobs
|
|
|
|
// we need to reestablish accessor position
|
|
|
|
if (request->req_blobs.locate(Firebird::locGreat, blob_temp_id))
|
|
|
|
continue;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Blob accounting inconsistent
|
|
|
|
fb_assert(false);
|
|
|
|
}
|
|
|
|
if (!request->req_blobs.getNext())
|
|
|
|
break;
|
2004-10-04 10:15:00 +02:00
|
|
|
} while (true);
|
2004-06-22 22:13:10 +02:00
|
|
|
|
|
|
|
request->req_blobs.clear();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Release arrays assigned by this request */
|
|
|
|
|
2004-03-19 07:14:53 +01:00
|
|
|
for (ArrayField** array = &transaction->tra_arrays; *array;) {
|
2001-05-23 15:26:42 +02:00
|
|
|
DEV_BLKCHK(*array, type_arr);
|
|
|
|
if ((*array)->arr_request == request)
|
|
|
|
BLB_release_array(*array);
|
|
|
|
else
|
|
|
|
array = &(*array)->arr_next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* release_bookmark(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ b o o k m a r k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Deallocate the passed bookmark.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BKM_release(node->nod_arg[e_relmark_id]);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
static void release_proc_save_points(jrd_req* request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ p r o c _ s a v e _ p o i n t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release temporary blobs assigned by this request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* sav_point = request->req_proc_sav_point;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-13 11:42:10 +01:00
|
|
|
if (request->req_transaction) {
|
|
|
|
while (sav_point) {
|
2004-03-18 06:56:06 +01:00
|
|
|
Savepoint* const temp_sav_point = sav_point->sav_next;
|
2003-12-13 11:42:10 +01:00
|
|
|
delete sav_point;
|
|
|
|
sav_point = temp_sav_point;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
request->req_proc_sav_point = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* seek_rse(thread_db* tdbb, jrd_req* request, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e e k _ r s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a nod_seek, which specifies
|
|
|
|
* a direction and offset in which to
|
|
|
|
* scroll a record selection expression.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_proceed) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* get input arguments */
|
|
|
|
|
2004-08-17 14:28:57 +02:00
|
|
|
const dsc* desc = EVL_expr(tdbb, node->nod_arg[e_seek_direction]);
|
|
|
|
const USHORT direction = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
MOV_get_long(desc, 0) : MAX_USHORT;
|
|
|
|
|
|
|
|
desc = EVL_expr(tdbb, node->nod_arg[e_seek_offset]);
|
|
|
|
const SLONG offset = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
MOV_get_long(desc, 0) : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSelExpr* rse = (RecordSelExpr*) node->nod_arg[e_seek_rse];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
seek_rsb(tdbb, request, rse->rse_rsb, direction, offset);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
static void seek_rsb(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-03-28 11:10:30 +02:00
|
|
|
jrd_req* request, RecordSource* rsb, USHORT direction, SLONG offset)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e e k _ r s b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allow scrolling through a stream as defined
|
|
|
|
* by the input rsb. Handles cracks, refresh
|
|
|
|
* ranges, and multiple seeking. Uses RSE_get_record ()
|
|
|
|
* to do the actual work.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
DEV_BLKCHK(rsb, type_rsb);
|
2003-12-31 06:36:12 +01:00
|
|
|
irsb* impure = (IRSB) ((UCHAR *) request + rsb->rsb_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* look past any boolean to the actual stream */
|
|
|
|
|
|
|
|
if (rsb->rsb_type == rsb_boolean) {
|
|
|
|
seek_rsb(tdbb, request, rsb->rsb_next, direction, offset);
|
|
|
|
|
|
|
|
/* set the backwards flag */
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
const irsb* next_impure =
|
|
|
|
(IRSB) ((UCHAR *) request + rsb->rsb_next->rsb_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (next_impure->irsb_flags & irsb_last_backwards)
|
|
|
|
impure->irsb_flags |= irsb_last_backwards;
|
|
|
|
else
|
|
|
|
impure->irsb_flags &= ~irsb_last_backwards;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* do simple boundary checking for bof and eof */
|
|
|
|
|
|
|
|
switch (direction) {
|
|
|
|
case blr_forward:
|
|
|
|
if (impure->irsb_flags & irsb_eof)
|
|
|
|
ERR_post(isc_stream_eof, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_backward:
|
|
|
|
if (impure->irsb_flags & irsb_bof)
|
|
|
|
ERR_post(isc_stream_bof, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_bof_forward:
|
|
|
|
case blr_eof_backward:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-08-20 10:25:45 +02:00
|
|
|
// was: BUGCHECK(232);
|
|
|
|
// replaced with this error to be consistent with find()
|
|
|
|
ERR_post(isc_invalid_direction, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* the actual offset to seek may be one less because the next time
|
|
|
|
through the blr_for loop we will seek one record--flag the fact
|
|
|
|
that a fetch is required on this stream in case it doesn't happen
|
|
|
|
(for example when GPRE generates BLR which does not stall prior to
|
|
|
|
the blr_for, as DSQL does) */
|
|
|
|
|
|
|
|
if (offset > 0)
|
|
|
|
switch (direction) {
|
|
|
|
case blr_forward:
|
|
|
|
case blr_bof_forward:
|
|
|
|
if (!(impure->irsb_flags & irsb_last_backwards)) {
|
|
|
|
offset--;
|
|
|
|
if (!(impure->irsb_flags & irsb_bof))
|
|
|
|
request->req_flags |= req_fetch_required;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_backward:
|
|
|
|
case blr_eof_backward:
|
|
|
|
if (impure->irsb_flags & irsb_last_backwards) {
|
|
|
|
offset--;
|
|
|
|
if (!(impure->irsb_flags & irsb_eof))
|
|
|
|
request->req_flags |= req_fetch_required;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* now do the actual seek */
|
|
|
|
|
|
|
|
switch (direction) {
|
|
|
|
case blr_forward: /* go forward from the current location */
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
if ((offset == 1) && request->req_begin_ranges)
|
|
|
|
impure->irsb_flags |= irsb_refresh;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* the rsb_backwards flag is used to indicate the direction to seek in;
|
|
|
|
this is sticky in the sense that after the user has seek'ed in the
|
|
|
|
backward direction, the next retrieval from a blr_for loop will also
|
|
|
|
be in the backward direction--this allows us to continue scrolling
|
|
|
|
without constantly sending messages to the engine */
|
|
|
|
|
|
|
|
impure->irsb_flags &= ~irsb_last_backwards;
|
|
|
|
|
|
|
|
while (offset) {
|
|
|
|
offset--;
|
|
|
|
if (!(RSE_get_record(tdbb, rsb, RSE_get_next)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_backward: /* go backward from the current location */
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
if ((offset == 1) && request->req_begin_ranges)
|
|
|
|
impure->irsb_flags |= irsb_refresh;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
impure->irsb_flags |= irsb_last_backwards;
|
|
|
|
|
|
|
|
while (offset) {
|
|
|
|
offset--;
|
|
|
|
if (!(RSE_get_record(tdbb, rsb, RSE_get_next)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_bof_forward: /* go forward from the beginning of the stream */
|
|
|
|
|
|
|
|
RSE_close(tdbb, rsb);
|
|
|
|
RSE_open(tdbb, rsb);
|
|
|
|
|
|
|
|
impure->irsb_flags &= ~irsb_last_backwards;
|
|
|
|
|
|
|
|
while (offset) {
|
|
|
|
offset--;
|
|
|
|
if (!(RSE_get_record(tdbb, rsb, RSE_get_next)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_eof_backward: /* go backward from the end of the stream */
|
|
|
|
|
|
|
|
RSE_close(tdbb, rsb);
|
|
|
|
RSE_open(tdbb, rsb);
|
|
|
|
|
|
|
|
/* if this is a stream type which uses bof and eof flags,
|
|
|
|
reverse the sense of bof and eof in this case */
|
|
|
|
|
|
|
|
if (impure->irsb_flags & irsb_bof) {
|
|
|
|
impure->irsb_flags &= ~irsb_bof;
|
|
|
|
impure->irsb_flags |= irsb_eof;
|
|
|
|
}
|
|
|
|
|
|
|
|
impure->irsb_flags |= irsb_last_backwards;
|
|
|
|
|
|
|
|
while (offset) {
|
|
|
|
offset--;
|
|
|
|
if (!(RSE_get_record(tdbb, rsb, RSE_get_next)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2004-08-20 10:25:45 +02:00
|
|
|
// Should never go here, because of the boundary
|
|
|
|
// check above, but anyway...
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(232);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
impure->irsb_flags &= ~irsb_refresh;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* selct(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e l e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a SELECT statement. This is more than a little
|
|
|
|
* obscure. We first set up the SELECT statement as the
|
|
|
|
* "message" and stall on receive (waiting for user send).
|
|
|
|
* EXE_send will then loop thru the sub-statements of select
|
|
|
|
* looking for the appropriate RECEIVE statement. When (or if)
|
|
|
|
* it finds it, it will set it up the next statement to be
|
|
|
|
* executed. The RECEIVE, then, will be entered with the
|
|
|
|
* operation "req_proceed."
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_message = node;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_receive;
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags |= req_stall;
|
|
|
|
return node;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* send_msg(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e n d _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a SEND statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
return (node->nod_arg[e_send_statement]);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
|
|
|
request->req_operation = jrd_req::req_send;
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_message = node->nod_arg[e_send_message];
|
|
|
|
request->req_flags |= req_stall;
|
|
|
|
return node;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_proceed:
|
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_parent;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return (node->nod_parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* set_bookmark(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t _ b o o k m a r k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set a stream to the location of the
|
|
|
|
* specified bookmark.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2004-03-19 07:14:53 +01:00
|
|
|
Bookmark* bookmark = BKM_lookup(node->nod_arg[e_setmark_id]);
|
2003-12-31 06:36:12 +01:00
|
|
|
const USHORT stream = (USHORT)(ULONG) node->nod_arg[e_setmark_stream];
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb = &request->req_rpb[stream];
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = *((RecordSource**) node->nod_arg[e_setmark_rsb]);
|
2003-12-31 06:36:12 +01:00
|
|
|
irsb* impure = (IRSB) ((UCHAR *) request + rsb->rsb_impure);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* check if the bookmark was at beginning or end of file
|
|
|
|
and flag the rsb accordingly */
|
|
|
|
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, 0);
|
|
|
|
if (bookmark->bkm_flags & bkm_bof)
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, irsb_bof);
|
|
|
|
else if (bookmark->bkm_flags & bkm_eof)
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, irsb_eof);
|
|
|
|
else if (bookmark->bkm_flags & bkm_crack) {
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, irsb_crack);
|
|
|
|
if (bookmark->bkm_flags & bkm_forced_crack)
|
|
|
|
RSE_MARK_CRACK(tdbb, rsb, irsb_forced_crack);
|
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
if (!RSE_set_bookmark(tdbb, rsb, rpb, bookmark)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
EXE_mark_crack(tdbb, rsb,
|
|
|
|
impure->irsb_flags & (irsb_crack | irsb_eof |
|
|
|
|
irsb_bof));
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void set_error(thread_db* tdbb, const xcp_repeat* exception, jrd_nod* msg_node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set status vector according to specified error condition
|
|
|
|
* and jump to handle error accordingly.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName name, relation_name;
|
2004-11-18 14:58:46 +01:00
|
|
|
TEXT message[XCP_MESSAGE_LENGTH + 1];
|
|
|
|
|
|
|
|
// since temp used as vary, we need size of vary::vary_length
|
|
|
|
// (USHORT) extra chars
|
|
|
|
TEXT temp[XCP_MESSAGE_LENGTH + sizeof(USHORT)];
|
2003-11-02 12:55:17 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_TDBB(tdbb);
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2003-05-26 17:45:29 +02:00
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
if (!exception) {
|
|
|
|
// retrieve the status vector and punt
|
2003-11-05 10:02:33 +01:00
|
|
|
request->req_last_xcp.copyTo(tdbb->tdbb_status_vector);
|
2003-11-02 12:55:17 +01:00
|
|
|
request->req_last_xcp.clear();
|
|
|
|
ERR_punt();
|
2002-09-28 16:04:35 +02:00
|
|
|
}
|
2003-11-02 12:55:17 +01:00
|
|
|
|
|
|
|
USHORT length = 0;
|
|
|
|
|
|
|
|
if (msg_node)
|
2002-09-28 16:04:35 +02:00
|
|
|
{
|
2002-10-29 17:27:47 +01:00
|
|
|
const char* string = 0;
|
2003-11-02 12:55:17 +01:00
|
|
|
// evaluate exception message and convert it to string
|
|
|
|
DSC* desc = EVL_expr(tdbb, msg_node);
|
2003-05-26 17:45:29 +02:00
|
|
|
if (desc && !(request->req_flags & req_null))
|
2002-12-11 10:39:41 +01:00
|
|
|
{
|
|
|
|
length = MOV_make_string(desc,
|
2005-05-28 00:45:31 +02:00
|
|
|
tdbb->tdbb_attachment->att_charset,
|
2002-12-11 10:39:41 +01:00
|
|
|
&string,
|
2004-02-20 07:43:27 +01:00
|
|
|
reinterpret_cast<vary*>(temp),
|
2002-12-11 10:39:41 +01:00
|
|
|
sizeof(temp));
|
|
|
|
length = MIN(length, sizeof(message) - 1);
|
2002-10-19 11:05:37 +02:00
|
|
|
|
2002-12-11 10:39:41 +01:00
|
|
|
/* dimitr: or should we throw an error here, i.e.
|
|
|
|
replace the above assignment with the following lines:
|
2002-10-19 11:05:37 +02:00
|
|
|
|
2002-12-11 10:39:41 +01:00
|
|
|
if (length > sizeof(message) - 1)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_imp_exc, isc_arg_gds, isc_blktoobig, 0);
|
2002-12-11 10:39:41 +01:00
|
|
|
*/
|
2002-10-19 11:05:37 +02:00
|
|
|
|
2002-12-11 10:39:41 +01:00
|
|
|
memcpy(message, string, length);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
length = 0;
|
|
|
|
}
|
2002-09-28 16:04:35 +02:00
|
|
|
}
|
|
|
|
message[length] = 0;
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
const TEXT* s;
|
|
|
|
|
2005-05-22 08:13:54 +02:00
|
|
|
switch (exception->xcp_type)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
case xcp_sql_code:
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_sqlerr, isc_arg_number, exception->xcp_code, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case xcp_gds_code:
|
2003-11-11 13:19:20 +01:00
|
|
|
if (exception->xcp_code == isc_check_constraint) {
|
2001-05-23 15:26:42 +02:00
|
|
|
MET_lookup_cnstrt_for_trigger(tdbb, name, relation_name,
|
|
|
|
request->req_trg_name);
|
2003-11-02 12:55:17 +01:00
|
|
|
ERR_post(exception->xcp_code,
|
2005-05-12 20:28:04 +02:00
|
|
|
isc_arg_string, ERR_cstring(name.c_str()),
|
|
|
|
isc_arg_string, ERR_cstring(relation_name.c_str()), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
2003-11-02 12:55:17 +01:00
|
|
|
ERR_post(exception->xcp_code, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case xcp_xcp_code:
|
2005-05-22 08:13:54 +02:00
|
|
|
// CVC: If we have the exception name, use it instead of the number.
|
|
|
|
// Solves SF Bug #494981.
|
2005-08-14 14:41:16 +02:00
|
|
|
MET_lookup_exception(tdbb, exception->xcp_code,
|
|
|
|
name, temp, sizeof(temp));
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (message[0])
|
|
|
|
s = message;
|
2002-09-28 16:04:35 +02:00
|
|
|
else if (temp[0])
|
|
|
|
s = temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
s = NULL;
|
2005-08-14 14:41:16 +02:00
|
|
|
|
2005-10-22 13:09:36 +02:00
|
|
|
if (s && name.length())
|
|
|
|
ERR_post(isc_except, isc_arg_number, exception->xcp_code,
|
|
|
|
isc_arg_gds, isc_random, isc_arg_string, ERR_cstring(name.c_str()),
|
2005-08-14 08:56:24 +02:00
|
|
|
isc_arg_gds, isc_random, isc_arg_string, ERR_cstring(s),
|
|
|
|
0);
|
2005-10-22 13:09:36 +02:00
|
|
|
else if (s)
|
|
|
|
ERR_post(isc_except, isc_arg_number, exception->xcp_code,
|
|
|
|
isc_arg_gds, isc_random, isc_arg_string, ERR_cstring(s),
|
|
|
|
0);
|
|
|
|
else if (name.length())
|
|
|
|
ERR_post(isc_except, isc_arg_number, exception->xcp_code,
|
|
|
|
isc_arg_gds, isc_random, isc_arg_string, ERR_cstring(name.c_str()),
|
|
|
|
0);
|
|
|
|
else
|
|
|
|
ERR_post(isc_except, isc_arg_number, exception->xcp_code, 0);
|
2005-08-14 14:41:16 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
fb_assert(false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* set_index(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t _ i n d e x
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a SET INDEX statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
if (request->req_operation == jrd_req::req_evaluate) {
|
2003-12-31 06:36:12 +01:00
|
|
|
const USHORT stream = (USHORT)(ULONG) node->nod_arg[e_index_stream];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb = &request->req_rpb[stream];
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if id is non-zero, get the index definition;
|
|
|
|
otherwise it indicates revert to natural order */
|
|
|
|
|
2004-08-17 14:28:57 +02:00
|
|
|
const dsc* desc = EVL_expr(tdbb, node->nod_arg[e_index_index]);
|
|
|
|
const USHORT id = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
MOV_get_long(desc, 0) : 0;
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc idx;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (id && BTR_lookup(tdbb, relation, id - 1, &idx))
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_indexnotdefined, isc_arg_string, relation->rel_name,
|
|
|
|
isc_arg_number, (SLONG) id, 0);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* generate a new rsb in place of the old */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
RSE_close(tdbb, *(RecordSource**) node->nod_arg[e_index_rsb]);
|
|
|
|
OPT_set_index(tdbb, request, (RecordSource**) node->nod_arg[e_index_rsb],
|
2003-09-01 09:58:04 +02:00
|
|
|
relation, id ? &idx : NULL);
|
2004-03-28 11:10:30 +02:00
|
|
|
RSE_open(tdbb, *(RecordSource**) node->nod_arg[e_index_rsb]);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* stall(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t a l l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a stall statement.
|
|
|
|
* This is like a blr_receive, except that there is no
|
|
|
|
* need for a gds__send () from the user (i.e. EXE_send () in the engine).
|
|
|
|
* A gds__receive () will unblock the user.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_sync:
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_parent;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_proceed:
|
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_parent;
|
|
|
|
|
|
|
|
default:
|
|
|
|
request->req_message = node;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_flags |= req_stall;
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* store(thread_db* tdbb, jrd_nod* node, SSHORT which_trig)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t o r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a STORE statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_req* trigger;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2005-03-27 17:01:56 +02:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
|
|
|
jrd_tra* transaction = request->req_transaction;
|
|
|
|
impure_state* impure = (impure_state*) ((SCHAR *) request + node->nod_impure);
|
|
|
|
SSHORT stream = (USHORT)(IPTR) node->nod_arg[e_sto_relation]->nod_arg[e_rel_stream];
|
|
|
|
record_param* rpb = &request->req_rpb[stream];
|
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2005-06-24 14:56:34 +02:00
|
|
|
if (request->req_records_affected.isReadOnly() &&
|
|
|
|
!request->req_records_affected.hasCursor())
|
|
|
|
{
|
|
|
|
request->req_records_affected.clear();
|
|
|
|
}
|
|
|
|
request->req_records_affected.bumpModified(false);
|
2001-05-23 15:26:42 +02:00
|
|
|
impure->sta_state = 0;
|
2004-03-18 06:56:06 +01:00
|
|
|
RLCK_reserve_relation(tdbb, transaction, relation, true, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (impure->sta_state)
|
|
|
|
return node->nod_parent;
|
|
|
|
|
|
|
|
if (transaction != dbb->dbb_sys_trans)
|
|
|
|
++transaction->tra_save_point->sav_verb_count;
|
|
|
|
|
|
|
|
if (relation->rel_pre_store &&
|
|
|
|
(which_trig != POST_TRIG) &&
|
|
|
|
(trigger = execute_triggers(tdbb, &relation->rel_pre_store,
|
2005-12-05 12:06:46 +01:00
|
|
|
NULL, rpb->rpb_record,
|
|
|
|
jrd_req::req_trigger_insert)))
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
|
|
|
trigger_failure(tdbb, trigger);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
if (node->nod_arg[e_sto_validate]) {
|
2001-05-23 15:26:42 +02:00
|
|
|
validate(tdbb, node->nod_arg[e_sto_validate]);
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* For optimum on-disk record compression, zero all unassigned
|
|
|
|
fields. In addition, zero the tail of assigned varying fields
|
|
|
|
so that previous remnants don't defeat compression efficiency. */
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: The code that was here was moved to its own routine: cleanup_rpb()
|
|
|
|
and replaced by the call shown above. */
|
|
|
|
|
|
|
|
cleanup_rpb(tdbb, rpb);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
if (relation->rel_file) {
|
2004-02-20 07:43:27 +01:00
|
|
|
EXT_store(rpb, reinterpret_cast<int*>(transaction));
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
|
|
|
else if (!relation->rel_view_rse)
|
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
USHORT bad_index;
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* bad_relation = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
VIO_store(tdbb, rpb, transaction);
|
2005-03-27 17:01:56 +02:00
|
|
|
const IDX_E error_code =
|
|
|
|
IDX_store(tdbb, rpb, transaction,
|
|
|
|
&bad_relation, &bad_index);
|
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
if (error_code) {
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_insert_count, bad_relation, true);
|
2001-12-29 12:41:29 +01:00
|
|
|
ERR_duplicate_error(error_code, bad_relation, bad_index);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (relation->rel_post_store &&
|
|
|
|
(which_trig != PRE_TRIG) &&
|
|
|
|
(trigger = execute_triggers(tdbb, &relation->rel_post_store,
|
2005-12-05 12:06:46 +01:00
|
|
|
NULL, rpb->rpb_record,
|
|
|
|
jrd_req::req_trigger_insert)))
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2003-05-04 17:39:42 +02:00
|
|
|
VIO_bump_count(tdbb, DBB_insert_count, relation, true);
|
2001-12-29 12:41:29 +01:00
|
|
|
trigger_failure(tdbb, trigger);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: Increment the counter only if we called VIO/EXT_store() and
|
|
|
|
we were successful. */
|
|
|
|
if (!(request->req_view_flags & req_first_store_return)) {
|
|
|
|
request->req_view_flags |= req_first_store_return;
|
|
|
|
if (relation->rel_view_rse) {
|
|
|
|
request->req_top_view_store = relation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (relation == request->req_top_view_store) {
|
|
|
|
if (which_trig == ALL_TRIGS || which_trig == POST_TRIG) {
|
|
|
|
request->req_records_inserted++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (relation->rel_file || !relation->rel_view_rse) {
|
|
|
|
request->req_records_inserted++;
|
2005-06-24 14:56:34 +02:00
|
|
|
request->req_records_affected.bumpModified(true);
|
2002-06-30 11:58:20 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-29 12:41:29 +01:00
|
|
|
if (transaction != dbb->dbb_sys_trans) {
|
2001-05-23 15:26:42 +02:00
|
|
|
--transaction->tra_save_point->sav_verb_count;
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (node->nod_arg[e_sto_statement2]) {
|
|
|
|
impure->sta_state = 1;
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_evaluate;
|
2001-05-23 15:26:42 +02:00
|
|
|
return node->nod_arg[e_sto_statement2];
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return node->nod_parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fall thru on evaluate to set up for store before executing sub-statement.
|
|
|
|
This involves finding the appropriate format, making sure a record block
|
|
|
|
exists for the stream and is big enough, and initialize all null flags
|
|
|
|
to "missing." */
|
|
|
|
|
2004-03-30 06:10:52 +02:00
|
|
|
const Format* format = MET_current(tdbb, relation);
|
2005-12-05 12:06:46 +01:00
|
|
|
Record* record = VIO_record(tdbb, rpb, format, tdbb->getDefaultPool());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
rpb->rpb_address = record->rec_data;
|
|
|
|
rpb->rpb_length = format->fmt_length;
|
|
|
|
rpb->rpb_format_number = format->fmt_version;
|
|
|
|
|
2002-06-30 11:58:20 +02:00
|
|
|
/* CVC: This small block added by Ann Harrison to
|
|
|
|
start with a clean empty buffer and so avoid getting
|
|
|
|
new record buffer with misleading information. Fixes
|
|
|
|
bug with incorrect blob sharing during insertion in
|
|
|
|
a stored procedure. */
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
UCHAR* p = record->rec_data;
|
|
|
|
{ // scope
|
|
|
|
const UCHAR* const data_end = p + rpb->rpb_length;
|
2002-06-30 11:58:20 +02:00
|
|
|
while (p < data_end)
|
|
|
|
*p++ = 0;
|
2003-12-31 06:36:12 +01:00
|
|
|
} // scope
|
2002-06-30 11:58:20 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Initialize all fields to missing */
|
|
|
|
|
|
|
|
p = record->rec_data;
|
2003-12-31 06:36:12 +01:00
|
|
|
SSHORT n = (format->fmt_count + 7) >> 3;
|
2001-12-29 12:41:29 +01:00
|
|
|
if (n) {
|
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = 0xff;
|
2001-12-29 12:41:29 +01:00
|
|
|
} while (--n);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return node->nod_arg[e_sto_statement];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
2004-03-11 06:04:26 +01:00
|
|
|
static jrd_nod* stream(thread_db* tdbb, jrd_nod* node)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t r e a m
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a STREAM statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(node, type_nod);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
RecordSource* rsb = ((RecordSelExpr*) node)->rse_rsb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (request->req_operation) {
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_evaluate:
|
2001-05-23 15:26:42 +02:00
|
|
|
RSE_open(tdbb, rsb);
|
2002-11-21 00:18:16 +01:00
|
|
|
request->req_operation = jrd_req::req_return;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
case jrd_req::req_return:
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->nod_parent;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
RSE_close(tdbb, rsb);
|
|
|
|
node = node->nod_parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static bool test_and_fixup_error(thread_db* tdbb, const PsqlException* conditions,
|
|
|
|
jrd_req* request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-09-28 16:04:35 +02:00
|
|
|
* t e s t _ a n d _ f i x u p _ e r r o r
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Test for match of current state with list of error conditions.
|
2002-09-28 16:04:35 +02:00
|
|
|
* Fix type and code of the exception.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
ISC_STATUS* status_vector = tdbb->tdbb_status_vector;
|
2003-12-31 06:36:12 +01:00
|
|
|
const SSHORT sqlcode = gds__sqlcode(status_vector);
|
2002-09-28 16:04:35 +02:00
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
bool found = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
for (USHORT i = 0; i < conditions->xcp_count; i++)
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
|
|
|
switch (conditions->xcp_rpt[i].xcp_type)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
case xcp_sql_code:
|
2002-09-28 16:04:35 +02:00
|
|
|
if (sqlcode == conditions->xcp_rpt[i].xcp_code)
|
|
|
|
{
|
2003-11-02 12:55:17 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case xcp_gds_code:
|
2002-09-28 16:04:35 +02:00
|
|
|
if (status_vector[1] == conditions->xcp_rpt[i].xcp_code)
|
|
|
|
{
|
2003-11-02 12:55:17 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case xcp_xcp_code:
|
2005-08-14 14:41:16 +02:00
|
|
|
{
|
|
|
|
// Look at set_error() routine to understand how the
|
|
|
|
// exception ID info is encoded inside the status vector.
|
2003-11-11 13:19:20 +01:00
|
|
|
if ((status_vector[1] == isc_except) &&
|
2005-10-22 13:09:36 +02:00
|
|
|
(status_vector[3] == conditions->xcp_rpt[i].xcp_code))
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2003-11-02 12:55:17 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2005-08-14 14:41:16 +02:00
|
|
|
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case xcp_default:
|
2003-11-02 12:55:17 +01:00
|
|
|
found = true;
|
|
|
|
break;
|
2005-08-14 14:41:16 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
fb_assert(false);
|
2003-11-02 12:55:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (found)
|
|
|
|
{
|
|
|
|
request->req_last_xcp.init(status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[0] = 0;
|
|
|
|
status_vector[1] = 0;
|
2003-11-02 12:55:17 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-09-28 16:04:35 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-02 12:55:17 +01:00
|
|
|
return found;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void trigger_failure(thread_db* tdbb, jrd_req* trigger)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t r i g g e r _ f a i l u r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Trigger failed, report error.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
EXE_unwind(tdbb, trigger);
|
2001-12-29 12:41:29 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
trigger->req_attachment = NULL;
|
|
|
|
trigger->req_flags &= ~req_in_use;
|
2004-10-30 21:41:54 +02:00
|
|
|
trigger->req_timestamp.invalidate();
|
2001-12-29 12:41:29 +01:00
|
|
|
|
|
|
|
if (trigger->req_flags & req_leave)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
trigger->req_flags &= ~req_leave;
|
2002-10-29 17:27:47 +01:00
|
|
|
const TEXT* msg;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (trigger->req_trg_name &&
|
2001-12-29 12:41:29 +01:00
|
|
|
(msg = MET_trigger_msg(tdbb,
|
|
|
|
trigger->req_trg_name,
|
|
|
|
trigger->req_label)))
|
|
|
|
{
|
|
|
|
if (trigger->req_flags & req_sys_trigger)
|
|
|
|
{
|
2003-04-10 08:49:16 +02:00
|
|
|
ISC_STATUS code = PAR_symbol_to_gdscode(msg);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (code)
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_integ_fail,
|
|
|
|
isc_arg_number, (SLONG) trigger->req_label,
|
|
|
|
isc_arg_gds, code, 0);
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_integ_fail,
|
|
|
|
isc_arg_number, (SLONG) trigger->req_label,
|
|
|
|
isc_arg_gds, isc_random, isc_arg_string, msg, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_integ_fail, isc_arg_number,
|
2001-05-23 15:26:42 +02:00
|
|
|
(SLONG) trigger->req_label, 0);
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
2001-12-29 12:41:29 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_punt();
|
2001-12-29 12:41:29 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void validate(thread_db* tdbb, jrd_nod* list)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* v a l i d a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a list of validation expressions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
BLKCHK(list, type_nod);
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
jrd_nod** ptr1 = list->nod_arg;
|
|
|
|
for (const jrd_nod* const* const end = ptr1 + list->nod_count;
|
2004-01-13 10:52:19 +01:00
|
|
|
ptr1 < end; ptr1++)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
if (!EVL_boolean(tdbb, (*ptr1)->nod_arg[e_val_boolean]))
|
|
|
|
{
|
|
|
|
/* Validation error -- report result */
|
2004-08-20 10:25:45 +02:00
|
|
|
const char* value;
|
|
|
|
TEXT temp[128];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_nod* node = (*ptr1)->nod_arg[e_val_value];
|
|
|
|
jrd_req* request = tdbb->tdbb_request;
|
2004-08-17 14:28:57 +02:00
|
|
|
const dsc* desc = EVL_expr(tdbb, node);
|
|
|
|
const USHORT length = (desc && !(request->req_flags & req_null)) ?
|
|
|
|
MOV_make_string(desc, ttype_dynamic, &value,
|
|
|
|
reinterpret_cast<vary*>(temp),
|
|
|
|
sizeof(temp)) : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-03-11 09:34:20 +01:00
|
|
|
if (!desc || (request->req_flags & req_null))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
value = "*** null ***";
|
|
|
|
}
|
|
|
|
else if (!length)
|
|
|
|
{
|
|
|
|
value = "";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
value = ERR_string(value, length);
|
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
const TEXT* name = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (node->nod_type == nod_field)
|
|
|
|
{
|
2004-01-21 08:18:30 +01:00
|
|
|
const USHORT stream = (USHORT)(IPTR) node->nod_arg[e_fld_stream];
|
|
|
|
const USHORT id = (USHORT)(IPTR) node->nod_arg[e_fld_id];
|
2003-12-31 06:36:12 +01:00
|
|
|
const jrd_rel* relation = request->req_rpb[stream].rpb_relation;
|
|
|
|
|
|
|
|
const jrd_fld* field;
|
2005-12-02 08:35:34 +01:00
|
|
|
const vec<jrd_fld*>* vector = relation->rel_fields;
|
2003-12-31 06:36:12 +01:00
|
|
|
if (vector && id < vector->count() &&
|
2005-12-02 08:35:34 +01:00
|
|
|
(field = (*vector)[id]))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
name = field->fld_name.c_str();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
{
|
|
|
|
name = "*** unknown ***";
|
|
|
|
}
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_not_valid, isc_arg_string, name,
|
|
|
|
isc_arg_string, value, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-04 14:09:11 +01:00
|
|
|
|
|
|
|
inline void verb_cleanup(thread_db* tdbb, jrd_tra* transaction)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* v e r b _ c l e a n u p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If an error happens during the backout of a savepoint, then the transaction
|
|
|
|
* must be marked 'dead' because that is the only way to clean up after a
|
|
|
|
* failed backout. The easiest way to do this is to kill the application
|
|
|
|
* by calling bugcheck.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
try {
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
}
|
|
|
|
catch (const std::exception&) {
|
|
|
|
if (tdbb->tdbb_database->dbb_flags & DBB_bugcheck) {
|
|
|
|
Firebird::status_exception::raise(tdbb->tdbb_status_vector);
|
|
|
|
}
|
|
|
|
BUGCHECK(290); // msg 290 error during savepoint backout
|
|
|
|
}
|
|
|
|
}
|