2001-05-23 15:26:42 +02:00
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// PROGRAM: C preprocess
|
|
|
|
// MODULE: c_cxx.cpp
|
|
|
|
// DESCRIPTION: C and C++ code generator
|
|
|
|
//
|
|
|
|
// 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.
|
2003-09-16 16:01:56 +02:00
|
|
|
// Contributor(s): ______________________________________.16/09/2003
|
2001-05-23 15:26:42 +02:00
|
|
|
// TMN (Mike Nordell) 11.APR.2001 - Reduce compiler warnings
|
|
|
|
//
|
2002-10-29 04:17:45 +01:00
|
|
|
// 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
|
|
|
|
//
|
2001-05-23 15:26:42 +02:00
|
|
|
//
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
2003-09-29 14:43:14 +02:00
|
|
|
// $Id: c_cxx.cpp,v 1.32 2003-09-29 12:43:03 robocop Exp $
|
2001-05-23 15:26:42 +02:00
|
|
|
//
|
|
|
|
|
2001-07-30 01:43:24 +02:00
|
|
|
#include "firebird.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/ib_stdio.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include <stdarg.h>
|
2001-07-30 01:43:24 +02:00
|
|
|
#include "../jrd/gds.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "gpre.h"
|
|
|
|
#include "../gpre/pat.h"
|
|
|
|
#include "../gpre/cmp_proto.h"
|
|
|
|
#include "../gpre/gpre_proto.h"
|
|
|
|
#include "../gpre/lang_proto.h"
|
|
|
|
#include "../gpre/pat_proto.h"
|
|
|
|
#include "../gpre/prett_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
|
|
|
|
|
|
|
|
extern TEXT *transaction_name;
|
|
|
|
|
|
|
|
static void align(int);
|
|
|
|
static void asgn_from(ACT, REF, int);
|
|
|
|
static void asgn_to(ACT, REF, int);
|
|
|
|
static void asgn_to_proc(REF, int);
|
|
|
|
static void gen_any(ACT, int);
|
|
|
|
static void gen_at_end(ACT, int);
|
|
|
|
static void gen_based(ACT, int);
|
|
|
|
static void gen_blob_close(ACT, USHORT);
|
|
|
|
static void gen_blob_end(ACT, USHORT);
|
|
|
|
static void gen_blob_for(ACT, USHORT);
|
|
|
|
static void gen_blob_open(ACT, USHORT);
|
2003-09-28 23:36:05 +02:00
|
|
|
static void gen_blr(void*, SSHORT, const char*);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_clear_handles(ACT, int);
|
|
|
|
static void gen_compatibility_symbol(TEXT *, TEXT *, TEXT *);
|
|
|
|
static void gen_compile(ACT, int);
|
|
|
|
static void gen_create_database(ACT, int);
|
2002-11-17 01:04:19 +01:00
|
|
|
static int gen_cursor_close(ACT, GPRE_REQ, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_cursor_init(ACT, int);
|
2002-11-17 01:04:19 +01:00
|
|
|
static int gen_cursor_open(ACT, GPRE_REQ, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_database(ACT, int);
|
|
|
|
static void gen_ddl(ACT, int);
|
|
|
|
static void gen_drop_database(ACT, int);
|
|
|
|
static void gen_dyn_close(ACT, int);
|
|
|
|
static void gen_dyn_declare(ACT, int);
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_dyn_describe(ACT, int, bool);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_dyn_execute(ACT, int);
|
|
|
|
static void gen_dyn_fetch(ACT, int);
|
|
|
|
static void gen_dyn_immediate(ACT, int);
|
|
|
|
static void gen_dyn_insert(ACT, int);
|
|
|
|
static void gen_dyn_open(ACT, int);
|
|
|
|
static void gen_dyn_prepare(ACT, int);
|
|
|
|
static void gen_emodify(ACT, int);
|
|
|
|
static void gen_estore(ACT, int);
|
|
|
|
static void gen_endfor(ACT, int);
|
|
|
|
static void gen_erase(ACT, int);
|
|
|
|
static SSHORT gen_event_block(ACT);
|
|
|
|
static void gen_event_init(ACT, int);
|
|
|
|
static void gen_event_wait(ACT, int);
|
|
|
|
static void gen_fetch(ACT, int);
|
|
|
|
static void gen_finish(ACT, int);
|
|
|
|
static void gen_for(ACT, int);
|
|
|
|
static void gen_function(ACT, int);
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_get_or_put_slice(ACT, REF, bool, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_get_segment(ACT, int);
|
|
|
|
static void gen_loop(ACT, int);
|
2003-09-10 21:48:53 +02:00
|
|
|
static TEXT *gen_name(char *, REF, bool);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_on_error(ACT, USHORT);
|
|
|
|
static void gen_procedure(ACT, int);
|
|
|
|
static void gen_put_segment(ACT, int);
|
|
|
|
static void gen_raw(UCHAR *, int);
|
|
|
|
static void gen_ready(ACT, int);
|
|
|
|
static void gen_receive(ACT, int, POR);
|
|
|
|
static void gen_release(ACT, int);
|
2002-11-17 01:04:19 +01:00
|
|
|
static void gen_request(GPRE_REQ);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_return_value(ACT, int);
|
|
|
|
static void gen_routine(ACT, int);
|
|
|
|
static void gen_s_end(ACT, int);
|
|
|
|
static void gen_s_fetch(ACT, int);
|
|
|
|
static void gen_s_start(ACT, int);
|
|
|
|
static void gen_segment(ACT, int);
|
|
|
|
static void gen_select(ACT, int);
|
|
|
|
static void gen_send(ACT, POR, int);
|
|
|
|
static void gen_slice(ACT, REF, int);
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_start(ACT, POR, int, bool);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void gen_store(ACT, int);
|
|
|
|
static void gen_t_start(ACT, int);
|
|
|
|
static void gen_tpb(TPB, int);
|
|
|
|
static void gen_trans(ACT, int);
|
|
|
|
static void gen_type(ACT, int);
|
|
|
|
static void gen_update(ACT, int);
|
|
|
|
static void gen_variable(ACT, int);
|
|
|
|
static void gen_whenever(SWE, int);
|
|
|
|
static void make_array_declaration(REF);
|
|
|
|
static TEXT *make_name(TEXT *, SYM);
|
2002-11-17 01:04:19 +01:00
|
|
|
static void make_ok_test(ACT, GPRE_REQ, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void make_port(POR, int);
|
2002-11-17 01:04:19 +01:00
|
|
|
static void make_ready(DBB, TEXT *, TEXT *, USHORT, GPRE_REQ);
|
2003-02-27 22:44:48 +01:00
|
|
|
static void printa(int, const char *, ...) ATTRIBUTE_FORMAT(2,3);
|
|
|
|
static void printb(TEXT *, ...) ATTRIBUTE_FORMAT(1,2);
|
2002-11-17 01:04:19 +01:00
|
|
|
static TEXT *request_trans(ACT, GPRE_REQ);
|
2001-05-23 15:26:42 +02:00
|
|
|
static TEXT *status_vector(ACT);
|
2003-09-10 21:48:53 +02:00
|
|
|
static void t_start_auto(ACT, GPRE_REQ, TEXT *, int, bool);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static int global_first_flag;
|
2001-05-23 15:26:42 +02:00
|
|
|
static TEXT *status_name;
|
|
|
|
|
|
|
|
#define INDENT 3
|
|
|
|
|
|
|
|
#define NULL_STRING "(char *)0"
|
|
|
|
#define NULL_STATUS "(long*) 0L"
|
|
|
|
#define NULL_SQLDA "(XSQLDA*) 0L"
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
#define GDS_INCLUDE "\"interbase:[syslib]gds.h\""
|
|
|
|
#endif
|
|
|
|
|
2001-07-12 07:46:06 +02:00
|
|
|
#ifdef DARWIN
|
|
|
|
#define GDS_INCLUDE "<Firebird/ibase.h>"
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifndef GDS_INCLUDE
|
|
|
|
#define GDS_INCLUDE "<ibase.h>"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef DCL_LONG
|
|
|
|
#define DCL_LONG "long"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef DCL_QUAD
|
|
|
|
#define DCL_QUAD "ISC_QUAD"
|
|
|
|
#endif
|
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
static inline void begin(const int column)
|
2003-09-16 16:01:56 +02:00
|
|
|
{
|
|
|
|
printa(column, "{");
|
|
|
|
}
|
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
static inline void endp(const int column)
|
2003-09-16 16:01:56 +02:00
|
|
|
{
|
|
|
|
printa(column, "}");
|
|
|
|
}
|
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
static inline void set_sqlcode(ACT action, const int column)
|
2003-09-12 18:35:40 +02:00
|
|
|
{
|
|
|
|
if (action->act_flags & ACT_sql)
|
2003-09-25 13:49:12 +02:00
|
|
|
printa(column, "SQLCODE = isc_sqlcode(%s);", status_name);
|
2003-09-12 18:35:40 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
void C_CXX_action( ACT action, int column)
|
|
|
|
{
|
|
|
|
|
|
|
|
status_name = "isc_status";
|
|
|
|
|
|
|
|
// Put leading braces where required
|
|
|
|
|
|
|
|
switch (action->act_type) {
|
|
|
|
case ACT_alter_database:
|
|
|
|
case ACT_alter_domain:
|
|
|
|
case ACT_alter_index:
|
|
|
|
case ACT_alter_table:
|
|
|
|
case ACT_blob_close:
|
|
|
|
case ACT_blob_create:
|
|
|
|
case ACT_blob_for:
|
|
|
|
case ACT_blob_open:
|
|
|
|
case ACT_clear_handles:
|
|
|
|
case ACT_close:
|
|
|
|
case ACT_commit:
|
|
|
|
case ACT_commit_retain_context:
|
|
|
|
case ACT_create_database:
|
|
|
|
case ACT_create_domain:
|
|
|
|
case ACT_create_generator:
|
|
|
|
case ACT_create_index:
|
|
|
|
case ACT_create_shadow:
|
|
|
|
case ACT_create_table:
|
|
|
|
case ACT_create_view:
|
|
|
|
case ACT_declare_filter:
|
|
|
|
case ACT_declare_udf:
|
|
|
|
case ACT_disconnect:
|
|
|
|
case ACT_drop_database:
|
|
|
|
case ACT_drop_domain:
|
|
|
|
case ACT_drop_filter:
|
|
|
|
case ACT_drop_index:
|
|
|
|
case ACT_drop_shadow:
|
|
|
|
case ACT_drop_table:
|
|
|
|
case ACT_drop_udf:
|
|
|
|
case ACT_drop_view:
|
|
|
|
case ACT_dyn_close:
|
|
|
|
case ACT_dyn_cursor:
|
|
|
|
case ACT_dyn_describe:
|
|
|
|
case ACT_dyn_describe_input:
|
|
|
|
case ACT_dyn_execute:
|
|
|
|
case ACT_dyn_fetch:
|
|
|
|
case ACT_dyn_grant:
|
|
|
|
case ACT_dyn_immediate:
|
|
|
|
case ACT_dyn_insert:
|
|
|
|
case ACT_dyn_open:
|
|
|
|
case ACT_dyn_prepare:
|
|
|
|
case ACT_dyn_revoke:
|
|
|
|
case ACT_fetch:
|
|
|
|
case ACT_finish:
|
|
|
|
case ACT_for:
|
|
|
|
case ACT_get_segment:
|
|
|
|
case ACT_get_slice:
|
|
|
|
case ACT_insert:
|
|
|
|
case ACT_loop:
|
|
|
|
case ACT_modify:
|
|
|
|
case ACT_open:
|
|
|
|
case ACT_prepare:
|
|
|
|
case ACT_procedure:
|
|
|
|
case ACT_put_slice:
|
|
|
|
case ACT_ready:
|
|
|
|
case ACT_release:
|
|
|
|
case ACT_rfinish:
|
|
|
|
case ACT_rollback:
|
|
|
|
case ACT_s_fetch:
|
|
|
|
case ACT_s_start:
|
|
|
|
case ACT_select:
|
|
|
|
case ACT_store:
|
|
|
|
case ACT_start:
|
|
|
|
case ACT_update:
|
|
|
|
case ACT_statistics:
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (action->act_type) {
|
|
|
|
case ACT_alter_database:
|
|
|
|
case ACT_alter_domain:
|
|
|
|
case ACT_alter_index:
|
|
|
|
case ACT_alter_table:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_any:
|
|
|
|
gen_any(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_at_end:
|
|
|
|
gen_at_end(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_b_declare:
|
|
|
|
gen_database(action, column);
|
|
|
|
gen_routine(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_basedon:
|
|
|
|
gen_based(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_blob_cancel:
|
|
|
|
gen_blob_close(action, (USHORT) column);
|
|
|
|
return;
|
|
|
|
case ACT_blob_close:
|
|
|
|
gen_blob_close(action, (USHORT) column);
|
|
|
|
break;
|
|
|
|
case ACT_blob_create:
|
|
|
|
gen_blob_open(action, (USHORT) column);
|
|
|
|
break;
|
|
|
|
case ACT_blob_for:
|
|
|
|
gen_blob_for(action, (USHORT) column);
|
|
|
|
return;
|
|
|
|
case ACT_blob_handle:
|
|
|
|
gen_segment(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_blob_open:
|
|
|
|
gen_blob_open(action, (USHORT) column);
|
|
|
|
break;
|
|
|
|
case ACT_clear_handles:
|
|
|
|
gen_clear_handles(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_close:
|
|
|
|
gen_s_end(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_commit:
|
|
|
|
gen_trans(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_commit_retain_context:
|
|
|
|
gen_trans(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_create_database:
|
|
|
|
gen_create_database(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_create_domain:
|
|
|
|
case ACT_create_generator:
|
|
|
|
case ACT_create_index:
|
|
|
|
case ACT_create_shadow:
|
|
|
|
case ACT_create_table:
|
|
|
|
case ACT_create_view:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_cursor:
|
|
|
|
gen_cursor_init(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_database:
|
|
|
|
gen_database(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_declare_filter:
|
|
|
|
case ACT_declare_udf:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_disconnect:
|
|
|
|
gen_finish(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_drop_database:
|
|
|
|
gen_drop_database(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_drop_domain:
|
|
|
|
case ACT_drop_filter:
|
|
|
|
case ACT_drop_index:
|
|
|
|
case ACT_drop_shadow:
|
|
|
|
case ACT_drop_table:
|
|
|
|
case ACT_drop_udf:
|
|
|
|
case ACT_drop_view:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_close:
|
|
|
|
gen_dyn_close(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_cursor:
|
|
|
|
gen_dyn_declare(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_describe:
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_dyn_describe(action, column, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
case ACT_dyn_describe_input:
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_dyn_describe(action, column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
case ACT_dyn_execute:
|
|
|
|
gen_dyn_execute(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_fetch:
|
|
|
|
gen_dyn_fetch(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_grant:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_immediate:
|
|
|
|
gen_dyn_immediate(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_insert:
|
|
|
|
gen_dyn_insert(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_open:
|
|
|
|
gen_dyn_open(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_prepare:
|
|
|
|
gen_dyn_prepare(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_dyn_revoke:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_endblob:
|
|
|
|
gen_blob_end(action, (USHORT) column);
|
|
|
|
return;
|
|
|
|
case ACT_enderror:{
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ACT_endfor:
|
|
|
|
gen_endfor(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_endmodify:
|
|
|
|
gen_emodify(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_endstore:
|
|
|
|
gen_estore(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_erase:
|
|
|
|
gen_erase(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_event_init:
|
|
|
|
gen_event_init(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_event_wait:
|
|
|
|
gen_event_wait(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_fetch:
|
|
|
|
gen_fetch(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_finish:
|
|
|
|
gen_finish(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_for:
|
|
|
|
gen_for(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_function:
|
|
|
|
gen_function(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_get_segment:
|
|
|
|
gen_get_segment(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_get_slice:
|
|
|
|
gen_slice(action, 0, column);
|
|
|
|
break;
|
|
|
|
case ACT_hctef:
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
case ACT_insert:
|
|
|
|
gen_s_start(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_loop:
|
|
|
|
gen_loop(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_on_error:
|
|
|
|
gen_on_error(action, (USHORT) column);
|
|
|
|
return;
|
|
|
|
case ACT_open:
|
|
|
|
gen_s_start(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_prepare:
|
|
|
|
gen_trans(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_procedure:
|
|
|
|
gen_procedure(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_put_segment:
|
|
|
|
gen_put_segment(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_put_slice:
|
|
|
|
gen_slice(action, 0, column);
|
|
|
|
break;
|
|
|
|
case ACT_ready:
|
|
|
|
gen_ready(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_release:
|
|
|
|
gen_release(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_rfinish:
|
|
|
|
gen_finish(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_rollback:
|
|
|
|
gen_trans(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_routine:
|
|
|
|
gen_routine(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_s_end:
|
|
|
|
gen_s_end(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_s_fetch:
|
|
|
|
gen_s_fetch(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_s_start:
|
|
|
|
gen_s_start(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_segment:
|
|
|
|
gen_segment(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_segment_length:
|
|
|
|
gen_segment(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_select:
|
|
|
|
gen_select(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_sql_dialect:
|
|
|
|
sw_sql_dialect = ((SDT) action->act_object)->sdt_dialect;
|
|
|
|
return;
|
|
|
|
case ACT_start:
|
|
|
|
gen_t_start(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_statistics:
|
|
|
|
gen_ddl(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_store:
|
|
|
|
gen_store(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_store2:
|
|
|
|
gen_return_value(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_type:
|
|
|
|
gen_type(action, column);
|
|
|
|
return;
|
|
|
|
case ACT_update:
|
|
|
|
gen_update(action, column);
|
|
|
|
break;
|
|
|
|
case ACT_variable:
|
|
|
|
gen_variable(action, column);
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Put in a trailing brace for those actions still with us
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql)
|
|
|
|
gen_whenever(action->act_whenever, column);
|
|
|
|
|
|
|
|
if (action->act_error)
|
|
|
|
ib_fprintf(out_file, ";");
|
|
|
|
else
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Align output to a specific column for output. If the
|
|
|
|
// column is negative, don't do anything.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void align( int column)
|
|
|
|
{
|
|
|
|
if (column < 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ib_putc('\n', out_file);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int i;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (i = column / 8; i; --i)
|
|
|
|
ib_putc('\t', out_file);
|
|
|
|
|
|
|
|
for (i = column % 8; i; --i)
|
|
|
|
ib_putc(' ', out_file);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Build an assignment from a host language variable to
|
|
|
|
// a port variable. The string assignments are a little
|
|
|
|
// hairy because the normal mode is varying (null
|
|
|
|
// terminated) strings, but the fixed subtype makes the
|
|
|
|
// string a byte stream. Furthering the complication, a
|
|
|
|
// single character byte stream is handled as a single byte,
|
|
|
|
// meaining that it is the byte, not the address of the
|
|
|
|
// byte.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void asgn_from( ACT action, REF reference, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT name[64], variable[20], temp[20];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (; reference; reference = reference->ref_next) {
|
2003-09-29 14:43:14 +02:00
|
|
|
bool slice_flag = false;
|
|
|
|
GPRE_FLD field = reference->ref_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_array_info) {
|
2003-09-29 14:43:14 +02:00
|
|
|
ACT slice_action;
|
|
|
|
REF source = reference->ref_friend;
|
|
|
|
if (source &&
|
2001-05-23 15:26:42 +02:00
|
|
|
(slice_action = (ACT) source->ref_slice) &&
|
2003-09-29 14:43:14 +02:00
|
|
|
slice_action->act_object)
|
|
|
|
{
|
2003-09-10 21:48:53 +02:00
|
|
|
slice_flag = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
slice_action->act_type = ACT_put_slice;
|
|
|
|
gen_slice(slice_action, 0, column);
|
|
|
|
}
|
|
|
|
else if (!(reference->ref_flags & REF_array_elem)) {
|
|
|
|
printa(column, "%s = isc_blob_null;",
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(name, reference, true));
|
|
|
|
gen_get_or_put_slice(action, reference, false, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!reference->ref_source && !reference->ref_value && !slice_flag)
|
|
|
|
continue;
|
|
|
|
align(column);
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(variable, reference, true);
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* value;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (slice_flag)
|
2003-09-10 21:48:53 +02:00
|
|
|
value = gen_name(temp, reference->ref_friend, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (reference->ref_source)
|
2003-09-10 21:48:53 +02:00
|
|
|
value = gen_name(temp, reference->ref_source, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
value = reference->ref_value;
|
|
|
|
|
|
|
|
if (!slice_flag &&
|
|
|
|
reference->ref_value &&
|
2003-09-29 14:43:14 +02:00
|
|
|
(reference->ref_flags & REF_array_elem))
|
|
|
|
{
|
|
|
|
field = field->fld_array;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field && field->fld_dtype <= dtype_cstring)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_sub_type == 1)
|
|
|
|
if (field->fld_length == 1)
|
|
|
|
ib_fprintf(out_file, "%s = %s;", variable, value);
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file,
|
|
|
|
"isc_ftof (%s, sizeof (%s), %s, %d);", value,
|
|
|
|
value, variable, field->fld_length);
|
|
|
|
else if (field->fld_flags & FLD_dbkey)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, %d);", value,
|
|
|
|
field->fld_length, variable, field->fld_length);
|
|
|
|
else if (sw_cstring)
|
2001-07-12 07:46:06 +02:00
|
|
|
ib_fprintf(out_file, "isc_vtov ((char*)%s, (char*)%s, %d);", value,
|
2001-05-23 15:26:42 +02:00
|
|
|
variable, field->fld_length);
|
|
|
|
else if (reference->ref_source)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, sizeof (%s), %s, %d);",
|
|
|
|
value, value, variable, field->fld_length);
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file, "isc_vtof (%s, %s, %d);", value,
|
|
|
|
variable, field->fld_length);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (!reference->ref_master
|
2003-09-29 14:43:14 +02:00
|
|
|
|| (reference->ref_flags & REF_literal))
|
|
|
|
{
|
|
|
|
ib_fprintf(out_file, "%s = %s;", variable, value);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
ib_fprintf(out_file, "if (%s < 0)", value);
|
|
|
|
align(column + 4);
|
|
|
|
ib_fprintf(out_file, "%s = -1;", variable);
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "else");
|
|
|
|
align(column + 4);
|
|
|
|
ib_fprintf(out_file, "%s = 0;", variable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Build an assignment to a host language variable from
|
|
|
|
// a port variable.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void asgn_to( ACT action, REF reference, int column)
|
|
|
|
{
|
|
|
|
char s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
REF source = reference->ref_friend;
|
|
|
|
GPRE_FLD field = source->ref_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (field) {
|
2003-09-29 14:43:14 +02:00
|
|
|
ACT slice_action;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_array_info && (slice_action = (ACT) source->ref_slice)) {
|
|
|
|
source->ref_value = reference->ref_value;
|
|
|
|
if (slice_action->act_object) {
|
|
|
|
slice_action->act_type = ACT_get_slice;
|
|
|
|
gen_slice(slice_action, source, column);
|
|
|
|
}
|
|
|
|
else
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_get_or_put_slice(action, source, true, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick up NULL value if one is there */
|
|
|
|
|
|
|
|
if (reference = reference->ref_null) {
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value,
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, reference, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, source, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_dtype > dtype_cstring)
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value, s);
|
|
|
|
else if (field->fld_sub_type == 1 && field->fld_length == 1)
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value, s);
|
|
|
|
else if (field->fld_flags & FLD_dbkey)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, %d);",
|
|
|
|
s, field->fld_length, reference->ref_value,
|
|
|
|
field->fld_length);
|
|
|
|
else if (!sw_cstring || field->fld_sub_type == 1)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, sizeof (%s));", s,
|
|
|
|
field->fld_length, reference->ref_value,
|
|
|
|
reference->ref_value);
|
|
|
|
else
|
2001-07-12 07:46:06 +02:00
|
|
|
ib_fprintf(out_file, "isc_vtov ((char*)%s, (char*)%s, sizeof (%s));", s,
|
2001-05-23 15:26:42 +02:00
|
|
|
reference->ref_value, reference->ref_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pick up NULL value if one is there
|
|
|
|
|
|
|
|
if (reference = reference->ref_null) {
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value,
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, reference, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Build an assignment to a host language variable from
|
|
|
|
// a port variable.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void asgn_to_proc( REF reference, int column)
|
|
|
|
{
|
|
|
|
char s[64];
|
|
|
|
|
|
|
|
for (; reference; reference = reference->ref_next) {
|
|
|
|
if (!reference->ref_value)
|
|
|
|
continue;
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_FLD field = reference->ref_field;
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, reference, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
|
|
|
|
if (field->fld_dtype > dtype_cstring)
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value, s);
|
|
|
|
else if (field->fld_sub_type == 1 && field->fld_length == 1)
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value, s);
|
|
|
|
else if (field->fld_flags & FLD_dbkey)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, %d);",
|
|
|
|
s, field->fld_length, reference->ref_value,
|
|
|
|
field->fld_length);
|
|
|
|
else if (!sw_cstring || field->fld_sub_type == 1)
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, sizeof (%s));", s,
|
|
|
|
field->fld_length, reference->ref_value,
|
|
|
|
reference->ref_value);
|
|
|
|
else
|
2001-07-12 07:46:06 +02:00
|
|
|
ib_fprintf(out_file, "isc_vtov ((char*)%s, (char*)%s, sizeof (%s));", s,
|
2001-05-23 15:26:42 +02:00
|
|
|
reference->ref_value, reference->ref_value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a function call for free standing ANY. Somebody else
|
|
|
|
// will need to generate the actual function.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_any( ACT action, int column)
|
|
|
|
{
|
|
|
|
align(column);
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ib_fprintf(out_file, "%s_r (&%s, &%s",
|
|
|
|
request->req_handle, request->req_handle, request->req_trans);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
POR port = request->req_vport;
|
|
|
|
if (port)
|
|
|
|
for (REF reference = port->por_references; reference;
|
2001-05-23 15:26:42 +02:00
|
|
|
reference = reference->ref_next)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_fprintf(out_file, ", %s", reference->ref_value);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ib_fprintf(out_file, ")");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate code for AT END clause of FETCH.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_at_end( ACT action, int column)
|
|
|
|
{
|
|
|
|
char s[20];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2003-09-10 21:48:53 +02:00
|
|
|
printa(column, "if (!%s) {", gen_name(s, request->req_eof, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Substitute for a BASED ON <field name> clause.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_based( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT s[64], *variable;
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT datatype;
|
|
|
|
SLONG length;
|
|
|
|
DIM dimension;
|
|
|
|
|
|
|
|
align(column);
|
2003-09-29 14:43:14 +02:00
|
|
|
BAS based_on = (BAS) action->act_object;
|
|
|
|
GPRE_FLD field = based_on->bas_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (based_on->bas_flags & BAS_segment) {
|
|
|
|
datatype = (sw_cstring) ? dtype_cstring : dtype_text;
|
|
|
|
if (!(length = field->fld_seg_length))
|
|
|
|
length = 256;
|
|
|
|
if (datatype == dtype_cstring)
|
|
|
|
length++;
|
|
|
|
}
|
|
|
|
else if (field->fld_array_info)
|
|
|
|
datatype = field->fld_array_info->ary_dtype;
|
|
|
|
else
|
|
|
|
datatype = field->fld_dtype;
|
|
|
|
|
|
|
|
switch (datatype) {
|
|
|
|
case dtype_short:
|
|
|
|
ib_fprintf(out_file, "short");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_long:
|
|
|
|
ib_fprintf(out_file, DCL_LONG);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_quad:
|
|
|
|
ib_fprintf(out_file, DCL_QUAD);
|
|
|
|
break;
|
|
|
|
|
|
|
|
// ** Begin date/time/timestamp *
|
|
|
|
case dtype_sql_date:
|
|
|
|
ib_fprintf(out_file, "ISC_DATE");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_sql_time:
|
|
|
|
ib_fprintf(out_file, "ISC_TIME");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_timestamp:
|
|
|
|
ib_fprintf(out_file, "ISC_TIMESTAMP");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_int64:
|
|
|
|
ib_fprintf(out_file, "ISC_INT64");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_blob:
|
|
|
|
ib_fprintf(out_file, "ISC_QUAD");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_cstring:
|
|
|
|
case dtype_text:
|
|
|
|
case dtype_varying:
|
|
|
|
ib_fprintf(out_file, "char");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_real:
|
|
|
|
ib_fprintf(out_file, "float");
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_double:
|
|
|
|
ib_fprintf(out_file, "double");
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
sprintf(s, "datatype %d unknown\n", field->fld_dtype);
|
|
|
|
IBERROR(s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// print the first variable, then precede the rest with commas
|
|
|
|
|
|
|
|
column += INDENT;
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
// Notice this variable was named first_flag, same than the global variable.
|
|
|
|
bool first = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (based_on->bas_variables) {
|
|
|
|
variable = (TEXT *) POP(&based_on->bas_variables);
|
2003-09-29 14:43:14 +02:00
|
|
|
if (!first)
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_fprintf(out_file, ",");
|
2003-09-29 14:43:14 +02:00
|
|
|
first = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "%s", variable);
|
|
|
|
if (based_on->bas_flags & BAS_segment) {
|
|
|
|
if (*variable != '*')
|
2002-09-24 14:57:10 +02:00
|
|
|
ib_fprintf(out_file, "[%ld]", length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (field->fld_array_info) {
|
|
|
|
/* Print out the dimension part of the declaration */
|
|
|
|
|
|
|
|
for (dimension = field->fld_array_info->ary_dimension; dimension;
|
|
|
|
dimension = dimension->dim_next)
|
2002-09-24 14:57:10 +02:00
|
|
|
ib_fprintf(out_file, " [%ld]",
|
2001-05-23 15:26:42 +02:00
|
|
|
dimension->dim_upper - dimension->dim_lower + 1);
|
|
|
|
|
|
|
|
if (field->fld_array_info->ary_dtype <= dtype_varying &&
|
|
|
|
field->fld_length > 1)
|
2003-04-01 13:49:33 +02:00
|
|
|
ib_fprintf(out_file, " [%d]", field->fld_array->fld_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
if (*variable != '*' &&
|
|
|
|
field->fld_dtype <= dtype_varying &&
|
|
|
|
(field->fld_sub_type != 1 || field->fld_length > 1))
|
|
|
|
// *???????
|
|
|
|
// if (*variable != '*' && field->fld_dtype <= dtype_varying &&
|
|
|
|
// field->fld_length > 1)
|
|
|
|
//
|
|
|
|
ib_fprintf(out_file, "[%d]", field->fld_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "%s\n", based_on->bas_terminator);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Make a blob FOR loop.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_blob_close( ACT action, USHORT column)
|
|
|
|
{
|
|
|
|
BLB blob;
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 = "isc_%IFcancel%ELclose%EN_blob (%V1, &%BH);";
|
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
|
|
|
column = gen_cursor_close(action, action->act_request, column);
|
|
|
|
blob = (BLB) action->act_request->req_blobs;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
blob = (BLB) action->act_object;
|
|
|
|
|
|
|
|
args.pat_blob = blob;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = (action->act_type == ACT_blob_cancel);
|
2001-05-23 15:26:42 +02:00
|
|
|
PATTERN_expand(column, pattern1, &args);
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// End a blob FOR loop.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_blob_end( ACT action, USHORT column)
|
|
|
|
{
|
|
|
|
BLB blob;
|
|
|
|
PAT args;
|
|
|
|
TEXT s1[32];
|
|
|
|
TEXT *pattern1 = "}\n\
|
|
|
|
isc_close_blob (%V1, &%BH);\n\
|
|
|
|
}";
|
|
|
|
|
|
|
|
args.pat_blob = blob = (BLB) action->act_object;
|
|
|
|
if (action->act_error) {
|
|
|
|
sprintf(s1, "%s2", status_name);
|
|
|
|
args.pat_vector1 = s1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
args.pat_vector1 = status_vector(0);
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = (action->act_type == ACT_blob_cancel);
|
2001-05-23 15:26:42 +02:00
|
|
|
PATTERN_expand(column, pattern1, &args);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Make a blob FOR loop.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_blob_for( ACT action, USHORT column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 = "%IFif (!%S1 [1]) {\n\
|
|
|
|
%ENwhile (1)\n\
|
|
|
|
{";
|
|
|
|
|
|
|
|
gen_blob_open(action, column);
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = (action->act_error != NULL);
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_string1 = status_name;
|
|
|
|
PATTERN_expand(column, pattern1, &args);
|
|
|
|
column += INDENT;
|
|
|
|
gen_get_segment(action, column);
|
|
|
|
printa(column, "if (%s [1] && (%s [1] != isc_segment)) break;",
|
|
|
|
status_name, status_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the call to open (or create) a blob.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_blob_open( ACT action, USHORT column)
|
|
|
|
{
|
|
|
|
BLB blob;
|
|
|
|
PAT args;
|
|
|
|
REF reference;
|
|
|
|
TEXT s[20];
|
|
|
|
TEXT *pattern1 =
|
|
|
|
"isc_%IFcreate%ELopen%EN_blob2 (%V1, &%DH, &%RT, &%BH, &%FR, (short) %N1, %I1);",
|
|
|
|
*pattern2 =
|
2001-07-12 07:46:06 +02:00
|
|
|
"isc_%IFcreate%ELopen%EN_blob2 (%V1, &%DH, &%RT, &%BH, &%FR, (short) 0, (%IFchar%ELunsigned char%EN*) 0);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (sw_auto && (action->act_flags & ACT_sql)) {
|
|
|
|
t_start_auto(action, action->act_request, status_vector(action),
|
2003-09-10 21:48:53 +02:00
|
|
|
column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", request_trans(action, action->act_request));
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((action->act_error && (action->act_type != ACT_blob_for)) ||
|
|
|
|
(action->act_flags & ACT_sql))
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
|
|
|
column = gen_cursor_open(action, action->act_request, column);
|
|
|
|
blob = (BLB) action->act_request->req_blobs;
|
|
|
|
reference = ((OPN) action->act_object)->opn_using;
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, reference, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
blob = (BLB) action->act_object;
|
|
|
|
reference = blob->blb_reference;
|
|
|
|
}
|
|
|
|
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = (action->act_type == ACT_blob_create); // open or create blob
|
|
|
|
args.pat_vector1 = status_vector(action); // status vector
|
|
|
|
args.pat_database = blob->blb_request->req_database; // database handle
|
|
|
|
args.pat_request = blob->blb_request; // transaction handle
|
|
|
|
args.pat_blob = blob; // blob handle
|
|
|
|
args.pat_reference = reference; // blob identifier
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_ident1 = blob->blb_bpb_ident;
|
|
|
|
|
|
|
|
if ((action->act_flags & ACT_sql) && action->act_type == ACT_blob_open) {
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "%s = %s;", s, reference->ref_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.pat_value1 = blob->blb_bpb_length)
|
|
|
|
PATTERN_expand(column, pattern1, &args);
|
|
|
|
else
|
|
|
|
PATTERN_expand(column, pattern2, &args);
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (action->act_type == ACT_blob_create) {
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
align(column + INDENT);
|
|
|
|
ib_fprintf(out_file, "%s = %s;", reference->ref_value, s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ((action->act_error && (action->act_type != ACT_blob_for)))
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Callback routine for BLR pretty printer.
|
|
|
|
//
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static void gen_blr(void* user_arg, SSHORT offset, const char* string)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
const char *q, *p1;
|
2003-09-28 23:36:05 +02:00
|
|
|
char *q1, d, line[256];
|
2003-09-10 21:48:53 +02:00
|
|
|
bool first_line = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int indent = 2 * INDENT;
|
|
|
|
const char* p = string;
|
2001-05-23 15:26:42 +02:00
|
|
|
while (*p == ' ') {
|
|
|
|
p++;
|
|
|
|
indent++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Limit indentation to 192 characters
|
|
|
|
|
|
|
|
indent = MIN(indent, 192);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int length = strlen(p);
|
2001-05-23 15:26:42 +02:00
|
|
|
do {
|
|
|
|
if (length + indent > 255) {
|
|
|
|
/* if we did not find somewhere to break between the 200th and 256th
|
|
|
|
character just print out 256 characters */
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
bool open_quote = false;
|
|
|
|
for (q = p; (q - p + indent) < 255; q++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((q - p + indent) > 220 && *q == ',' && !open_quote)
|
|
|
|
break;
|
|
|
|
if (*q == '\'' && *(q - 1) != '\\')
|
|
|
|
open_quote = !open_quote;
|
|
|
|
}
|
2003-09-28 23:36:05 +02:00
|
|
|
++q;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
q = p + strlen(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace all occurrences of gds__ (or gds__) with isc_ */
|
|
|
|
|
2003-09-28 23:36:05 +02:00
|
|
|
for (q1 = line, p1 = p; p1 < q;)
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((*q1++ = *p1++) == 'g')
|
2003-09-28 23:36:05 +02:00
|
|
|
if (p1 < q && (*q1++ = *p1++) == 'd')
|
|
|
|
if (p1 < q && (*q1++ = *p1++) == 's')
|
|
|
|
if (p1 < q && (*q1++ = *p1++) == '_')
|
|
|
|
if (p1 < q && ((d = *p1++) == '_' || d == '$'))
|
2001-05-23 15:26:42 +02:00
|
|
|
strncpy(q1 - 4, "isc", 3);
|
|
|
|
else
|
|
|
|
*q1++ = d;
|
|
|
|
*q1 = 0;
|
|
|
|
printa(indent, line);
|
|
|
|
length = length - (q - p);
|
|
|
|
p = q;
|
|
|
|
if (first_line) {
|
|
|
|
indent = MIN(indent + INDENT, 192);
|
2003-09-10 21:48:53 +02:00
|
|
|
first_line = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
} while (length > 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Zap all know handles.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_clear_handles( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
for (GPRE_REQ request = requests; request; request = request->req_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(request->req_flags & REQ_exp_hand))
|
|
|
|
printa(column, "%s = 0;", request->req_handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a symbol to ease compatibility with V3.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_compatibility_symbol(
|
|
|
|
TEXT * symbol,
|
|
|
|
TEXT * v4_prefix, TEXT * trailer)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
const char* v3_prefix = (isLangCpp(sw_language)) ? "gds_" : "gds__";
|
2002-01-04 12:34:22 +01:00
|
|
|
// v3_prefix = (sw_language == lang_cxx) ? "gds_" : "gds__";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ib_fprintf(out_file, "#define %s%s\t%s%s%s\n", v3_prefix, symbol,
|
|
|
|
v4_prefix, symbol, trailer);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text to compile a request.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_compile( ACT action, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 =
|
2003-02-28 14:20:36 +01:00
|
|
|
"isc_compile_request%IF2%EN (%V1, (FRBRD**) &%DH, (FRBRD**) &%RH, (short) sizeof (%RI), (char *) %RI);",
|
2001-05-23 15:26:42 +02:00
|
|
|
*pattern2 = "if (!%RH%IF && %S1%EN)";
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
|
|
|
args.pat_request = request;
|
|
|
|
DBB db = request->req_database;
|
|
|
|
args.pat_database = db;
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_string1 = request_trans(action, request);
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = (sw_auto && (action->act_error
|
2001-05-23 15:26:42 +02:00
|
|
|
|| (action->act_flags & ACT_sql)));
|
|
|
|
|
|
|
|
if (sw_auto)
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, request, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern2, &args);
|
|
|
|
|
|
|
|
args.pat_condition = !(request->req_flags & REQ_exp_hand);
|
|
|
|
args.pat_value1 = request->req_length;
|
|
|
|
PATTERN_expand((USHORT) (column + INDENT), pattern1, &args);
|
|
|
|
|
|
|
|
// If blobs are present, zero out all of the blob handles. After this
|
|
|
|
// point, the handles are the user's responsibility
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
BLB blob = request->req_blobs;
|
|
|
|
if (blob) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ib_fprintf(out_file, "\n");
|
|
|
|
align(column - INDENT);
|
|
|
|
for (; blob; blob = blob->blb_next)
|
|
|
|
ib_fprintf(out_file, "isc_%d = ", blob->blb_ident);
|
|
|
|
ib_fprintf(out_file, "0;");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a call to create a database.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_create_database( ACT action, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 =
|
|
|
|
"isc_create_database (%V1, %N1, \"%DF\", &%DH, %IF%S1, %S2%EL(short) 0, (char*) 0%EN, 0);";
|
|
|
|
TEXT s1[32], s2[32], trname[32];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = ((MDBB) action->act_object)->mdbb_dpb_request;
|
|
|
|
DBB db = (DBB) request->req_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
sprintf(s1, "isc_%dl", request->req_ident);
|
|
|
|
sprintf(trname, "isc_%dt", request->req_ident);
|
|
|
|
|
|
|
|
if (request->req_flags & REQ_extend_dpb) {
|
|
|
|
sprintf(s2, "isc_%dp", request->req_ident);
|
|
|
|
if (request->req_length)
|
|
|
|
printa(column, "%s = isc_%d;", s2, request->req_ident);
|
|
|
|
else
|
|
|
|
printa(column, "%s = (char*) 0;", s2);
|
|
|
|
|
|
|
|
printa(column,
|
|
|
|
"isc_expand_dpb (&%s, &%s, isc_dpb_user_name, %s, isc_dpb_password, %s, isc_dpb_sql_role_name, %s, isc_dpb_lc_messages, %s, isc_dpb_lc_ctype, %s, 0);",
|
|
|
|
s2, s1,
|
|
|
|
db->dbb_r_user ? db->dbb_r_user : "(char*) 0",
|
|
|
|
db->dbb_r_password ? db->dbb_r_password : "(char*) 0",
|
|
|
|
db->dbb_r_sql_role ? db->dbb_r_sql_role : "(char*) 0",
|
|
|
|
db->dbb_r_lc_messages ? db->dbb_r_lc_messages : "(char*) 0",
|
|
|
|
db->dbb_r_lc_ctype ? db->dbb_r_lc_ctype : "(char*) 0");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sprintf(s2, "isc_%d", request->req_ident);
|
|
|
|
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_request = request;
|
|
|
|
args.pat_database = db;
|
|
|
|
args.pat_value1 = strlen(db->dbb_filename);
|
|
|
|
args.pat_condition = (request->req_length
|
|
|
|
|| (request->req_flags & REQ_extend_dpb));
|
|
|
|
args.pat_string1 = s1;
|
|
|
|
args.pat_string2 = s2;
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
|
|
|
|
// if the dpb was extended, free it here
|
|
|
|
|
|
|
|
if (request->req_flags & REQ_extend_dpb) {
|
|
|
|
if (request->req_length)
|
|
|
|
printa(column, "if (%s != isc_%d)", s2, request->req_ident);
|
2001-07-12 07:46:06 +02:00
|
|
|
printa(column + (request->req_length ? INDENT : 0), "isc_free ((char*) %s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
s2);
|
|
|
|
|
|
|
|
/* reset the length of the dpb */
|
|
|
|
|
|
|
|
printa(column, "%s = %d;", s1, request->req_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
request = action->act_request;
|
|
|
|
printa(column, "if (!%s [1])", status_name);
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
2002-12-16 16:59:18 +01:00
|
|
|
"isc_start_transaction (%s, (FRBRD**) &%s, (short) 1, &%s, (short) 0, (char*) 0);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(action), trname, db->dbb_name->sym_string);
|
|
|
|
printa(column, "if (%s)", trname);
|
|
|
|
column += INDENT;
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "isc_ddl (%s, &%s, &%s, (short) %d, isc_%d);",
|
|
|
|
status_vector(action),
|
|
|
|
request->req_database->dbb_name->sym_string,
|
|
|
|
trname, request->req_length, request->req_ident);
|
|
|
|
column -= INDENT;
|
|
|
|
printa(column, "if (!%s [1])", status_name);
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column + INDENT, "isc_commit_transaction (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(action), trname);
|
|
|
|
printa(column, "if (%s [1])", status_name);
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column + INDENT, "isc_rollback_transaction (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(NULL), trname);
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column - INDENT, "else");
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for END_STREAM.
|
|
|
|
//
|
|
|
|
|
2002-11-17 01:04:19 +01:00
|
|
|
static int gen_cursor_close( ACT action, GPRE_REQ request, int column)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 =
|
|
|
|
"if (%RIs && !isc_dsql_free_statement (%V1, &%RIs, %N1))";
|
|
|
|
|
|
|
|
args.pat_request = request;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_value1 = 1;
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return column;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text to initialize a cursor.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_cursor_init( ACT action, int column)
|
|
|
|
{
|
|
|
|
|
|
|
|
// If blobs are present, zero out all of the blob handles. After this
|
|
|
|
// point, the handles are the user's responsibility
|
|
|
|
|
|
|
|
if (action->act_request->
|
2003-09-29 14:43:14 +02:00
|
|
|
req_flags & (REQ_sql_blob_open | REQ_sql_blob_create))
|
|
|
|
{
|
|
|
|
printa(column, "isc_%d = 0;",
|
|
|
|
action->act_request->req_blobs->blb_ident);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text to open an embedded SQL cursor.
|
|
|
|
//
|
|
|
|
|
2002-11-17 01:04:19 +01:00
|
|
|
static int gen_cursor_open( ACT action, GPRE_REQ request, int column)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT s[64];
|
|
|
|
TEXT *pattern1 = "if (!%RIs && %RH%IF && %DH%EN)",
|
|
|
|
*pattern2 = "if (!%RIs%IF && %DH%EN)",
|
|
|
|
*pattern3 = "isc_dsql_alloc_statement2 (%V1, &%DH, &%RIs);",
|
|
|
|
*pattern4 = "if (%RIs%IF && %S3%EN)",
|
|
|
|
*pattern5 = "if (!isc_dsql_set_cursor_name (%V1, &%RIs, %S1, 0) &&",
|
|
|
|
*pattern6 =
|
|
|
|
"!isc_dsql_execute_m (%V1, &%S3, &%RIs, 0, %S2, %N2, 0, %S2))";
|
|
|
|
|
|
|
|
args.pat_request = request;
|
|
|
|
args.pat_database = request->req_database;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = sw_auto;
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_string1 = make_name(s, ((OPN) action->act_object)->opn_cursor);
|
|
|
|
args.pat_string2 = NULL_STRING;
|
|
|
|
args.pat_string3 = request_trans(action, request);
|
|
|
|
args.pat_value2 = -1;
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column,
|
|
|
|
(action->act_type == ACT_open) ? pattern1 : pattern2,
|
|
|
|
&args);
|
|
|
|
PATTERN_expand((USHORT) (column + INDENT), pattern3, &args);
|
|
|
|
PATTERN_expand((USHORT) column, pattern4, &args);
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
PATTERN_expand((USHORT) column, pattern5, &args);
|
|
|
|
column += INDENT;
|
|
|
|
PATTERN_expand((USHORT) column, pattern6, &args);
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return column;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate insertion text for the database statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_database( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
if (global_first_flag++ != 0)
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "\n/**** GDS Preprocessor Definitions ****/\n");
|
|
|
|
ib_fprintf(out_file, "#ifndef JRD_IBASE_H\n#include %s\n#endif\n",
|
|
|
|
GDS_INCLUDE);
|
|
|
|
|
|
|
|
printa(column, "static %sISC_QUAD", CONST_STR);
|
|
|
|
printa(column + INDENT,
|
|
|
|
"isc_blob_null = {0,0};\t/* initializer for blobs */");
|
|
|
|
if (sw_language == lang_c)
|
|
|
|
printa(column,
|
|
|
|
"static %slong *gds__null = 0;\t/* dummy status vector */",
|
|
|
|
CONST_STR);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* scope = "";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
bool all_static = true;
|
|
|
|
bool all_extern = true;
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB db;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (db = isc_databases; db; db = db->dbb_next) {
|
|
|
|
all_static = all_static && (db->dbb_scope == DBB_STATIC);
|
|
|
|
all_extern = all_extern && (db->dbb_scope == DBB_EXTERN);
|
|
|
|
if (db->dbb_scope == DBB_STATIC)
|
|
|
|
scope = "static ";
|
2003-09-10 21:48:53 +02:00
|
|
|
else if (db->dbb_scope == DBB_EXTERN && TRUE)
|
2001-05-23 15:26:42 +02:00
|
|
|
scope = "extern ";
|
|
|
|
printa(column, "%sisc_db_handle", scope);
|
|
|
|
if (!all_extern)
|
|
|
|
printa(column + INDENT, "%s = 0;\t\t/* database handle */\n",
|
|
|
|
db->dbb_name->sym_string);
|
|
|
|
else
|
|
|
|
printa(column + INDENT, "%s;\t\t/* database handle */\n",
|
|
|
|
db->dbb_name->sym_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (all_static)
|
|
|
|
scope = "static ";
|
|
|
|
else if (all_extern)
|
|
|
|
scope = "extern ";
|
|
|
|
|
|
|
|
printa(column, "%sisc_tr_handle", scope);
|
|
|
|
if (!all_extern)
|
|
|
|
printa(column + INDENT, "%s = 0;\t\t/* default transaction handle */",
|
|
|
|
transaction_name);
|
|
|
|
else
|
|
|
|
printa(column + INDENT, "%s;\t\t/* default transaction handle */",
|
|
|
|
transaction_name);
|
|
|
|
|
|
|
|
printa(column, "%slong", scope);
|
|
|
|
column += INDENT;
|
|
|
|
printa(column, "%s [20],\t/* status vector */", status_name);
|
|
|
|
printa(column, "%s2 [20];\t/* status vector */", status_name);
|
|
|
|
printa(column - INDENT, "%s%s", scope, DCL_LONG);
|
|
|
|
printa(column, "isc_array_length, \t/* array return size */");
|
|
|
|
printa(column, "SQLCODE;\t\t/* SQL status code */");
|
|
|
|
if (sw_dyn_using) {
|
|
|
|
printa(column - INDENT,
|
|
|
|
"static struct isc_sqlda { /* SQLDA for internal use */");
|
|
|
|
printa(column, "char sqldaid [8];");
|
|
|
|
printa(column, "%s sqldabc;", DCL_LONG);
|
|
|
|
printa(column, "short sqln;");
|
|
|
|
printa(column, "short sqld;");
|
|
|
|
printa(column, "SQLVAR sqlvar[%d];", sw_dyn_using);
|
|
|
|
printa(column, "}");
|
|
|
|
}
|
|
|
|
|
|
|
|
column -= INDENT;
|
|
|
|
|
|
|
|
for (db = isc_databases; db; db = db->dbb_next)
|
2003-09-29 14:43:14 +02:00
|
|
|
for (tpb* tpb_iterator = db->dbb_tpbs;
|
2003-09-05 12:14:08 +02:00
|
|
|
tpb_iterator;
|
|
|
|
tpb_iterator = tpb_iterator->tpb_dbb_next)
|
|
|
|
{
|
|
|
|
gen_tpb(tpb_iterator, column);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// generate event parameter block for each event in module
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
SSHORT max_count = 0;
|
|
|
|
for (LLS stack_ptr = events; stack_ptr; stack_ptr = stack_ptr->lls_next) {
|
|
|
|
SSHORT count = gen_event_block((ACT) (stack_ptr->lls_object));
|
2001-05-23 15:26:42 +02:00
|
|
|
max_count = MAX(count, max_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (max_count)
|
|
|
|
printa(column, "%s%s isc_events [%d];\t/* event vector */", scope,
|
|
|
|
DCL_LONG, max_count);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (GPRE_REQ request = requests; request; request = request->req_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_request(request);
|
|
|
|
|
|
|
|
/* Array declarations */
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
if (POR port = request->req_primary)
|
|
|
|
for (REF reference = port->por_references; reference;
|
2001-05-23 15:26:42 +02:00
|
|
|
reference = reference->ref_next)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (reference->ref_flags & REF_fetch_array)
|
|
|
|
make_array_declaration(reference);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "\n\n");
|
|
|
|
gen_compatibility_symbol("blob_null", "isc_",
|
|
|
|
"\t/* compatibility symbols */");
|
|
|
|
// ****
|
|
|
|
//gen_compatibility_symbol ("database", "isc_", "");
|
|
|
|
//gen_compatibility_symbol ("trans", "isc_", "");
|
|
|
|
//***
|
|
|
|
gen_compatibility_symbol("status", "isc_", "");
|
|
|
|
gen_compatibility_symbol("status2", "isc_", "");
|
|
|
|
gen_compatibility_symbol("array_length", "isc_", "");
|
|
|
|
if (max_count)
|
|
|
|
gen_compatibility_symbol("events", "isc_", "");
|
|
|
|
gen_compatibility_symbol("count", "isc_", "");
|
|
|
|
gen_compatibility_symbol("slack", "isc_", "");
|
|
|
|
gen_compatibility_symbol("utility", "isc_",
|
|
|
|
"\t/* end of compatibility symbols */");
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "\n#ifndef isc_version4\n");
|
|
|
|
ib_fprintf(out_file, " Generate a compile-time error.\n");
|
|
|
|
ib_fprintf(out_file,
|
|
|
|
" Picking up a V3 include file after preprocessing with V4 GPRE.\n");
|
|
|
|
ib_fprintf(out_file, "#endif\n");
|
|
|
|
ib_fprintf(out_file, "\n/**** end of GPRE definitions ****/\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a call to update metadata.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_ddl( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
// Set up command type for call to RDB$DDL
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (sw_auto) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, 0, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", transaction_name);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "isc_ddl (%s, &%s, &%s, (short) %d, isc_%d);",
|
|
|
|
status_vector(action),
|
|
|
|
request->req_database->dbb_name->sym_string,
|
|
|
|
transaction_name, request->req_length, request->req_ident);
|
|
|
|
|
|
|
|
if (sw_auto) {
|
|
|
|
column -= INDENT;
|
|
|
|
printa(column, "if (!%s [1])", status_name);
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column + INDENT, "isc_commit_transaction (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(action), transaction_name);
|
|
|
|
printa(column, "if (%s [1])", status_name);
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column + INDENT, "isc_rollback_transaction (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(NULL), transaction_name);
|
|
|
|
}
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a call to create a database.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_drop_database( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB db = (DBB) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
|
|
|
|
ib_fprintf(out_file,
|
|
|
|
"isc_drop_database (%s, %d, \"%s\", rdb$k_db_type_gds);",
|
|
|
|
status_vector(action),
|
|
|
|
strlen(db->dbb_filename), db->dbb_filename);
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_close( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_close (%s, %s);",
|
|
|
|
status_name, make_name(s, statement->dyn_cursor_name));
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_declare( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s1[64], s2[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_declare (%s, %s, %s);",
|
|
|
|
status_name,
|
|
|
|
make_name(s1, statement->dyn_statement_name),
|
|
|
|
make_name(s2, statement->dyn_cursor_name));
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_dyn_describe(ACT action,
|
|
|
|
int column,
|
|
|
|
bool bind_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
TEXT s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_describe%s (%s, %s, %d, %s);",
|
|
|
|
bind_flag ? "_bind" : "",
|
|
|
|
status_name,
|
|
|
|
make_name(s, statement->dyn_statement_name),
|
|
|
|
sw_sql_dialect, statement->dyn_sqlda);
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_execute( ACT action, int column)
|
|
|
|
{
|
|
|
|
|
|
|
|
DYN statement = (DYN) action->act_object;
|
|
|
|
TEXT *transaction;
|
|
|
|
TEXT s[64];
|
2003-09-05 12:14:08 +02:00
|
|
|
gpre_req* request;
|
|
|
|
gpre_req req_const;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (statement->dyn_trans) {
|
|
|
|
transaction = statement->dyn_trans;
|
|
|
|
request = &req_const;
|
|
|
|
request->req_trans = transaction;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
transaction = transaction_name;
|
|
|
|
request = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sw_auto) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, request, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", transaction);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (statement->dyn_sqlda2)
|
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_execute2 (%s, &%s, %s, %d, %s, %s);",
|
|
|
|
status_name,
|
|
|
|
transaction,
|
|
|
|
make_name(s, statement->dyn_statement_name),
|
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_STRING,
|
|
|
|
statement->dyn_sqlda2);
|
|
|
|
else
|
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_execute (%s, &%s, %s, %d, %s);",
|
|
|
|
status_name,
|
|
|
|
transaction,
|
|
|
|
make_name(s, statement->dyn_statement_name),
|
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_STRING);
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_fetch( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2003-09-12 18:35:40 +02:00
|
|
|
printa(column, "SQLCODE = isc_embed_dsql_fetch (%s, %s, %d, %s);",
|
|
|
|
status_name, make_name(s, statement->dyn_cursor_name),
|
2001-05-23 15:26:42 +02:00
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_SQLDA);
|
|
|
|
|
|
|
|
printa(column, "if (SQLCODE != 100) SQLCODE = isc_sqlcode (%s);",
|
|
|
|
status_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate code for an EXECUTE IMMEDIATE dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_immediate( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT *transaction;
|
2003-09-05 12:14:08 +02:00
|
|
|
gpre_req* request;
|
|
|
|
gpre_req req_const;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
|
|
|
DBB database = statement->dyn_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->dyn_trans) {
|
|
|
|
transaction = statement->dyn_trans;
|
|
|
|
request = &req_const;
|
|
|
|
request->req_trans = transaction;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
transaction = transaction_name;
|
|
|
|
request = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sw_auto) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, request, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", transaction);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
printa(column,
|
|
|
|
(statement->dyn_sqlda2) ?
|
|
|
|
"isc_embed_dsql_execute_immed2 (%s, &%s, &%s, 0, %s, %d, %s, %s);"
|
|
|
|
: "isc_embed_dsql_execute_immed (%s, &%s, &%s, 0, %s, %d, %s);",
|
|
|
|
status_name, database->dbb_name->sym_string, transaction,
|
|
|
|
statement->dyn_string, sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_SQLDA,
|
|
|
|
(statement->dyn_sqlda2) ? statement->dyn_sqlda2 : NULL_SQLDA);
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_insert( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_insert (%s, %s, %d, %s);",
|
|
|
|
status_name,
|
|
|
|
make_name(s, statement->dyn_cursor_name),
|
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_SQLDA);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_open( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT *transaction, s[64];
|
2003-09-05 12:14:08 +02:00
|
|
|
gpre_req* request;
|
|
|
|
gpre_req req_const;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->dyn_trans) {
|
|
|
|
transaction = statement->dyn_trans;
|
|
|
|
request = &req_const;
|
|
|
|
request->req_trans = transaction;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
transaction = transaction_name;
|
|
|
|
request = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sw_auto) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, request, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", transaction);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
make_name(s, statement->dyn_cursor_name);
|
|
|
|
|
|
|
|
printa(column,
|
|
|
|
(statement->dyn_sqlda2) ?
|
|
|
|
"isc_embed_dsql_open2 (%s, &%s, %s, %d, %s, %s);" :
|
|
|
|
"isc_embed_dsql_open (%s, &%s, %s, %d, %s);",
|
|
|
|
status_name,
|
|
|
|
transaction,
|
|
|
|
s,
|
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_SQLDA,
|
|
|
|
(statement->dyn_sqlda2) ? statement->dyn_sqlda2 : NULL_SQLDA);
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a dynamic SQL statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_dyn_prepare( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[64], *transaction;
|
2003-09-05 12:14:08 +02:00
|
|
|
gpre_req* request;
|
|
|
|
gpre_req req_const;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-08-12 12:05:47 +02:00
|
|
|
DYN statement = (DYN) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (statement->dyn_trans) {
|
|
|
|
transaction = statement->dyn_trans;
|
|
|
|
request = &req_const;
|
|
|
|
request->req_trans = transaction;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
transaction = transaction_name;
|
|
|
|
request = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sw_auto) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, request, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (%s)", transaction);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
2003-08-12 12:05:47 +02:00
|
|
|
DBB database = statement->dyn_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column,
|
|
|
|
"isc_embed_dsql_prepare (%s, &%s, &%s, %s, 0, %s, %d, %s);",
|
|
|
|
status_name, database->dbb_name->sym_string, transaction,
|
|
|
|
make_name(s, statement->dyn_statement_name), statement->dyn_string,
|
|
|
|
sw_sql_dialect,
|
|
|
|
(statement->dyn_sqlda) ? statement->dyn_sqlda : NULL_SQLDA);
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for END_MODIFY.
|
|
|
|
//
|
|
|
|
// Trickier because a fixed subtype single character
|
|
|
|
// field is a single character, not a pointer to a
|
|
|
|
// single character.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_emodify( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s1[20], s2[20];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
UPD modify = (UPD) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (REF reference = modify->upd_port->por_references; reference;
|
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
REF source = reference->ref_source;
|
|
|
|
if (!source)
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_FLD field = reference->ref_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s1, source, true);
|
|
|
|
gen_name(s2, reference, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_dtype > dtype_cstring ||
|
|
|
|
(field->fld_sub_type == 1 && field->fld_length == 1))
|
|
|
|
ib_fprintf(out_file, "%s = %s;", s2, s1);
|
|
|
|
else if (sw_cstring && !field->fld_sub_type)
|
2001-07-12 07:46:06 +02:00
|
|
|
ib_fprintf(out_file, "isc_vtov ((char*)%s, (char*)%s, %d);",
|
2001-05-23 15:26:42 +02:00
|
|
|
s1, s2, field->fld_length);
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, %d, %s, %d);",
|
|
|
|
s1, field->fld_length, s2, field->fld_length);
|
|
|
|
if (field->fld_array_info)
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_get_or_put_slice(action, reference, false, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gen_send(action, modify->upd_port, column);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for END_STORE.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_estore( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// if we did a store ... returning_values aka store2
|
|
|
|
// just wrap up pending error and return
|
|
|
|
|
|
|
|
if (request->req_type == REQ_store2) {
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (action->act_error)
|
|
|
|
column += INDENT;
|
|
|
|
|
|
|
|
align(column);
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_start(action, request->req_primary, column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate definitions associated with a single request.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_endfor( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
column += INDENT;
|
|
|
|
|
|
|
|
if (request->req_sync)
|
|
|
|
gen_send(action, request->req_sync, column);
|
|
|
|
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for ERASE.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_erase( ACT action, int column)
|
|
|
|
{
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
UPD erase = (UPD) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_send(action, erase->upd_port, column);
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate event parameter blocks for use
|
|
|
|
// with a particular call to isc_event_wait.
|
|
|
|
//
|
|
|
|
|
|
|
|
static SSHORT gen_event_block( ACT action)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD init = (GPRE_NOD) action->act_object;
|
|
|
|
SYM event_name = (SYM) init->nod_arg[0];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int ident = CMP_next_ident();
|
2002-11-11 20:19:43 +01:00
|
|
|
init->nod_arg[2] = (GPRE_NOD) ident;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(0, "static %schar\n *isc_%da, *isc_%db;", CONST_STR, ident,
|
|
|
|
ident);
|
|
|
|
printa(0, "static short\n isc_%dl;", ident);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD list = init->nod_arg[1];
|
2001-05-23 15:26:42 +02:00
|
|
|
return list->nod_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for EVENT_INIT.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_event_init( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD *ptr, *end, node;
|
2001-05-23 15:26:42 +02:00
|
|
|
REF reference;
|
|
|
|
PAT args;
|
|
|
|
TEXT variable[20];
|
|
|
|
TEXT
|
|
|
|
* pattern1 =
|
|
|
|
"isc_%N1l = isc_event_block (&isc_%N1a, &isc_%N1b, (short) %N2",
|
|
|
|
*pattern2 =
|
|
|
|
"isc_wait_for_event (%V1, &%DH, isc_%N1l, isc_%N1a, isc_%N1b);",
|
|
|
|
*pattern3 =
|
|
|
|
"isc_event_counts (isc_events, isc_%N1l, isc_%N1a, isc_%N1b);";
|
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD init = (GPRE_NOD) action->act_object;
|
|
|
|
GPRE_NOD event_list = init->nod_arg[1];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
args.pat_database = (DBB) init->nod_arg[3];
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_value1 = (int) init->nod_arg[2];
|
|
|
|
args.pat_value2 = (int) event_list->nod_count;
|
|
|
|
|
|
|
|
// generate call to dynamically generate event blocks
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
|
|
|
|
for (ptr = event_list->nod_arg, end = ptr + event_list->nod_count;
|
|
|
|
ptr < end; ptr++) {
|
|
|
|
node = *ptr;
|
|
|
|
if (node->nod_type == nod_field) {
|
|
|
|
reference = (REF) node->nod_arg[0];
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(variable, reference, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printb(", %s", variable);
|
|
|
|
}
|
|
|
|
else
|
2003-08-09 20:00:14 +02:00
|
|
|
printb(", %s", (TEXT *) node->nod_arg[0]);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
printb(");");
|
|
|
|
|
|
|
|
// generate actual call to event_wait
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern2, &args);
|
|
|
|
|
|
|
|
// get change in event counts, copying event parameter block for reuse
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern3, &args);
|
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for EVENT_WAIT.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_event_wait( ACT action, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT s[64];
|
|
|
|
TEXT
|
|
|
|
* pattern1 =
|
|
|
|
"isc_wait_for_event (%V1, &%DH, isc_%N1l, isc_%N1a, isc_%N1b);",
|
|
|
|
*pattern2 =
|
|
|
|
"isc_event_counts (isc_events, isc_%N1l, isc_%N1a, isc_%N1b);";
|
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
SYM event_name = (SYM) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// go through the stack of events, checking to see if the
|
|
|
|
// event has been initialized and getting the event identifier
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB database;
|
|
|
|
int ident = -1;
|
|
|
|
for (LLS stack_ptr = events; stack_ptr; stack_ptr = stack_ptr->lls_next) {
|
|
|
|
ACT event_action = (ACT) stack_ptr->lls_object;
|
|
|
|
GPRE_NOD event_init = (GPRE_NOD) event_action->act_object;
|
|
|
|
SYM stack_name = (SYM) event_init->nod_arg[0];
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!strcmp(event_name->sym_string, stack_name->sym_string)) {
|
|
|
|
ident = (int) event_init->nod_arg[2];
|
|
|
|
database = (DBB) event_init->nod_arg[3];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ident < 0) {
|
|
|
|
sprintf(s, "event handle \"%s\" not found", event_name->sym_string);
|
|
|
|
IBERROR(s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
args.pat_database = database;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_value1 = (int) ident;
|
|
|
|
|
|
|
|
// generate calls to wait on the event and to fill out the events array
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
PATTERN_expand((USHORT) column, pattern2, &args);
|
|
|
|
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate replacement text for the SQL FETCH statement. The
|
|
|
|
// epilog FETCH statement is handled by GEN_S_FETCH (generate
|
|
|
|
// stream fetch).
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_fetch( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[20];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2003-09-29 14:43:14 +02:00
|
|
|
POR port = request->req_aport;
|
|
|
|
if (port) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* set up the reference to point to the correct value
|
|
|
|
in the linked list of values, and prepare for the
|
|
|
|
next FETCH statement if applicable */
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
REF reference;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (reference = port->por_references; reference;
|
2003-09-29 14:43:14 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
VAL value = reference->ref_values;
|
2001-05-23 15:26:42 +02:00
|
|
|
reference->ref_value = value->val_value;
|
|
|
|
reference->ref_values = value->val_next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find the direction and offset parameters */
|
|
|
|
|
|
|
|
reference = port->por_references;
|
2003-09-29 14:43:14 +02:00
|
|
|
SCHAR* offset = reference->ref_value;
|
2001-05-23 15:26:42 +02:00
|
|
|
reference = reference->ref_next;
|
2003-09-29 14:43:14 +02:00
|
|
|
SCHAR* direction = reference->ref_value;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* the direction in which the engine will scroll is sticky, so check to see
|
|
|
|
the last direction passed to the engine; if the direction is the same and
|
|
|
|
the offset is 1, then there is no need to pass the message; this prevents
|
|
|
|
extra packets and allows for batch fetches in either direction */
|
|
|
|
|
|
|
|
printa(column, "if (isc_%ddirection %% 2 != %s || %s != 1)",
|
|
|
|
request->req_ident, direction, offset);
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* assign the direction and offset parameters to the appropriate message,
|
|
|
|
then send the message to the engine */
|
|
|
|
|
|
|
|
asgn_from(action, port->por_references, column);
|
|
|
|
gen_send(action, port, column);
|
|
|
|
printa(column, "isc_%ddirection = %s;", request->req_ident,
|
|
|
|
direction);
|
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (request->req_sync) {
|
|
|
|
gen_send(action, request->req_sync, column);
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gen_receive(action, column, request->req_primary);
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
column += INDENT;
|
2003-09-10 21:48:53 +02:00
|
|
|
printa(column, "if (%s)", gen_name(s, request->req_eof, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD var_list = (GPRE_NOD) action->act_object;
|
|
|
|
if (var_list)
|
|
|
|
for (int i = 0; i < var_list->nod_count; i++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
asgn_to(action, (REF) (var_list->nod_arg[i]), column);
|
|
|
|
}
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column - INDENT, "else");
|
|
|
|
printa(column, "SQLCODE = 100;");
|
|
|
|
|
|
|
|
if (request->req_sync) {
|
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
if (port) {
|
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for FINISH
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_finish( ACT action, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 = "if (%S2)\n\
|
2002-12-16 16:59:18 +01:00
|
|
|
isc_%S1_transaction (%V1, (FRBRD**) &%S2);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_string2 = transaction_name;
|
|
|
|
|
|
|
|
if (sw_auto || ((action->act_flags & ACT_sql) &&
|
|
|
|
(action->act_type != ACT_disconnect))) {
|
2001-07-12 07:46:06 +02:00
|
|
|
args.pat_string1 = (TEXT*)
|
|
|
|
((action->act_type != ACT_rfinish) ? "commit" : "rollback");
|
2001-05-23 15:26:42 +02:00
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
}
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB db = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// the user supplied one or more db_handles
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (RDY ready = (RDY) action->act_object; ready; ready = ready->rdy_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
db = ready->rdy_database;
|
|
|
|
printa(column, "isc_detach_database (%s, &%s);",
|
|
|
|
status_vector(action), db->dbb_name->sym_string);
|
|
|
|
}
|
|
|
|
// no hanbdles, so we finish all known databases
|
|
|
|
|
|
|
|
if (!db)
|
|
|
|
for (db = isc_databases; db; db = db->dbb_next) {
|
|
|
|
if ((action->act_error || (action->act_flags & ACT_sql)) &&
|
|
|
|
(db != isc_databases))
|
|
|
|
printa(column, "if (%s && !%s [1]) ",
|
|
|
|
db->dbb_name->sym_string, status_name);
|
|
|
|
else
|
|
|
|
printa(column, "if (%s)", db->dbb_name->sym_string);
|
|
|
|
printa(column + INDENT, "isc_detach_database (%s, &%s);",
|
|
|
|
status_vector(action), db->dbb_name->sym_string);
|
|
|
|
}
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for FOR statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_for( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[20];
|
|
|
|
|
|
|
|
gen_s_start(action, column);
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
|
|
|
printa(column, "if (!%s [1]) {", status_name);
|
|
|
|
|
|
|
|
printa(column, "while (1)");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_receive(action, column, request->req_primary);
|
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
|
|
|
printa(column, "if (!%s || %s [1]) break;",
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, request->req_eof, true), status_name);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
printa(column, "if (!%s) break;",
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, request->req_eof, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
if (POR port = action->act_request->req_primary)
|
|
|
|
for (REF reference = port->por_references; reference;
|
2003-09-13 14:22:11 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
if (reference->ref_field->fld_array_info)
|
|
|
|
gen_get_or_put_slice(action, reference, true, column);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a function for free standing ANY or statistical.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_function( ACT function, int column)
|
|
|
|
{
|
|
|
|
REF reference;
|
|
|
|
TEXT *dtype, s[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
ACT action = (ACT) function->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_type != ACT_any) {
|
|
|
|
IBERROR("can't generate function");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-03-27 18:15:48 +01:00
|
|
|
ib_fprintf(out_file, "static %s_r (request, transaction ", request->req_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
POR port = request->req_vport;
|
|
|
|
if (port)
|
2001-05-23 15:26:42 +02:00
|
|
|
for (reference = port->por_references; reference;
|
|
|
|
reference = reference->ref_next)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
|
|
|
ib_fprintf(out_file, ", %s",
|
|
|
|
gen_name(s, reference->ref_source, true));
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ib_fprintf(out_file,
|
2003-03-27 18:15:48 +01:00
|
|
|
")\n isc_req_handle\trequest;\n isc_tr_handle\ttransaction;\n");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (port)
|
|
|
|
for (reference = port->por_references; reference;
|
2003-09-29 14:43:14 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
GPRE_FLD field = reference->ref_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (field->fld_dtype) {
|
|
|
|
case dtype_short:
|
|
|
|
dtype = "short";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_long:
|
|
|
|
dtype = DCL_LONG;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_cstring:
|
|
|
|
case dtype_text:
|
|
|
|
dtype = "char*";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_quad:
|
|
|
|
dtype = DCL_QUAD;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// ** Begin date/time/timestamp *
|
|
|
|
case dtype_sql_date:
|
|
|
|
dtype = "ISC_DATE";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_sql_time:
|
|
|
|
dtype = "ISC_TIME";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_timestamp:
|
|
|
|
dtype = "ISC_TIMESTAMP";
|
|
|
|
break;
|
|
|
|
// ** End date/time/timestamp *
|
|
|
|
|
|
|
|
case dtype_int64:
|
|
|
|
dtype = "ISC_INT64";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_blob:
|
|
|
|
dtype = "ISC_QUAD";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_real:
|
|
|
|
dtype = "float";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_double:
|
|
|
|
dtype = "double";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
IBERROR("gen_function: unsupported datatype");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ib_fprintf(out_file, " %s\t%s;\n", dtype,
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, reference->ref_source, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "{\n");
|
|
|
|
for (port = request->req_ports; port; port = port->por_next)
|
|
|
|
make_port(port, column);
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "\n\n");
|
|
|
|
gen_s_start(action, 0);
|
|
|
|
gen_receive(action, column, request->req_primary);
|
|
|
|
|
|
|
|
for (port = request->req_ports; port; port = port->por_next)
|
|
|
|
for (reference = port->por_references; reference;
|
2003-09-13 14:22:11 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
if (reference->ref_field->fld_array_info)
|
|
|
|
gen_get_or_put_slice(action, reference, true, column);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
port = request->req_primary;
|
|
|
|
ib_fprintf(out_file, "\nreturn %s;\n}\n",
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(s, port->por_references, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a call to isc_get_slice
|
|
|
|
// or isc_put_slice for an array.
|
|
|
|
//
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_get_or_put_slice(ACT action,
|
|
|
|
REF reference,
|
|
|
|
bool get,
|
|
|
|
int column)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT s1[25], s2[10], s4[10];
|
|
|
|
TEXT *pattern1 =
|
2003-02-28 14:20:36 +01:00
|
|
|
"isc_get_slice (%V1, &%DH, &%RT, &%S2, (short) %N1, (char *) %S3, 0, (%S6*) 0, (%S6) %L1, %S5, &isc_array_length);";
|
2001-05-23 15:26:42 +02:00
|
|
|
TEXT *pattern2 =
|
2003-02-28 14:20:36 +01:00
|
|
|
"isc_put_slice (%V1, &%DH, &%RT, &%S2, (short) %N1, (char *) %S3, 0, (%S6*) 0, (%S6) %L1, (void *)%S5);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!(reference->ref_flags & REF_fetch_array))
|
|
|
|
return;
|
|
|
|
|
|
|
|
args.pat_request = action->act_request;
|
2003-09-11 04:13:46 +02:00
|
|
|
args.pat_condition = get; // get or put slice
|
2003-09-10 21:48:53 +02:00
|
|
|
args.pat_vector1 = status_vector(action); // status vector
|
|
|
|
args.pat_database = action->act_request->req_database; // database handle
|
|
|
|
gen_name(s1, reference, true); // blob handle
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_string2 = s1;
|
2003-09-10 21:48:53 +02:00
|
|
|
args.pat_value1 = reference->ref_sdl_length; // slice description length
|
|
|
|
sprintf(s2, "isc_%d", reference->ref_sdl_ident); // slice description
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_string3 = s2;
|
|
|
|
|
|
|
|
args.pat_long1 = reference->ref_field->fld_array_info->ary_size;
|
|
|
|
/* slice size */
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql)
|
|
|
|
args.pat_string5 = reference->ref_value;
|
|
|
|
else {
|
|
|
|
sprintf(s4, "isc_%d",
|
|
|
|
reference->ref_field->fld_array_info->ary_ident);
|
|
|
|
args.pat_string5 = s4; /* array name */
|
|
|
|
}
|
|
|
|
|
|
|
|
args.pat_string6 = DCL_LONG;
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, (get) ? pattern1 : pattern2, &args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (action->act_flags & ACT_sql)
|
|
|
|
gen_whenever(action->act_whenever, column);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the code to do a get segment.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_get_segment( ACT action, int column)
|
|
|
|
{
|
|
|
|
BLB blob;
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 =
|
|
|
|
"%IF%S1 [1] = %ENisc_get_segment (%V1, &%BH, &%I1, (short) sizeof (%I2), %I2);";
|
|
|
|
|
|
|
|
if (action->act_error && (action->act_type != ACT_blob_for))
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql)
|
|
|
|
blob = (BLB) action->act_request->req_blobs;
|
|
|
|
else
|
|
|
|
blob = (BLB) action->act_object;
|
|
|
|
|
|
|
|
args.pat_blob = blob;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_condition = !(action->act_error
|
|
|
|
|| (action->act_flags & ACT_sql));
|
|
|
|
args.pat_ident1 = blob->blb_len_ident;
|
|
|
|
args.pat_ident2 = blob->blb_buff_ident;
|
|
|
|
args.pat_string1 = status_name;
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
2003-09-29 14:43:14 +02:00
|
|
|
REF into = action->act_object;
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (!SQLCODE || SQLCODE == 101)");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "isc_ftof (isc_%d, isc_%d, %s, isc_%d);",
|
|
|
|
blob->blb_buff_ident, blob->blb_len_ident,
|
|
|
|
into->ref_value, blob->blb_len_ident);
|
|
|
|
if (into->ref_null_value) {
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "%s = isc_%d;",
|
|
|
|
into->ref_null_value, blob->blb_len_ident);
|
|
|
|
}
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text to compile and start a SQL command
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_loop( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT name[20];
|
|
|
|
|
|
|
|
gen_s_start(action, column);
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
|
|
|
POR port = request->req_primary;
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (!SQLCODE) ");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_receive(action, column, port);
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(name, port->por_references, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (!SQLCODE && !%s)", name);
|
|
|
|
printa(column + INDENT, "SQLCODE = 100;");
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a name for a reference. Name is constructed from
|
|
|
|
// port and parameter idents.
|
|
|
|
//
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
static TEXT *gen_name(char *string,
|
|
|
|
REF reference,
|
|
|
|
bool as_blob)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
if (reference->ref_field->fld_array_info && !as_blob)
|
|
|
|
sprintf(string, "isc_%d",
|
|
|
|
reference->ref_field->fld_array_info->ary_ident);
|
|
|
|
else if (reference->ref_port)
|
|
|
|
sprintf(string, "isc_%d.isc_%d",
|
|
|
|
reference->ref_port->por_ident, reference->ref_ident);
|
|
|
|
else
|
|
|
|
sprintf(string, "isc_%d", reference->ref_ident);
|
|
|
|
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a block to handle errors.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_on_error( ACT action, USHORT column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
ACT err_action = (ACT) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((err_action->act_type == ACT_get_segment) ||
|
|
|
|
(err_action->act_type == ACT_put_segment) ||
|
|
|
|
(err_action->act_type == ACT_endblob))
|
|
|
|
printa(column,
|
|
|
|
"if (%s [1] && (%s [1] != isc_segment) && (%s [1] != isc_segstr_eof))",
|
|
|
|
status_name, status_name, status_name);
|
|
|
|
else
|
|
|
|
printa(column, "if (%s [1])", status_name);
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate code for an EXECUTE PROCEDURE.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_procedure( ACT action, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern;
|
|
|
|
|
|
|
|
column += INDENT;
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
|
|
|
POR in_port = request->req_vport;
|
|
|
|
POR out_port = request->req_primary;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
dbb* database = request->req_database;
|
2003-09-05 12:14:08 +02:00
|
|
|
args.pat_database = database;
|
2001-05-23 15:26:42 +02:00
|
|
|
args.pat_request = action->act_request;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_request = request;
|
|
|
|
args.pat_port = in_port;
|
|
|
|
args.pat_port2 = out_port;
|
|
|
|
if (in_port && in_port->por_length)
|
|
|
|
pattern =
|
|
|
|
"isc_transact_request (%V1, %RF%DH%RE, %RF%RT%RE, sizeof (%RI), %RI, (short) %PL, (char *) %RF%PI%RE, (short) %QL, (char *) %RF%QI%RE);";
|
|
|
|
else
|
|
|
|
pattern =
|
|
|
|
"isc_transact_request (%V1, %RF%DH%RE, %RF%RT%RE, sizeof (%RI), %RI, 0, 0, (short) %QL, (char *) %RF%QI%RE);";
|
|
|
|
|
|
|
|
// Get database attach and transaction started
|
|
|
|
|
|
|
|
if (sw_auto)
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, 0, status_vector(action), column, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// Move in input values
|
|
|
|
|
|
|
|
asgn_from(action, request->req_values, column);
|
|
|
|
|
|
|
|
// Execute the procedure
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column, pattern, &args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// Move out output values
|
|
|
|
|
|
|
|
asgn_to_proc(request->req_references, column);
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the code to do a put segment.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_put_segment( ACT action, int column)
|
|
|
|
{
|
|
|
|
BLB blob;
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 = "%IF%S1 [1] = %ENisc_put_segment (%V1, &%BH, %I1, %I2);";
|
|
|
|
|
|
|
|
if (!action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql) {
|
|
|
|
blob = (BLB) action->act_request->req_blobs;
|
2003-09-29 14:43:14 +02:00
|
|
|
REF from = action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "isc_%d = %s;",
|
|
|
|
blob->blb_len_ident, from->ref_null_value);
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "isc_ftof (%s, isc_%d, isc_%d, isc_%d);",
|
|
|
|
from->ref_value, blob->blb_len_ident,
|
|
|
|
blob->blb_buff_ident, blob->blb_len_ident);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
blob = (BLB) action->act_object;
|
|
|
|
|
|
|
|
args.pat_blob = blob;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_condition = !(action->act_error
|
|
|
|
|| (action->act_flags & ACT_sql));
|
|
|
|
args.pat_ident1 = blob->blb_len_ident;
|
|
|
|
args.pat_ident2 = blob->blb_buff_ident;
|
|
|
|
args.pat_string1 = status_name;
|
|
|
|
PATTERN_expand((USHORT) column, pattern1, &args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_flags & ACT_sql)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate BLR/MBLR/etc. in raw, numeric form. Ugly but dense.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_raw( UCHAR * blr, int request_length)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT buffer[80];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* p = buffer;
|
|
|
|
TEXT* limit = buffer + 60;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (int count = request_length; count; count--) {
|
|
|
|
const TEXT c = *blr++;
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((c >= 'A' && c <= 'Z') || c == '$' || c == '_')
|
|
|
|
sprintf(p, "'%c'", c);
|
|
|
|
else
|
|
|
|
sprintf(p, "%d", c);
|
|
|
|
while (*p)
|
|
|
|
p++;
|
|
|
|
if (count - 1)
|
|
|
|
*p++ = ',';
|
|
|
|
if (p < limit)
|
|
|
|
continue;
|
|
|
|
*p = 0;
|
|
|
|
printa(INDENT, buffer);
|
|
|
|
p = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p = 0;
|
|
|
|
printa(INDENT, buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for READY
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_ready( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* vector = status_vector(action);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (RDY ready = (RDY) action->act_object; ready; ready = ready->rdy_next) {
|
|
|
|
DBB db = ready->rdy_database;
|
|
|
|
TEXT* filename = ready->rdy_filename;
|
|
|
|
if (!filename)
|
2001-05-23 15:26:42 +02:00
|
|
|
filename = db->dbb_runtime;
|
|
|
|
if ((action->act_error || (action->act_flags & ACT_sql)) &&
|
|
|
|
ready != (RDY) action->act_object)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (!%s [1]) {", status_name);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
make_ready(db, filename, vector, (USHORT) column, ready->rdy_request);
|
|
|
|
if ((action->act_error || (action->act_flags & ACT_sql)) &&
|
|
|
|
ready != (RDY) action->act_object)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a send or receive call for a port.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_receive( ACT action, int column, POR port)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern =
|
2002-12-16 16:59:18 +01:00
|
|
|
"isc_receive (%V1, (FRBRD**) &%RH, (short) %PN, (short) %PL, &%PI, (short) %RL);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
args.pat_request = action->act_request;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_port = port;
|
|
|
|
PATTERN_expand((USHORT) column, pattern, &args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for RELEASE_REQUESTS
|
|
|
|
// For active databases, call isc_release_request.
|
|
|
|
// for all others, just zero the handle. For the
|
|
|
|
// release request calls, ignore error returns, which
|
|
|
|
// are likely if the request was compiled on a database
|
|
|
|
// which has been released and re-readied. If there is
|
|
|
|
// a serious error, it will be caught on the next statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_release( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB exp_db = (DBB) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (GPRE_REQ request = requests; request; request = request->req_next) {
|
|
|
|
DBB db = request->req_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (exp_db && db != exp_db)
|
|
|
|
continue;
|
|
|
|
if (db && request->req_handle && !(request->req_flags & REQ_exp_hand)) {
|
|
|
|
printa(column, "if (%s && %s)",
|
|
|
|
db->dbb_name->sym_string, request->req_handle);
|
|
|
|
printa(column + INDENT, "isc_release_request (%s, &%s);",
|
|
|
|
status_name, request->req_handle);
|
|
|
|
printa(column, "%s = 0;", request->req_handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate definitions associated with a single request.
|
|
|
|
//
|
|
|
|
|
2002-11-17 01:04:19 +01:00
|
|
|
static void gen_request( GPRE_REQ request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-09-11 04:13:46 +02:00
|
|
|
if (!(request->req_flags & (REQ_exp_hand | REQ_sql_blob_open
|
|
|
|
| REQ_sql_blob_create)) && request->req_type != REQ_slice
|
|
|
|
&& request->req_type != REQ_procedure)
|
|
|
|
{
|
|
|
|
printa(0, "static isc_req_handle\n %s = 0;\t\t/* request handle */\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_handle);
|
2003-09-11 04:13:46 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// check the case where we need to extend the dpb dynamically at runtime,
|
|
|
|
// in which case we need dpb length and a pointer to be defined even if
|
|
|
|
// there is no static dpb defined
|
|
|
|
|
|
|
|
if (request->req_flags & REQ_extend_dpb) {
|
|
|
|
printa(0, "static char\n *isc_%dp;", request->req_ident);
|
|
|
|
if (!request->req_length)
|
|
|
|
printa(0, "static short\n isc_%dl = %d;", request->req_ident,
|
|
|
|
request->req_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request->req_type == REQ_create_database)
|
|
|
|
printa(0, "static %s\n *isc_%dt;", DCL_LONG, request->req_ident);
|
|
|
|
|
|
|
|
if (request->req_flags & (REQ_sql_blob_open | REQ_sql_blob_create))
|
|
|
|
printa(0,
|
|
|
|
"static isc_stmt_handle\n isc_%ds;\t\t/* sql statement handle */",
|
|
|
|
request->req_ident);
|
|
|
|
|
|
|
|
if (request->req_length) {
|
|
|
|
if (request->req_flags & REQ_sql_cursor)
|
|
|
|
printa(0,
|
|
|
|
"static isc_stmt_handle\n isc_%ds;\t\t/* sql statement handle */",
|
|
|
|
request->req_ident);
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
if (request->req_flags & REQ_scroll)
|
|
|
|
printa(0,
|
|
|
|
"static short\n isc_%ddirection;\t\t/* last direction sent to engine */",
|
|
|
|
request->req_ident);
|
|
|
|
#endif
|
|
|
|
printa(0, "static %sshort\n isc_%dl = %d;",
|
|
|
|
(request->req_flags & REQ_extend_dpb) ? "" : CONST_STR,
|
|
|
|
request->req_ident, request->req_length);
|
|
|
|
printa(0, "static %schar\n isc_%d [] = {",
|
2003-03-27 18:15:48 +01:00
|
|
|
CONST_STR, request->req_ident);
|
2003-09-29 14:43:14 +02:00
|
|
|
const TEXT* string_type = "blr";
|
2001-05-23 15:26:42 +02:00
|
|
|
if (sw_raw) {
|
|
|
|
gen_raw(request->req_blr, request->req_length);
|
|
|
|
switch (request->req_type) {
|
|
|
|
case REQ_create_database:
|
|
|
|
case REQ_ready:
|
|
|
|
string_type = "dpb";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REQ_ddl:
|
|
|
|
string_type = "dyn";
|
|
|
|
break;
|
|
|
|
case REQ_slice:
|
|
|
|
string_type = "sdl";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
string_type = "blr";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
switch (request->req_type) {
|
|
|
|
case REQ_create_database:
|
|
|
|
case REQ_ready:
|
|
|
|
string_type = "dpb";
|
2003-09-28 23:36:05 +02:00
|
|
|
if (PRETTY_print_cdb(request->req_blr, gen_blr, 0, 0))
|
2003-09-11 04:13:46 +02:00
|
|
|
{
|
|
|
|
IBERROR("internal error during parameter generation");
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case REQ_ddl:
|
|
|
|
string_type = "dyn";
|
2003-09-28 23:36:05 +02:00
|
|
|
if (PRETTY_print_dyn(request->req_blr, gen_blr, 0, 0))
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2003-09-11 04:13:46 +02:00
|
|
|
IBERROR("internal error during dynamic DDL generation");
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
case REQ_slice:
|
|
|
|
string_type = "sdl";
|
2003-09-28 23:36:05 +02:00
|
|
|
if (PRETTY_print_sdl(request->req_blr, gen_blr, 0, 0))
|
2003-09-11 04:13:46 +02:00
|
|
|
{
|
|
|
|
IBERROR("internal error during SDL generation");
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
string_type = "blr";
|
2003-09-28 23:36:05 +02:00
|
|
|
if (gds__print_blr(request->req_blr, gen_blr, 0, 0))
|
2003-09-11 04:13:46 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
IBERROR("internal error during BLR generation");
|
2003-09-11 04:13:46 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
printa(INDENT, "};\t/* end of %s string for request isc_%d */\n",
|
2003-03-27 18:15:48 +01:00
|
|
|
string_type, request->req_ident);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Print out slice description language if there are arrays associated with request
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (POR port = request->req_ports; port; port = port->por_next)
|
|
|
|
for (REF reference = port->por_references; reference;
|
2003-09-11 04:13:46 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
if (reference->ref_sdl) {
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(0, "static %sshort\n isc_%dl = %d;", CONST_STR,
|
|
|
|
reference->ref_sdl_ident, reference->ref_sdl_length);
|
|
|
|
printa(0, "static %schar\n isc_%d [] = {", CONST_STR,
|
|
|
|
reference->ref_sdl_ident);
|
|
|
|
if (sw_raw)
|
|
|
|
gen_raw((UCHAR *) reference->ref_sdl,
|
|
|
|
reference->ref_sdl_length);
|
|
|
|
else
|
2003-09-28 23:36:05 +02:00
|
|
|
if (PRETTY_print_sdl(reference->ref_sdl, gen_blr, 0, 0))
|
2003-09-11 04:13:46 +02:00
|
|
|
{
|
|
|
|
IBERROR("internal error during SDL generation");
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(INDENT,
|
|
|
|
"};\t/* end of sdl string for request isc_%d */\n",
|
|
|
|
reference->ref_sdl_ident);
|
|
|
|
}
|
2003-09-11 04:13:46 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// Print out any blob parameter blocks required
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (BLB blob = request->req_blobs; blob; blob = blob->blb_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (blob->blb_bpb_length) {
|
|
|
|
printa(0, "static %schar\n isc_%d [] = {", CONST_STR,
|
|
|
|
blob->blb_bpb_ident);
|
|
|
|
gen_raw(blob->blb_bpb, blob->blb_bpb_length);
|
|
|
|
printa(INDENT, "};\n");
|
|
|
|
}
|
|
|
|
// If this is a GET_SLICE/PUT_slice, allocate some variables
|
|
|
|
|
|
|
|
if (request->req_type == REQ_slice) {
|
|
|
|
printa(0, "static %s", DCL_LONG);
|
|
|
|
printa(INDENT, "isc_%dv [%d],", request->req_ident,
|
|
|
|
MAX(request->req_slice->slc_parameters, 1));
|
|
|
|
printa(INDENT, "isc_%ds;", request->req_ident);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate receive call for a port
|
|
|
|
// in a store2 statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_return_value( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (action->act_pair->act_error)
|
|
|
|
column += INDENT;
|
|
|
|
align(column);
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_start(action, request->req_primary, column, true);
|
2003-09-29 14:43:14 +02:00
|
|
|
UPD update = (UPD) action->act_object;
|
|
|
|
REF reference = update->upd_references;
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_receive(action, column, reference->ref_port);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Process routine head. If there are requests in the
|
|
|
|
// routine, insert local definitions.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_routine( ACT action, int column)
|
|
|
|
{
|
|
|
|
column += INDENT;
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (GPRE_REQ request = (GPRE_REQ) action->act_object; request;
|
|
|
|
request = request->req_routine)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (request->req_type == REQ_any)
|
|
|
|
continue;
|
2003-09-29 14:43:14 +02:00
|
|
|
for (POR port = request->req_ports; port; port = port->por_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
make_port(port, column);
|
2003-09-29 14:43:14 +02:00
|
|
|
for (BLB blob = request->req_blobs; blob; blob = blob->blb_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "isc_blob_handle\t\tisc_%d;\t\t/* blob handle */",
|
|
|
|
blob->blb_ident);
|
|
|
|
printa(column, "char\t\t\tisc_%d [%d];\t/* blob segment */",
|
|
|
|
blob->blb_buff_ident, blob->blb_seg_length);
|
|
|
|
printa(column, "unsigned short\tisc_%d;\t\t/* segment length */",
|
|
|
|
blob->blb_len_ident);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for END_STREAM.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_s_end( ACT action, int column)
|
|
|
|
{
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2003-09-29 14:43:14 +02:00
|
|
|
|
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_type == ACT_close)
|
|
|
|
column = gen_cursor_close(action, request, column);
|
|
|
|
|
|
|
|
printa(column, "isc_unwind_request (%s, &%s, %s);",
|
|
|
|
status_vector(action),
|
|
|
|
request->req_handle, request->req_request_level);
|
|
|
|
|
|
|
|
if (action->act_type == ACT_close) {
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for FETCH.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_s_fetch( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (request->req_sync)
|
|
|
|
gen_send(action, request->req_sync, column);
|
|
|
|
|
|
|
|
gen_receive(action, column, request->req_primary);
|
|
|
|
|
|
|
|
if (!action->act_pair && !action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text to compile and start a stream. This is
|
|
|
|
// used both by START_STREAM and FOR
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_s_start( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gen_compile(action, column);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
POR port = request->req_vport;
|
|
|
|
if (port)
|
2001-05-23 15:26:42 +02:00
|
|
|
asgn_from(action, port->por_references, column);
|
|
|
|
|
|
|
|
if (action->act_type == ACT_open)
|
|
|
|
column = gen_cursor_open(action, request, column);
|
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql)) {
|
|
|
|
make_ok_test(action, request, column);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_start(action, port, column, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql))
|
|
|
|
column -= INDENT;
|
|
|
|
|
|
|
|
if (action->act_type == ACT_open) {
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
column -= INDENT;
|
|
|
|
}
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Substitute for a segment, segment length, or blob handle.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_segment( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
BLB blob = (BLB) action->act_object;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(column, "isc_%d",
|
|
|
|
(action->act_type == ACT_segment) ? blob->blb_buff_ident :
|
|
|
|
(action->act_type == ACT_segment_length) ? blob->blb_len_ident :
|
|
|
|
blob->blb_ident);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// generate code for a singleton select.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_select( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT name[20];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
|
|
|
POR port = request->req_primary;
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(name, request->req_eof, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gen_s_start(action, column);
|
|
|
|
printa(column, "if (!SQLCODE) ");
|
|
|
|
column += INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
gen_receive(action, column, port);
|
|
|
|
printa(column, "if (!SQLCODE)");
|
|
|
|
column += INDENT;
|
|
|
|
printa(column, "if (%s)", name);
|
|
|
|
column += INDENT;
|
|
|
|
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_NOD var_list = (GPRE_NOD) action->act_object;
|
|
|
|
if (var_list)
|
|
|
|
for (int i = 0; i < var_list->nod_count; i++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
asgn_to(action, (REF) var_list->nod_arg[i], column);
|
|
|
|
}
|
|
|
|
if (request->req_database->dbb_flags & DBB_v3) {
|
|
|
|
gen_receive(action, column, port);
|
|
|
|
printa(column, "if (!SQLCODE && %s)", name);
|
|
|
|
printa(column + INDENT, "SQLCODE = -1;");
|
|
|
|
}
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(column - INDENT, "else");
|
|
|
|
printa(column, "SQLCODE = 100;");
|
|
|
|
column -= INDENT;
|
|
|
|
column -= INDENT;
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a send or receive call for a port.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_send( ACT action, POR port, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern =
|
2002-12-16 16:59:18 +01:00
|
|
|
"isc_send (%V1, (FRBRD**) &%RH, (short) %PN, (short) %PL, &%PI, (short) %RL);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
args.pat_request = action->act_request;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_port = port;
|
|
|
|
PATTERN_expand((USHORT) column, pattern, &args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate support for get/put slice statement.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_slice( ACT action, REF var_reference, int column)
|
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 = "isc_get_slice (%V1, &%DH, &%RT, &%FR, (short) %N1, \
|
2003-02-28 14:20:36 +01:00
|
|
|
(char *) %I1, (short) %N2, %I1v, %I1s, %S5, &isc_array_length);";
|
2001-05-23 15:26:42 +02:00
|
|
|
TEXT *pattern2 = "isc_put_slice (%V1, &%DH, &%RT, &%FR, (short) %N1, \
|
2003-02-28 14:20:36 +01:00
|
|
|
(char *) %I1, (short) %N2, %I1v, %I1s, %S5);";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
|
|
|
SLC slice = (SLC) action->act_object;
|
|
|
|
GPRE_REQ parent_request = slice->slc_parent_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// Compute array size
|
|
|
|
|
|
|
|
printa(column, "isc_%ds = %d", request->req_ident,
|
|
|
|
slice->slc_field->fld_array->fld_length);
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
slc::slc_repeat *tail, *end;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (tail = slice->slc_rpt, end = tail + slice->slc_dimensions;
|
|
|
|
tail < end; ++tail)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (tail->slc_upper != tail->slc_lower) {
|
2003-09-29 14:43:14 +02:00
|
|
|
REF lower = (REF) tail->slc_lower->nod_arg[0];
|
|
|
|
REF upper = (REF) tail->slc_upper->nod_arg[0];
|
2001-05-23 15:26:42 +02:00
|
|
|
if (lower->ref_value)
|
|
|
|
ib_fprintf(out_file, " * ( %s - %s + 1)", upper->ref_value,
|
|
|
|
lower->ref_value);
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file, " * ( %s + 1)", upper->ref_value);
|
|
|
|
}
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ib_fprintf(out_file, ";");
|
|
|
|
|
|
|
|
// Make assignments to variable vector
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
REF reference;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (reference = request->req_values; reference;
|
2003-09-29 14:43:14 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
printa(column, "isc_%dv [%d] = %s;",
|
|
|
|
request->req_ident, reference->ref_id, reference->ref_value);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
args.pat_reference =
|
|
|
|
(var_reference ? var_reference : slice->slc_field_ref);
|
|
|
|
args.pat_request = parent_request; /* blob id request */
|
|
|
|
args.pat_vector1 = status_vector(action); /* status vector */
|
|
|
|
args.pat_database = parent_request->req_database; /* database handle */
|
|
|
|
args.pat_string1 = action->act_request->req_trans; /* transaction handle */
|
|
|
|
args.pat_value1 = request->req_length; /* slice descr. length */
|
|
|
|
args.pat_ident1 = request->req_ident; /* request name */
|
|
|
|
args.pat_value2 = slice->slc_parameters * sizeof(SLONG); /* parameter length */
|
|
|
|
|
|
|
|
if (!(reference = var_reference))
|
|
|
|
reference = (REF) slice->slc_array->nod_arg[0];
|
|
|
|
args.pat_string5 = reference->ref_value; /* array name */
|
|
|
|
|
|
|
|
PATTERN_expand((USHORT) column,
|
|
|
|
(action->act_type == ACT_get_slice) ? pattern1 : pattern2,
|
|
|
|
&args);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (action->act_flags & ACT_sql)
|
|
|
|
gen_whenever(action->act_whenever, column);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate either a START or START_AND_SEND depending
|
|
|
|
// on whether or a not a port is present. If this START
|
|
|
|
// or START_AND_SEND is being generated for a STORE or a
|
|
|
|
// MODIFY statement, generate PUT_SLICE calls, as well.
|
|
|
|
//
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
static void gen_start(ACT action,
|
|
|
|
POR port,
|
|
|
|
int column,
|
|
|
|
bool sending)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
PAT args;
|
|
|
|
TEXT *pattern1 =
|
2002-12-16 16:59:18 +01:00
|
|
|
"isc_start_and_send (%V1, (FRBRD**) &%RH, (FRBRD**) &%S1, (short) %PN, (short) %PL, &%PI, (short) %RL);";
|
|
|
|
TEXT *pattern2 = "isc_start_request (%V1, (FRBRD**) &%RH, (FRBRD**) &%S1, (short) %RL);";
|
2001-05-23 15:26:42 +02:00
|
|
|
REF reference;
|
|
|
|
|
|
|
|
if (port && sending) {
|
|
|
|
for (reference = port->por_references; reference;
|
2003-09-13 14:22:11 +02:00
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
if (reference->ref_field->fld_array_info)
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_get_or_put_slice(action, reference, false, column);
|
2003-09-13 14:22:11 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
args.pat_request = action->act_request;
|
|
|
|
args.pat_vector1 = status_vector(action);
|
|
|
|
args.pat_port = port;
|
|
|
|
args.pat_string1 = request_trans(action, action->act_request);
|
|
|
|
PATTERN_expand((USHORT) column, (port) ? pattern1 : pattern2, &args);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate text for STORE statement. This includes the compile
|
|
|
|
// call and any variable initialization required.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_store( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT name[64];
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_REQ request = action->act_request;
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
gen_compile(action, column);
|
|
|
|
|
|
|
|
if (action->act_error || (action->act_flags & ACT_sql)) {
|
|
|
|
make_ok_test(action, request, column);
|
|
|
|
column += INDENT;
|
|
|
|
if (action->act_error)
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize any blob fields
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
POR port = request->req_primary;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (REF reference = port->por_references; reference;
|
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
|
|
|
GPRE_FLD field = reference->ref_field;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (field->fld_flags & FLD_blob)
|
|
|
|
printa(column, "%s = isc_blob_null;",
|
2003-09-10 21:48:53 +02:00
|
|
|
gen_name(name, reference, true));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for START_TRANSACTION.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_t_start( ACT action, int column)
|
|
|
|
{
|
2002-11-17 01:04:19 +01:00
|
|
|
GPRE_TRA trans;
|
2003-09-05 12:14:08 +02:00
|
|
|
tpb* tpb_iterator;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* vector = status_vector(action);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// if this is a purely default transaction, just let it through
|
|
|
|
|
2002-11-17 01:04:19 +01:00
|
|
|
if (!action || !(trans = (GPRE_TRA) action->act_object)) {
|
2003-09-10 21:48:53 +02:00
|
|
|
t_start_auto(action, 0, vector, column, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// build a complete statement, including tpb's. Ready db's
|
|
|
|
|
|
|
|
if (sw_auto)
|
2003-09-05 12:14:08 +02:00
|
|
|
for (tpb_iterator = trans->tra_tpb;
|
|
|
|
tpb_iterator;
|
|
|
|
tpb_iterator = tpb_iterator->tpb_tra_next)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
DBB db = tpb_iterator->tpb_database;
|
|
|
|
TEXT* filename = db->dbb_runtime;
|
|
|
|
if (filename || !(db->dbb_flags & DBB_sqlca)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
printa(column, "if (!%s)", db->dbb_name->sym_string);
|
|
|
|
make_ready(db, filename, vector, (USHORT) (column + INDENT),
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column, "isc_start_transaction (%s, (FRBRD**) &%s, (short) %d",
|
2001-05-23 15:26:42 +02:00
|
|
|
vector,
|
|
|
|
(trans->tra_handle) ? trans->tra_handle : transaction_name,
|
|
|
|
trans->tra_db_count);
|
|
|
|
|
|
|
|
// Some systems don't like infinitely long lines. Limit them to 256.
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int remaining = 256 - column - strlen(vector) -
|
2001-05-23 15:26:42 +02:00
|
|
|
strlen((trans->tra_handle) ? trans->tra_handle : transaction_name) -
|
|
|
|
31;
|
|
|
|
|
2003-09-05 12:14:08 +02:00
|
|
|
for (tpb_iterator = trans->tra_tpb;
|
|
|
|
tpb_iterator;
|
|
|
|
tpb_iterator = tpb_iterator->tpb_tra_next)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
int length = strlen(tpb_iterator->tpb_database->dbb_name->sym_string)
|
|
|
|
+ 22;
|
|
|
|
if (length > remaining) {
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column + INDENT);
|
|
|
|
remaining = 256 - column - INDENT;
|
|
|
|
}
|
|
|
|
remaining -= length;
|
|
|
|
ib_fprintf(out_file, ", &%s, (short) %d, isc_tpb_%d",
|
2003-09-05 12:14:08 +02:00
|
|
|
tpb_iterator->tpb_database->dbb_name->sym_string,
|
|
|
|
tpb_iterator->tpb_length, tpb_iterator->tpb_ident);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, ");");
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a TPB in the output file
|
|
|
|
//
|
|
|
|
|
2003-09-05 12:14:08 +02:00
|
|
|
static void gen_tpb(tpb* tpb_buffer, int column)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT buffer[80];
|
|
|
|
SSHORT length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
printa(column, "static %schar\n", CONST_STR);
|
|
|
|
column += INDENT;
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* p = buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (length = 0; length < column; length++)
|
|
|
|
*p++ = ' ';
|
|
|
|
|
2003-09-05 12:14:08 +02:00
|
|
|
sprintf(p, "isc_tpb_%d [%d] = {", tpb_buffer->tpb_ident, tpb_buffer->tpb_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
while (*p)
|
|
|
|
p++;
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
SSHORT tpb_length = tpb_buffer->tpb_length;
|
|
|
|
TEXT* text = (TEXT*) tpb_buffer->tpb_string;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (--tpb_length >= 0) {
|
2003-09-29 14:43:14 +02:00
|
|
|
const TEXT c = *text++;
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((c >= 'A' && c <= 'Z') || c == '$' || c == '_')
|
|
|
|
sprintf(p, "'%c'", c);
|
|
|
|
else
|
|
|
|
sprintf(p, "%d", c);
|
|
|
|
while (*p)
|
|
|
|
p++;
|
|
|
|
if (tpb_length)
|
|
|
|
*p++ = ',';
|
|
|
|
if (p - buffer > 60) {
|
|
|
|
*p = 0;
|
|
|
|
ib_fprintf(out_file, " %s\n", buffer);
|
|
|
|
p = buffer;
|
|
|
|
for (length = 0; length < column + INDENT; length++)
|
|
|
|
*p++ = ' ';
|
|
|
|
*p = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "%s};\n", buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for COMMIT, ROLLBACK, PREPARE, and SAVE
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_trans( ACT action, int column)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (action->act_type == ACT_commit_retain_context)
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column, "isc_commit_retaining (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector(action),
|
|
|
|
(action->act_object) ? (TEXT *) (action->
|
|
|
|
act_object) :
|
|
|
|
transaction_name);
|
|
|
|
else
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column, "isc_%s_transaction (%s, (FRBRD**) &%s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
(action->act_type ==
|
|
|
|
ACT_commit) ? "commit" : (action->act_type ==
|
|
|
|
ACT_rollback) ? "rollback" :
|
|
|
|
"prepare", status_vector(action),
|
|
|
|
(action->act_object) ? (TEXT *) (action->
|
|
|
|
act_object) :
|
|
|
|
transaction_name);
|
|
|
|
|
2003-09-12 18:35:40 +02:00
|
|
|
set_sqlcode(action, column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Substitute for a variable reference.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_type( ACT action, int column)
|
|
|
|
{
|
|
|
|
|
2003-08-09 20:00:14 +02:00
|
|
|
printa(column, "%ld", (SLONG) action->act_object);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for UPDATE ... WHERE CURRENT OF ...
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_update( ACT action, int column)
|
|
|
|
{
|
2003-09-29 14:43:14 +02:00
|
|
|
UPD modify = (UPD) action->act_object;
|
|
|
|
POR port = modify->upd_port;
|
2001-05-23 15:26:42 +02:00
|
|
|
asgn_from(action, port->por_references, column);
|
|
|
|
gen_send(action, port, column);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Substitute for a variable reference.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_variable( ACT action, int column)
|
|
|
|
{
|
|
|
|
TEXT s[20];
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
printa(column, gen_name(s, action->act_object, false));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate tests for any WHENEVER clauses that may have been declared.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void gen_whenever( SWE label, int column)
|
|
|
|
{
|
|
|
|
TEXT *condition;
|
|
|
|
|
|
|
|
while (label) {
|
|
|
|
switch (label->swe_condition) {
|
|
|
|
case SWE_error:
|
|
|
|
condition = "SQLCODE < 0";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SWE_warning:
|
|
|
|
condition = "SQLCODE > 0 && SQLCODE != 100";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SWE_not_found:
|
|
|
|
condition = "SQLCODE == 100";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "if (%s) goto %s;", condition, label->swe_label);
|
|
|
|
label = label->swe_next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate a declaration of an array in the
|
|
|
|
// output file.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void make_array_declaration( REF reference)
|
|
|
|
{
|
|
|
|
|
2002-11-30 18:45:02 +01:00
|
|
|
GPRE_FLD field = reference->ref_field;
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* name = field->fld_symbol->sym_string;
|
2001-05-23 15:26:42 +02:00
|
|
|
TEXT s[64];
|
|
|
|
TEXT *dtype;
|
|
|
|
|
|
|
|
// Don't generate multiple declarations for the array. V3 Bug 569.
|
|
|
|
|
|
|
|
if (field->fld_array_info->ary_declared)
|
|
|
|
return;
|
|
|
|
|
2003-09-11 04:13:46 +02:00
|
|
|
field->fld_array_info->ary_declared = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
switch (field->fld_array_info->ary_dtype) {
|
|
|
|
case dtype_short:
|
|
|
|
dtype = "short";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_long:
|
|
|
|
dtype = DCL_LONG;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_cstring:
|
|
|
|
case dtype_text:
|
|
|
|
case dtype_varying:
|
|
|
|
dtype = "char ";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_quad:
|
|
|
|
dtype = DCL_QUAD;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// ** Begin date/time/timestamp *
|
|
|
|
case dtype_sql_date:
|
|
|
|
dtype = "ISC_DATE";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_sql_time:
|
|
|
|
dtype = "ISC_TIME";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_timestamp:
|
|
|
|
dtype = "ISC_TIMESTAMP";
|
|
|
|
break;
|
|
|
|
// ** End date/time/timestamp *
|
|
|
|
|
|
|
|
case dtype_int64:
|
|
|
|
dtype = "ISC_INT64";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_float:
|
|
|
|
dtype = "float ";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_double:
|
|
|
|
dtype = "double";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
sprintf(s, "datatype %d unknown for field %s",
|
|
|
|
field->fld_array_info->ary_dtype, name);
|
|
|
|
IBERROR(s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, "static %s isc_%d", dtype,
|
|
|
|
field->fld_array_info->ary_ident);
|
|
|
|
|
|
|
|
// Print out the dimension part of the declaration
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (DIM dimension = field->fld_array_info->ary_dimension; dimension;
|
2001-05-23 15:26:42 +02:00
|
|
|
dimension = dimension->dim_next)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
2003-04-01 13:49:33 +02:00
|
|
|
ib_fprintf(out_file, " [%ld]",
|
2001-05-23 15:26:42 +02:00
|
|
|
dimension->dim_upper - dimension->dim_lower + 1);
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (field->fld_array_info->ary_dtype <= dtype_varying)
|
|
|
|
ib_fprintf(out_file, " [%d]", field->fld_array->fld_length);
|
|
|
|
|
|
|
|
// Print out the database field
|
|
|
|
|
|
|
|
ib_fprintf(out_file, ";\t/* %s */\n", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Turn a symbol into a varying string.
|
|
|
|
//
|
|
|
|
|
|
|
|
static TEXT *make_name( TEXT * string, SYM symbol)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (symbol->sym_type == SYM_delimited_cursor) {
|
|
|
|
/** All This elaborate code is just to put double quotes around
|
|
|
|
the cursor names and to escape any embeded quotes.
|
|
|
|
**/
|
|
|
|
int i = 0;
|
|
|
|
strcpy(string, "\"\\\"");
|
|
|
|
for (i = strlen(string); *symbol->sym_string; i++) {
|
|
|
|
if (*symbol->sym_string == '\"' || *symbol->sym_string == '\'') {
|
|
|
|
string[i++] = '\\';
|
|
|
|
string[i++] = *symbol->sym_string;
|
|
|
|
string[i++] = '\\';
|
|
|
|
}
|
|
|
|
string[i] = *symbol->sym_string++;
|
|
|
|
}
|
|
|
|
string[i] = 0;
|
|
|
|
strcat(string, "\\\"\"");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sprintf(string, "\"%s\"", symbol->sym_string);
|
|
|
|
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate code to test existence of compiled request with
|
|
|
|
// active transaction
|
|
|
|
//
|
|
|
|
|
2002-11-17 01:04:19 +01:00
|
|
|
static void make_ok_test( ACT action, GPRE_REQ request, int column)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
printa(column, "if (%s && %s)",
|
|
|
|
request_trans(action, request), request->req_handle);
|
|
|
|
else
|
|
|
|
printa(column, "if (%s)", request->req_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Insert a port record description in output.
|
|
|
|
//
|
|
|
|
|
|
|
|
static void make_port( POR port, int column)
|
|
|
|
{
|
|
|
|
TEXT s[80];
|
|
|
|
TEXT *dtype;
|
|
|
|
|
|
|
|
printa(column, "struct {");
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
for (REF reference = port->por_references; reference;
|
|
|
|
reference = reference->ref_next)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column + INDENT);
|
2003-09-29 14:43:14 +02:00
|
|
|
GPRE_FLD field = reference->ref_field;
|
|
|
|
TEXT* name;
|
|
|
|
SYM symbol = field->fld_symbol;
|
|
|
|
if (symbol)
|
2001-05-23 15:26:42 +02:00
|
|
|
name = symbol->sym_string;
|
|
|
|
else
|
|
|
|
name = "<expression>";
|
|
|
|
if (reference->ref_value && (reference->ref_flags & REF_array_elem))
|
|
|
|
field = field->fld_array;
|
2003-09-29 14:43:14 +02:00
|
|
|
int fld_len = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (field->fld_dtype) {
|
|
|
|
case dtype_short:
|
|
|
|
dtype = "short";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_long:
|
|
|
|
dtype = DCL_LONG;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_cstring:
|
|
|
|
case dtype_text:
|
|
|
|
dtype = "char ";
|
|
|
|
if (field->fld_sub_type != 1 || field->fld_length > 1)
|
|
|
|
fld_len = field->fld_length;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_quad:
|
|
|
|
dtype = DCL_QUAD;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// ** Begin date/time/timestamp *
|
|
|
|
case dtype_sql_date:
|
|
|
|
dtype = "ISC_DATE";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_sql_time:
|
|
|
|
dtype = "ISC_TIME";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_timestamp:
|
|
|
|
dtype = "ISC_TIMESTAMP";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_int64:
|
|
|
|
dtype = "ISC_INT64";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_blob:
|
|
|
|
dtype = "ISC_QUAD";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_float:
|
|
|
|
dtype = "float ";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case dtype_double:
|
|
|
|
dtype = "double";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
sprintf(s, "datatype %d unknown for field %s, msg %d",
|
|
|
|
field->fld_dtype, name, port->por_msg_number);
|
|
|
|
IBERROR(s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (fld_len)
|
|
|
|
ib_fprintf(out_file, " %s isc_%d [%d];\t/* %s */",
|
|
|
|
dtype, reference->ref_ident, fld_len, name);
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file, " %s isc_%d;\t/* %s */",
|
|
|
|
dtype, reference->ref_ident, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
printa(column, "} isc_%d;", port->por_ident);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the actual insertion text for a
|
|
|
|
// ready;
|
|
|
|
//
|
|
|
|
|
|
|
|
static void make_ready(
|
|
|
|
DBB db,
|
|
|
|
TEXT * filename,
|
2002-11-17 01:04:19 +01:00
|
|
|
TEXT * vector, USHORT column, GPRE_REQ request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
TEXT s1[32], s2[32];
|
|
|
|
|
|
|
|
if (request) {
|
|
|
|
sprintf(s1, "isc_%dl", request->req_ident);
|
|
|
|
|
|
|
|
if (request->req_flags & REQ_extend_dpb)
|
|
|
|
sprintf(s2, "isc_%dp", request->req_ident);
|
|
|
|
else
|
|
|
|
sprintf(s2, "isc_%d", request->req_ident);
|
|
|
|
|
|
|
|
/* if the dpb needs to be extended at runtime to include items
|
|
|
|
in host variables, do so here; this assumes that there is
|
|
|
|
always a request generated for runtime variables */
|
|
|
|
|
|
|
|
if (request->req_flags & REQ_extend_dpb) {
|
|
|
|
if (request->req_length)
|
|
|
|
printa(column, "%s = isc_%d;", s2, request->req_ident);
|
|
|
|
else
|
|
|
|
printa(column, "%s = (char*) 0;", s2);
|
|
|
|
|
|
|
|
printa(column,
|
|
|
|
"isc_expand_dpb (&%s, &%s, isc_dpb_user_name, %s, isc_dpb_password, %s, isc_dpb_sql_role_name, %s, isc_dpb_lc_messages, %s, isc_dpb_lc_ctype, %s, 0);",
|
|
|
|
s2, s1, db->dbb_r_user ? db->dbb_r_user : "(char*) 0",
|
|
|
|
db->dbb_r_password ? db->dbb_r_password : "(char*) 0",
|
|
|
|
db->dbb_r_sql_role ? db->dbb_r_sql_role : "(char*) 0",
|
|
|
|
db->dbb_r_lc_messages ? db->
|
|
|
|
dbb_r_lc_messages : "(char*) 0",
|
|
|
|
db->dbb_r_lc_ctype ? db->dbb_r_lc_ctype : "(char*) 0");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// generate the attach database itself
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* dpb_size_ptr = "0";
|
|
|
|
TEXT* dpb_ptr = "(char*) 0";
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
align(column);
|
|
|
|
if (filename)
|
|
|
|
ib_fprintf(out_file,
|
|
|
|
"isc_attach_database (%s, 0, %s, &%s, %s, %s);",
|
|
|
|
vector,
|
|
|
|
filename,
|
|
|
|
db->dbb_name->sym_string,
|
|
|
|
(request ? s1 : dpb_size_ptr), (request ? s2 : dpb_ptr));
|
|
|
|
else
|
|
|
|
ib_fprintf(out_file,
|
|
|
|
"isc_attach_database (%s, 0, \"%s\", &%s, %s, %s);",
|
|
|
|
vector,
|
|
|
|
db->dbb_filename,
|
|
|
|
db->dbb_name->sym_string,
|
|
|
|
(request ? s1 : dpb_size_ptr), (request ? s2 : dpb_ptr));
|
|
|
|
|
|
|
|
// if the dpb was extended, free it here
|
|
|
|
|
|
|
|
if (request && request->req_flags & REQ_extend_dpb) {
|
|
|
|
if (request->req_length)
|
|
|
|
printa(column, "if (%s != isc_%d)", s2, request->req_ident);
|
2001-07-12 07:46:06 +02:00
|
|
|
printa(column + (request->req_length ? INDENT : 0), "isc_free ((char*) %s);",
|
2001-05-23 15:26:42 +02:00
|
|
|
s2);
|
|
|
|
|
|
|
|
/* reset the length of the dpb */
|
|
|
|
|
|
|
|
printa(column, "%s = %d;", s1, request->req_length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Print a fixed string at a particular column.
|
|
|
|
//
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static void printa( int column, const char* string, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
va_list ptr;
|
|
|
|
|
|
|
|
VA_START(ptr, string);
|
|
|
|
align(column);
|
|
|
|
ib_vfprintf(out_file, string, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Print a fixed string at a particular column.
|
|
|
|
//
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static void printb( TEXT* string, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
va_list ptr;
|
|
|
|
|
|
|
|
VA_START(ptr, string);
|
|
|
|
ib_vfprintf(out_file, string, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the appropriate transaction handle.
|
|
|
|
//
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static TEXT* request_trans( ACT action, GPRE_REQ request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
if (action->act_type == ACT_open) {
|
|
|
|
TEXT *trname = ((OPN) action->act_object)->opn_trans;
|
|
|
|
if (!trname) {
|
|
|
|
trname = transaction_name;
|
|
|
|
}
|
|
|
|
return trname;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (request) ? request->req_trans : transaction_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate the appropriate status vector parameter for a gds
|
|
|
|
// call depending on where or not the action has an error clause.
|
|
|
|
//
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
static TEXT* status_vector( ACT action)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
if (action && (action->act_error || (action->act_flags & ACT_sql)))
|
|
|
|
return status_name;
|
|
|
|
|
|
|
|
return NULL_STATUS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//____________________________________________________________
|
|
|
|
//
|
|
|
|
// Generate substitution text for START_TRANSACTION.
|
|
|
|
// The complications include the fact that all databases
|
|
|
|
// must be readied, and that everything should stop if
|
|
|
|
// any thing fails so we don't trash the status vector.
|
|
|
|
//
|
|
|
|
|
2003-09-10 21:48:53 +02:00
|
|
|
static void t_start_auto(ACT action,
|
|
|
|
GPRE_REQ request,
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* vector,
|
2003-09-10 21:48:53 +02:00
|
|
|
int column,
|
|
|
|
bool test)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
DBB db;
|
2003-09-29 14:43:14 +02:00
|
|
|
int count;
|
|
|
|
TEXT buffer[256];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
TEXT* trname = request_trans(action, request);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// find out whether we're using a status vector or not
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int stat = !strcmp(vector, status_name);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
// this is a default transaction, make sure all databases are ready
|
|
|
|
|
2003-09-16 16:01:56 +02:00
|
|
|
begin(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (sw_auto) {
|
|
|
|
buffer[0] = 0;
|
|
|
|
for (count = 0, db = isc_databases; db; db = db->dbb_next, count++)
|
2003-09-29 14:43:14 +02:00
|
|
|
{
|
|
|
|
TEXT* filename = db->dbb_runtime;
|
|
|
|
if (filename || !(db->dbb_flags & DBB_sqlca)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column);
|
|
|
|
ib_fprintf(out_file, "if (!%s", db->dbb_name->sym_string);
|
|
|
|
if (stat && buffer[0])
|
|
|
|
ib_fprintf(out_file, " && !%s [1]", vector);
|
|
|
|
ib_fprintf(out_file, ")");
|
|
|
|
make_ready(db, filename, vector, (USHORT) (column + INDENT),
|
|
|
|
0);
|
|
|
|
if (buffer[0])
|
|
|
|
strcat(buffer, " && ");
|
|
|
|
strcat(buffer, db->dbb_name->sym_string);
|
|
|
|
}
|
2003-09-29 14:43:14 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!buffer[0])
|
|
|
|
strcpy(buffer, "1");
|
|
|
|
if (test)
|
|
|
|
printa(column, "if (%s && !%s)", buffer, trname);
|
|
|
|
else
|
|
|
|
printa(column, "if (%s)", buffer);
|
|
|
|
column += INDENT;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
for (count = 0, db = isc_databases; db; db = db->dbb_next, count++);
|
|
|
|
|
2002-12-16 16:59:18 +01:00
|
|
|
printa(column, "isc_start_transaction (%s, (FRBRD**) &%s, (short) %d",
|
2001-05-23 15:26:42 +02:00
|
|
|
vector, trname, count);
|
|
|
|
|
|
|
|
// Some systems don't like infinitely long lines. Limit them to 256.
|
|
|
|
|
2003-09-29 14:43:14 +02:00
|
|
|
int remaining = 256 - column - strlen(vector) - strlen(trname) - 31;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (db = isc_databases; db; db = db->dbb_next) {
|
2003-09-29 14:43:14 +02:00
|
|
|
int length = strlen(db->dbb_name->sym_string) + 17;
|
|
|
|
if (length > remaining) {
|
2001-05-23 15:26:42 +02:00
|
|
|
align(column + INDENT);
|
|
|
|
remaining = 256 - column - INDENT;
|
|
|
|
}
|
|
|
|
remaining -= length;
|
|
|
|
ib_fprintf(out_file, ", &%s, (short) 0, (char*) 0",
|
|
|
|
db->dbb_name->sym_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
ib_fprintf(out_file, ");");
|
|
|
|
|
|
|
|
if (sw_auto)
|
|
|
|
column -= INDENT;
|
|
|
|
|
2003-09-16 16:01:56 +02:00
|
|
|
endp(column);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|