8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-28 02:43:03 +01:00
firebird-mirror/src/qli/gener.cpp
fsg 08f9f20489 First step to allow qli to work with relations containing bigints
It is possible now to copy table contents between databases, insert bigints via qli and show works nearly as expected.
Displaying bigints and evaluation of expressions containing bigints still doesn't work.
2004-06-03 14:00:59 +00:00

1904 lines
42 KiB
C++

/*
* PROGRAM: JRD Command Oriented Query Language
* MODULE: gener.cpp
* DESCRIPTION: BLR Generation Module
*
* The contents of this file are subject to the Interbase Public
* License Version 1.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy
* of the License at http://www.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
/*
$Id: gener.cpp,v 1.36 2004-06-03 14:00:56 fsg Exp $
*/
#include "firebird.h"
#include <stdio.h>
#include <string.h>
#include "../jrd/ibase.h"
#include "../qli/dtr.h"
#include "../jrd/align.h"
#include "../qli/exe.h"
#include "../qli/compile.h"
#include "../qli/report.h"
#include "../qli/all_proto.h"
#include "../qli/compi_proto.h"
#include "../qli/err_proto.h"
#include "../qli/gener_proto.h"
#include "../qli/meta_proto.h"
#include "../qli/mov_proto.h"
#include "../jrd/gds_proto.h"
#ifdef DEV_BUILD
static void explain(const UCHAR*);
static void explain_index_tree(SSHORT, const TEXT*, const SCHAR**, SSHORT*);
static void explain_printf(SSHORT, const TEXT*, const TEXT*);
#endif
static void gen_any(qli_nod*, qli_req*);
static void gen_assignment(qli_nod*, qli_req*);
static void gen_control_break(qli_brk*, qli_req*);
static void gen_compile(qli_req*);
static void gen_descriptor(const dsc*, qli_req*);
static void gen_erase(qli_nod*, qli_req*);
static void gen_expression(qli_nod*, qli_req*);
static void gen_field(qli_nod*, qli_req*);
static void gen_for(qli_nod*, qli_req*);
static void gen_function(qli_nod*, qli_req*);
static void gen_if(qli_nod*, qli_req*);
static void gen_literal(const dsc*, qli_req*);
static void gen_map(qli_map*, qli_req*);
static void gen_modify(qli_nod*, qli_req*);
static void gen_parameter(const qli_par*, qli_req*);
static void gen_print_list(qli_nod*, qli_req*);
static void gen_report(qli_nod*, qli_req*);
static void gen_request(qli_req*);
static void gen_rse(qli_nod*, qli_req*);
static void gen_send_receive(const qli_msg*, USHORT);
static void gen_sort(qli_nod*, qli_req*, const UCHAR);
static void gen_statement(qli_nod*, qli_req*);
static void gen_statistical(qli_nod*, qli_req*);
static void gen_store(qli_nod*, qli_req*);
#ifdef DEV_BUILD
static const SCHAR explain_info[] = { isc_info_access_path };
#endif
qli_nod* GEN_generate( qli_nod* node)
{
/**************************************
*
* G E N _ g e n e r a t e
*
**************************************
*
* Functional description
* Generate and compile BLR.
*
**************************************/
gen_statement(node, 0);
return node;
}
void GEN_release(void)
{
/**************************************
*
* G E N _ r e l e a s e
*
**************************************
*
* Functional description
* Release any compiled requests following execution or abandonment
* of a request. Just recurse around and release requests.
*
**************************************/
for (; QLI_requests; QLI_requests = QLI_requests->req_next) {
if (QLI_requests->req_handle)
{
ISC_STATUS_ARRAY status_vector;
isc_release_request(status_vector, &QLI_requests->req_handle);
}
qli_rlb* rlb = QLI_requests->req_blr;
GEN_rlb_release (rlb);
}
}
qli_rlb* GEN_rlb_extend(qli_rlb* rlb)
{
/**************************************
*
* G E N _ r l b _ e x t e n d
*
**************************************
*
* Functional description
*
* Allocate a new larger request language buffer
* and copy generated request language into it
*
**************************************/
if (!rlb)
rlb = (qli_rlb*) ALLOCD(type_rlb);
const UCHAR* old_string = rlb->rlb_base;
const ULONG l = (UCHAR *) rlb->rlb_data - (UCHAR *) rlb->rlb_base;
rlb->rlb_length += RLB_BUFFER_SIZE;
UCHAR* new_string = (UCHAR*) ALLQ_malloc((SLONG) rlb->rlb_length);
if (old_string) {
MOVQ_fast((SCHAR*) old_string, (SCHAR*) new_string, l);
ALLQ_free((SCHAR*) old_string);
}
rlb->rlb_base = new_string;
rlb->rlb_data = new_string + l;
rlb->rlb_limit = rlb->rlb_data + RLB_BUFFER_SIZE - RLB_SAFETY_MARGIN;
return (rlb);
}
void GEN_rlb_release( qli_rlb* rlb)
{
/**************************************
*
* G E N _ r l b _ r e l e a s e
*
**************************************
*
* Functional description
*
* Release the request language buffer
* contents. The buffer itself will go away
* with the default pool.
*
**************************************/
if (!rlb)
return;
if (rlb->rlb_base) {
ALLQ_free((SCHAR*) rlb->rlb_base);
rlb->rlb_base = NULL;
rlb->rlb_length = 0;
rlb->rlb_data = NULL;
rlb->rlb_limit = NULL;
}
}
#ifdef DEV_BUILD
static void explain(const UCHAR* explain_buffer)
{
/**************************************
*
* e x p l a i n
*
**************************************
*
* Functional description
* Print out the access path for a blr
* request in pretty-printed form.
*
**************************************/
SSHORT level = 0;
SCHAR relation_name[32];
// CVC: This function may have the same bugs than the internal function
// used in the engine, that received fixes in FB1 & FB1.5.
if (*explain_buffer++ != isc_info_access_path)
return;
SSHORT buffer_length = *explain_buffer++;
buffer_length += (*explain_buffer++) << 8;
while (buffer_length > 0) {
buffer_length--;
switch (*explain_buffer++) {
case isc_info_rsb_begin:
explain_printf(level, "isc_info_rsb_begin,\n", 0);
level++;
break;
case isc_info_rsb_end:
explain_printf(level, "isc_info_rsb_end,\n", 0);
level--;
break;
case isc_info_rsb_relation:
{
explain_printf(level, "isc_info_rsb_relation, ", 0);
buffer_length--;
SSHORT length = *explain_buffer++;
buffer_length -= length;
char* r = relation_name;
while (length--) {
*r++ = *explain_buffer;
putchar(*explain_buffer++);
}
printf(",\n");
*r++ = 0;
break;
}
case isc_info_rsb_type:
buffer_length--;
explain_printf(level, "isc_info_rsb_type, ", 0);
switch (*explain_buffer++) {
case isc_info_rsb_unknown:
printf("unknown type\n");
break;
case isc_info_rsb_indexed:
printf("isc_info_rsb_indexed,\n");
level++;
explain_index_tree(level, relation_name,
reinterpret_cast<const SCHAR**>(&explain_buffer),
&buffer_length);
level--;
break;
/* for join types, advance past the count byte
of substreams; we don't need it */
case isc_info_rsb_merge:
buffer_length--;
printf("isc_info_rsb_merge, %d,\n", *explain_buffer++);
break;
case isc_info_rsb_cross:
buffer_length--;
printf("isc_info_rsb_cross, %d,\n", *explain_buffer++);
break;
case isc_info_rsb_navigate:
printf("isc_info_rsb_navigate,\n");
level++;
explain_index_tree(level, relation_name,
reinterpret_cast<const SCHAR**>(&explain_buffer),
&buffer_length);
level--;
break;
case isc_info_rsb_sequential:
printf("isc_info_rsb_sequential,\n");
break;
case isc_info_rsb_sort:
printf("isc_info_rsb_sort,\n");
break;
case isc_info_rsb_first:
printf("isc_info_rsb_first,\n");
break;
case isc_info_rsb_boolean:
printf("isc_info_rsb_boolean,\n");
break;
case isc_info_rsb_union:
printf("isc_info_rsb_union,\n");
break;
case isc_info_rsb_aggregate:
printf("isc_info_rsb_aggregate,\n");
break;
case isc_info_rsb_ext_sequential:
printf("isc_info_rsb_ext_sequential,\n");
break;
case isc_info_rsb_ext_indexed:
printf("isc_info_rsb_ext_indexed,\n");
break;
case isc_info_rsb_ext_dbkey:
printf("isc_info_rsb_ext_dbkey,\n");
break;
case isc_info_rsb_left_cross:
printf("isc_info_rsb_left_cross,\n");
break;
case isc_info_rsb_select:
printf("isc_info_rsb_select,\n");
break;
case isc_info_rsb_sql_join:
printf("isc_info_rsb_sql_join,\n");
break;
case isc_info_rsb_simulate:
printf("isc_info_rsb_simulate,\n");
break;
case isc_info_rsb_sim_cross:
printf("isc_info_rsb_sim_cross,\n");
break;
case isc_info_rsb_once:
printf("isc_info_rsb_once,\n");
break;
case isc_info_rsb_procedure:
printf("isc_info_rsb_procedure,\n");
break;
}
break;
default:
explain_printf(level, "unknown info item\n", 0);
break;
}
}
}
#endif
#ifdef DEV_BUILD
static void explain_index_tree(
SSHORT level,
const TEXT* relation_name,
const SCHAR** explain_buffer_ptr,
SSHORT* buffer_length)
{
/**************************************
*
* e x p l a i n _ i n d e x _ t r e e
*
**************************************
*
* Functional description
* Print out an index tree access path.
*
**************************************/
TEXT index_name[32];
SCHAR index_info[256];
SSHORT length;
const SCHAR* explain_buffer = *explain_buffer_ptr;
(*buffer_length)--;
switch (*explain_buffer++) {
case isc_info_rsb_and:
explain_printf(level, "isc_info_rsb_and,\n", 0);
level++;
explain_index_tree(level, relation_name, &explain_buffer,
buffer_length);
explain_index_tree(level, relation_name, &explain_buffer,
buffer_length);
level--;
break;
case isc_info_rsb_or:
explain_printf(level, "isc_info_rsb_or,\n", 0);
level++;
explain_index_tree(level, relation_name, &explain_buffer,
buffer_length);
explain_index_tree(level, relation_name, &explain_buffer,
buffer_length);
level--;
break;
case isc_info_rsb_dbkey:
explain_printf(level, "isc_info_rsb_dbkey,\n", 0);
break;
case isc_info_rsb_index:
explain_printf(level, "isc_info_rsb_index, ", 0);
(*buffer_length)--;
length = (SSHORT) *explain_buffer++;
strncpy(index_name, explain_buffer, length);
index_name[length] = 0;
*buffer_length -= length;
explain_buffer += length;
MET_index_info(relation_name, index_name, index_info);
printf("%s\n", index_info);
break;
}
if (*explain_buffer == isc_info_rsb_end) {
explain_buffer++;
(*buffer_length)--;
}
*explain_buffer_ptr = explain_buffer;
}
#endif
#ifdef DEV_BUILD
static void explain_printf( SSHORT level, const TEXT* control, const TEXT* string)
{
/**************************************
*
* e x p l a i n _ p r i n t f
*
**************************************
*
* Functional description
* Do a printf with formatting.
*
**************************************/
while (level--)
printf(" ");
if (string)
printf(control, string);
else
printf(control);
}
#endif
static void gen_any( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ a n y
*
**************************************
*
* Functional description
* Generate the BLR for a statistical expresionn.
*
**************************************/
qli_rlb* rlb;
/* If there is a request associated with the statement, prepare to
generate BLR. Otherwise assume that a request is alrealdy initialized. */
qli_req* new_request = (qli_req*) node->nod_arg[e_any_request];
if (new_request) {
request = new_request;
gen_request(request);
const qli_msg* receive = (qli_msg*) node->nod_arg[e_any_send];
if (receive)
gen_send_receive(receive, blr_receive);
const qli_msg* send = (qli_msg*) node->nod_arg[e_any_receive];
gen_send_receive(send, blr_send);
rlb = CHECK_RLB(request->req_blr);
STUFF(blr_if);
}
else
rlb = CHECK_RLB(request->req_blr);
STUFF(blr_any);
gen_rse(node->nod_arg[e_any_rse], request);
USHORT value; // not feasible to change it to bool.
dsc desc;
if (new_request) {
desc.dsc_dtype = dtype_short;
desc.dsc_length = sizeof(SSHORT);
desc.dsc_scale = 0;
desc.dsc_sub_type = 0;
desc.dsc_address = (UCHAR*) &value;
STUFF(blr_assignment);
value = TRUE;
gen_literal(&desc, request);
gen_parameter(node->nod_import, request);
STUFF(blr_assignment);
value = FALSE;
gen_literal(&desc, request);
gen_parameter(node->nod_import, request);
gen_compile(request);
}
}
static void gen_assignment( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ a s s i g n m e n t
*
**************************************
*
* Functional description
* Generate an assignment statement.
*
**************************************/
qli_nod* from = node->nod_arg[e_asn_from];
// Handle a local expression locally
if (node->nod_flags & NOD_local) {
gen_expression(from, 0);
return;
}
// Generate a remote assignment
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(blr_assignment);
qli_nod* target = node->nod_arg[e_asn_to];
/* If we are referencing a parameter of another request, compile
the request first, the make a reference. Otherwise, just compile
the expression in the context of this request */
const qli_nod* reference = target->nod_arg[e_fld_reference];
if (reference) {
gen_expression(from, 0);
gen_parameter(reference->nod_import, request);
}
else
gen_expression(from, request);
qli_nod* validation = node->nod_arg[e_asn_valid];
if (validation)
gen_expression(validation, 0);
gen_expression(target, request);
}
static void gen_control_break( qli_brk* control, qli_req* request)
{
/**************************************
*
* g e n _ c o n t r o l _ b r e a k
*
**************************************
*
* Functional description
* Process report writer control breaks.
*
**************************************/
for (; control; control = control->brk_next) {
if (control->brk_field)
gen_expression((qli_nod*) control->brk_field, request);
if (control->brk_line)
gen_print_list((qli_nod*) control->brk_line, request);
}
}
static void gen_compile( qli_req* request)
{
/**************************************
*
* g e n _ c o m p i l e
*
**************************************
*
* Functional description
* Finish off BLR generation for a request, and get it compiled.
*
**************************************/
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(blr_end);
STUFF(blr_eoc);
if (QLI_blr)
gds__print_blr(rlb->rlb_base, 0, 0, 0);
const USHORT length = (UCHAR *) rlb->rlb_data - (UCHAR *) rlb->rlb_base;
DBB dbb = request->req_database;
ISC_STATUS_ARRAY status_vector;
if (isc_compile_request(status_vector, &dbb->dbb_handle,
&request->req_handle, length,
(const char*) rlb->rlb_base)) {
GEN_rlb_release (rlb);
ERRQ_database_error(dbb, status_vector);
}
#ifdef DEV_BUILD
SCHAR explain_buffer[512];
if (QLI_explain &&
!isc_request_info(status_vector, &request->req_handle, 0,
sizeof(explain_info), explain_info,
sizeof(explain_buffer), explain_buffer))
{
explain((UCHAR*) explain_buffer);
}
#endif
GEN_rlb_release (rlb);
}
static void gen_descriptor(const dsc* desc, qli_req* request)
{
/**************************************
*
* g e n _ d e s c r i p t o r
*
**************************************
*
* Functional description
* Generator a field descriptor. This is used for generating both
* message declarations and literals.
*
**************************************/
qli_rlb* rlb = CHECK_RLB(request->req_blr);
switch (desc->dsc_dtype) {
case dtype_text:
STUFF(blr_text);
STUFF_WORD(desc->dsc_length);
break;
case dtype_varying:
STUFF(blr_varying);
STUFF_WORD((desc->dsc_length - sizeof(SSHORT)));
break;
case dtype_cstring:
STUFF(blr_cstring);
STUFF_WORD(desc->dsc_length);
break;
case dtype_short:
STUFF(blr_short);
STUFF(desc->dsc_scale);
break;
case dtype_int64:
STUFF(blr_int64);
STUFF(desc->dsc_scale);
break;
case dtype_long:
STUFF(blr_long);
STUFF(desc->dsc_scale);
break;
case dtype_blob:
case dtype_quad:
STUFF(blr_quad);
STUFF(desc->dsc_scale);
break;
case dtype_real:
STUFF(blr_float);
break;
case dtype_double:
STUFF(blr_double);
break;
case dtype_timestamp:
STUFF(blr_timestamp);
break;
case dtype_sql_date:
STUFF(blr_sql_date);
break;
case dtype_sql_time:
STUFF(blr_sql_time);
break;
default:
ERRQ_bugcheck(352); // Msg352 gen_descriptor: dtype not recognized
}
}
static void gen_erase( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ e r a s e
*
**************************************
*
* Functional description
* Generate the BLR for an erase statement. Pretty simple.
*
**************************************/
const qli_msg* message = (qli_msg*) node->nod_arg[e_era_message];
if (message) {
request = (qli_req*) node->nod_arg[e_era_request];
gen_send_receive(message, blr_receive);
}
qli_rlb* rlb = CHECK_RLB(request->req_blr);
qli_ctx* context = (qli_ctx*) node->nod_arg[e_era_context];
STUFF(blr_erase);
STUFF(context->ctx_context);
}
static void gen_expression(qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ e x p r e s s i o n
*
**************************************
*
* Functional description
* Generate the BLR for a boolean or value expression.
*
**************************************/
USHORT operatr = 0;
qli_rlb* rlb = 0;
if (node->nod_flags & NOD_local)
request = NULL;
/* return;*/
else if (request)
rlb = CHECK_RLB(request->req_blr);
switch (node->nod_type) {
case nod_any:
gen_any(node, request);
return;
case nod_unique:
if (request) {
STUFF(blr_unique);
gen_rse(node->nod_arg[e_any_rse], request);
}
return;
case nod_constant:
if (request)
gen_literal(&node->nod_desc, request);
return;
case nod_field:
gen_field(node, request);
return;
case nod_format:
gen_expression(node->nod_arg[e_fmt_value], request);
return;
case nod_map:
{
qli_map* map = (qli_map*) node->nod_arg[e_map_map];
const qli_ctx* context = (qli_ctx*) node->nod_arg[e_map_context];
if (context->ctx_request != request &&
map->map_node->nod_type == nod_field)
{
gen_field(map->map_node, request);
return;
}
STUFF(blr_fid);
STUFF(context->ctx_context);
STUFF_WORD(map->map_position);
return;
}
case nod_eql:
operatr = blr_eql;
break;
case nod_neq:
operatr = blr_neq;
break;
case nod_gtr:
operatr = blr_gtr;
break;
case nod_geq:
operatr = blr_geq;
break;
case nod_leq:
operatr = blr_leq;
break;
case nod_lss:
operatr = blr_lss;
break;
case nod_containing:
operatr = blr_containing;
break;
case nod_matches:
operatr = blr_matching;
break;
case nod_sleuth:
operatr = blr_matching2;
break;
case nod_like:
operatr = (node->nod_count == 2) ? blr_like : blr_ansi_like;
break;
case nod_starts:
operatr = blr_starting;
break;
case nod_missing:
operatr = blr_missing;
break;
case nod_between:
operatr = blr_between;
break;
case nod_and:
operatr = blr_and;
break;
case nod_or:
operatr = blr_or;
break;
case nod_not:
operatr = blr_not;
break;
case nod_add:
operatr = blr_add;
break;
case nod_subtract:
operatr = blr_subtract;
break;
case nod_multiply:
operatr = blr_multiply;
break;
case nod_divide:
operatr = blr_divide;
break;
case nod_negate:
operatr = blr_negate;
break;
case nod_concatenate:
operatr = blr_concatenate;
break;
case nod_substr:
operatr = blr_substring;
break;
case nod_function:
gen_function(node, request);
return;
case nod_agg_average:
case nod_agg_count:
case nod_agg_max:
case nod_agg_min:
case nod_agg_total:
case nod_average:
case nod_count:
case nod_max:
case nod_min:
case nod_total:
case nod_from:
gen_statistical(node, request);
return;
case nod_rpt_average:
case nod_rpt_count:
case nod_rpt_max:
case nod_rpt_min:
case nod_rpt_total:
case nod_running_total:
case nod_running_count:
if (node->nod_arg[e_stt_value])
gen_expression(node->nod_arg[e_stt_value], request);
if (node->nod_export)
gen_parameter(node->nod_export, request);
request = NULL;
break;
case nod_prompt:
case nod_variable:
if (node->nod_export)
gen_parameter(node->nod_export, request);
return;
case nod_edit_blob:
case nod_reference:
return;
case nod_null:
operatr = blr_null;
break;
case nod_user_name:
operatr = blr_user_name;
break;
case nod_upcase:
operatr = blr_upcase;
break;
default:
if (request && node->nod_export) {
gen_parameter(node->nod_export, request);
return;
}
ERRQ_bugcheck(353); // Msg353 gen_expression: not understood
}
if (request) {
rlb = CHECK_RLB(request->req_blr);
STUFF(operatr);
}
qli_nod** ptr = node->nod_arg;
for (qli_nod** const end = ptr + node->nod_count; ptr < end; ptr++)
gen_expression(*ptr, request);
if (!node->nod_desc.dsc_address && node->nod_desc.dsc_length)
CMP_alloc_temp(node);
}
static void gen_field( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ f i e l d
*
**************************************
*
* Functional description
* Compile BLR to handle a field reference. Things varying
* substantially depend on whether the field reference spans
* requests or not.
*
**************************************/
// If there isn't a request specified, this is a reference.
if (!request)
return;
qli_rlb* rlb = CHECK_RLB(request->req_blr);
const qli_fld* field = (qli_fld*) node->nod_arg[e_fld_field];
const qli_ctx* context = (qli_ctx*) node->nod_arg[e_fld_context];
/* If the field referenced is in this request, just generate a field
reference. */
if (context->ctx_request == request) {
qli_nod* args = node->nod_arg[e_fld_subs];
if (args)
STUFF(blr_index);
STUFF(blr_fid);
STUFF(context->ctx_context);
STUFF_WORD(field->fld_id);
if (args) {
STUFF(args->nod_count);
qli_nod** ptr = args->nod_arg;
for (const qli_nod* const* const end = ptr + args->nod_count;
ptr < end; ++ptr)
{
gen_expression(*ptr, request);
}
}
return;
}
/* The field is in a different request. Make a parameter reference
instead. */
gen_parameter(node->nod_export, request);
}
static void gen_for( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ f o r
*
**************************************
*
* Functional description
* Generate BLR for a FOR loop, included synchronization messages.
*
**************************************/
/* If there is a request associated with the statement, prepare to
generate BLR. Otherwise assume that a request is alrealdy initialized. */
if (node->nod_arg[e_for_request]) {
request = (qli_req*) node->nod_arg[e_for_request];
gen_request(request);
}
qli_rlb* rlb = CHECK_RLB(request->req_blr);
/* If the statement requires an end of file marker, build a BEGIN/END around
the whole statement. */
const qli_msg* message = (qli_msg*) node->nod_arg[e_for_receive];
if (message)
STUFF(blr_begin);
// If there is a message to be sent, build a receive for it
if (node->nod_arg[e_for_send])
gen_send_receive((qli_msg*) node->nod_arg[e_for_send], blr_receive);
// Generate the FOR loop proper.
STUFF(blr_for);
gen_rse(node->nod_arg[e_for_rse], request);
STUFF(blr_begin);
// If data is to be received (included EOF), build a send
const qli_par* eof = 0;
dsc desc;
USHORT value;
if (message) {
gen_send_receive(message, blr_send);
STUFF(blr_begin);
// Build assigments for all values referenced.
for (const qli_par* parameter = message->msg_parameters; parameter;
parameter = parameter->par_next)
{
if (parameter->par_value) {
STUFF(blr_assignment);
gen_expression(parameter->par_value, request);
gen_parameter(parameter, request);
}
}
// Next, make a FALSE for the end of file parameter
eof = (qli_par*) node->nod_arg[e_for_eof];
desc.dsc_dtype = dtype_short;
desc.dsc_length = sizeof(SSHORT);
desc.dsc_scale = 0;
desc.dsc_sub_type = 0;
desc.dsc_address = (UCHAR*) &value;
STUFF(blr_assignment);
value = FALSE;
gen_literal(&desc, request);
gen_parameter(eof, request);
STUFF(blr_end);
}
// Build the body of the loop.
const qli_msg* continuation = request->req_continue;
if (continuation) {
STUFF(blr_label);
const USHORT label = request->req_label++;
STUFF(label);
STUFF(blr_loop);
STUFF(blr_select);
gen_send_receive(continuation, blr_receive);
STUFF(blr_leave);
STUFF(label);
}
qli_nod* sub = node->nod_arg[e_for_statement];
gen_statement(sub, request);
STUFF(blr_end);
if (continuation)
STUFF(blr_end);
// Finish off by building a SEND to indicate end of file
if (message) {
gen_send_receive(message, blr_send);
STUFF(blr_assignment);
value = TRUE;
gen_literal(&desc, request);
gen_parameter(eof, request);
STUFF(blr_end);
}
// If this is our request, compile it.
if (node->nod_arg[e_for_request])
gen_compile(request);
}
static void gen_function( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ f u n c t i o n
*
**************************************
*
* Functional description
* Generate blr for a function reference.
*
**************************************/
qli_req* new_request;
qli_rlb* rlb;
/* If there is a request associated with the statement, prepare to
generate BLR. Otherwise assume that a request is already initialized. */
if (request && (request->req_flags & (REQ_project | REQ_group_by))) {
new_request = NULL;
rlb = CHECK_RLB(request->req_blr);
}
else if (new_request = (qli_req*) node->nod_arg[e_fun_request]) {
request = new_request;
gen_request(request);
const qli_msg* receive = (qli_msg*) node->nod_arg[e_fun_send];
if (receive)
gen_send_receive(receive, blr_receive);
const qli_msg* send = (qli_msg*) node->nod_arg[e_fun_receive];
gen_send_receive(send, blr_send);
rlb = CHECK_RLB(request->req_blr);
STUFF(blr_assignment);
}
else
rlb = CHECK_RLB(request->req_blr);
// Generate function body
STUFF(blr_function);
qli_fun* function = (qli_fun*) node->nod_arg[e_fun_function];
qli_symbol* symbol = function->fun_symbol;
STUFF(symbol->sym_length);
for (const UCHAR* p = (UCHAR *) symbol->sym_string; *p;)
STUFF(*p++);
// Generate function arguments
qli_nod* args = node->nod_arg[e_fun_args];
STUFF(args->nod_count);
qli_nod** ptr = args->nod_arg;
for (const qli_nod* const* const end = ptr + args->nod_count; ptr < end; ptr++)
gen_expression(*ptr, request);
if (new_request) {
gen_parameter(node->nod_import, request);
gen_compile(request);
}
}
static void gen_if( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ i f
*
**************************************
*
* Functional description
* Generate code for an IF statement. Depending
* on the environment, this may be either a QLI
* context IF or a database context IF.
*
**************************************/
/* If the statement is local to QLI, force the sub-
statements/expressions to be local also. If not
local, generate BLR. */
if (node->nod_flags & NOD_local) {
gen_expression(node->nod_arg[e_if_boolean], 0);
gen_statement(node->nod_arg[e_if_true], request);
if (node->nod_arg[e_if_false])
gen_statement(node->nod_arg[e_if_false], request);
}
else {
// probably the var is irrelevant, but not the function call,
// as it makes sure there's enough room.
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(blr_if);
gen_expression(node->nod_arg[e_if_boolean], request);
STUFF(blr_begin);
gen_statement(node->nod_arg[e_if_true], request);
STUFF(blr_end);
if (node->nod_arg[e_if_false]) {
STUFF(blr_begin);
gen_statement(node->nod_arg[e_if_false], request);
STUFF(blr_end);
}
else
STUFF(blr_end);
}
}
static void gen_literal(const dsc* desc, qli_req* request)
{
/**************************************
*
* g e n _ l i t e r a l
*
**************************************
*
* Functional description
* Generate a literal value expression.
*
**************************************/
SLONG value;
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(blr_literal);
gen_descriptor(desc, request);
const UCHAR* p = desc->dsc_address;
USHORT l = desc->dsc_length;
switch (desc->dsc_dtype) {
case dtype_short:
value = *(SSHORT *) p;
STUFF_WORD(value);
break;
case dtype_long:
case dtype_sql_time:
case dtype_sql_date:
value = *(SLONG *) p;
STUFF_WORD(value);
STUFF_WORD(value >> 16);
break;
case dtype_int64:
case dtype_quad:
case dtype_timestamp:
value = *(SLONG *) p;
STUFF_WORD(value);
STUFF_WORD(value >> 16);
value = *(SLONG *) (p + 4);
STUFF_WORD(value);
STUFF_WORD(value >> 16);
break;
default:
if (l)
do {
STUFF(*p++);
} while (--l);
}
}
static void gen_map(qli_map* map, qli_req* request)
{
/**************************************
*
* g e n _ m a p
*
**************************************
*
* Functional description
* Generate a value map for a record selection expression.
*
**************************************/
qli_map* temp;
qli_rlb* rlb = CHECK_RLB(request->req_blr);
USHORT count = 0;
for (temp = map; temp; temp = temp->map_next)
if (temp->map_node->nod_type != nod_function)
temp->map_position = count++;
STUFF(blr_map);
STUFF_WORD(count);
for (temp = map; temp; temp = temp->map_next)
if (temp->map_node->nod_type != nod_function) {
STUFF_WORD(temp->map_position);
gen_expression(temp->map_node, request);
}
}
static void gen_modify( qli_nod* node, qli_req* org_request)
{
/**************************************
*
* g e n _ m o d i f y
*
**************************************
*
* Functional description
* Generate a MODIFY statement.
*
**************************************/
qli_req* request = (qli_req*) node->nod_arg[e_mod_request];
qli_rlb* rlb = CHECK_RLB(request->req_blr);
if (node->nod_arg[e_mod_send])
gen_send_receive((qli_msg*) node->nod_arg[e_mod_send], blr_receive);
qli_nod** ptr = &node->nod_arg[e_mod_count];
for (const qli_nod* const* const end = ptr + node->nod_count; ptr < end; ptr++)
{
qli_ctx* context = (qli_ctx*) *ptr;
STUFF(blr_modify);
STUFF(context->ctx_source->ctx_context);
STUFF(context->ctx_context);
}
STUFF(blr_begin);
gen_statement(node->nod_arg[e_mod_statement], request);
STUFF(blr_end);
}
static void gen_parameter(const qli_par* parameter, qli_req* request)
{
/**************************************
*
* g e n _ p a r a m e t e r
*
**************************************
*
* Functional description
* Generate a simple parameter reference.
*
**************************************/
qli_rlb* rlb = CHECK_RLB(request->req_blr);
const qli_msg* message = parameter->par_message;
if (!parameter->par_missing) {
STUFF(blr_parameter);
STUFF(message->msg_number);
STUFF_WORD(parameter->par_parameter);
}
else {
STUFF(blr_parameter2);
STUFF(message->msg_number);
STUFF_WORD(parameter->par_parameter);
STUFF_WORD(parameter->par_missing->par_parameter);
}
}
static void gen_print_list( qli_nod* list, qli_req* request)
{
/**************************************
*
* g e n _ p r i n t _ l i s t
*
**************************************
*
* Functional description
* Generate BLR for a print list.
*
**************************************/
qli_nod** ptr = list->nod_arg;
for (const qli_nod* const* const end = ptr + list->nod_count; ptr < end; ptr++)
{
qli_print_item* item = (qli_print_item*) *ptr;
if (item->itm_type == item_value)
gen_expression(item->itm_value, request);
}
}
static void gen_report( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ r e p o r t
*
**************************************
*
* Functional description
* Compile the body of a report specification.
*
**************************************/
qli_rpt* report = (qli_rpt*) node->nod_arg[e_prt_list];
gen_control_break(report->rpt_top_rpt, request);
gen_control_break(report->rpt_top_page, request);
gen_control_break(report->rpt_top_breaks, request);
qli_nod* list = report->rpt_detail_line;
if (list)
gen_print_list(list, request);
gen_control_break(report->rpt_bottom_breaks, request);
gen_control_break(report->rpt_bottom_page, request);
gen_control_break(report->rpt_bottom_rpt, request);
}
static void gen_request( qli_req* request)
{
/**************************************
*
* g e n _ r e q u e s t
*
**************************************
*
* Functional description
* Prepare to generation and compile a request.
*
**************************************/
qli_par* param;
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(blr_version4);
STUFF(blr_begin);
// Build declarations for all messages.
for (qli_msg* message = request->req_messages; message;
message = message->msg_next)
{
message->msg_length = 0;
for (param = message->msg_parameters; param; param = param->par_next) {
const dsc* desc = &param->par_desc;
param->par_parameter = message->msg_parameter++;
const USHORT alignment = type_alignments[desc->dsc_dtype];
if (alignment)
message->msg_length =
FB_ALIGN(message->msg_length, alignment);
param->par_offset = message->msg_length;
message->msg_length += desc->dsc_length;
qli_par* missing_param = param->par_missing;
if (missing_param) {
missing_param->par_parameter = message->msg_parameter++;
message->msg_length =
FB_ALIGN(message->msg_length, sizeof(USHORT));
desc = &missing_param->par_desc;
missing_param->par_offset = message->msg_length;
message->msg_length += desc->dsc_length;
}
}
STUFF(blr_message);
STUFF(message->msg_number);
STUFF_WORD(message->msg_parameter);
qli_str* string = (qli_str*) ALLOCDV(type_str, message->msg_length);
message->msg_buffer = (UCHAR*) string->str_data;
for (param = message->msg_parameters; param; param = param->par_next) {
gen_descriptor(&param->par_desc, request);
qli_par* missing_param = param->par_missing;
if (missing_param)
gen_descriptor(&missing_param->par_desc, request);
}
}
}
static void gen_rse( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ r s e
*
**************************************
*
* Functional description
* Generate a record selection expression.
*
**************************************/
qli_rlb* rlb = CHECK_RLB(request->req_blr);
if ((NOD_T) (IPTR) node->nod_arg[e_rse_join_type] == (NOD_T) 0)
STUFF(blr_rse);
else
STUFF(blr_rs_stream);
STUFF(node->nod_count);
// Check for aggregate case
qli_nod* list;
qli_ctx* context = (qli_ctx*) node->nod_arg[e_rse_count];
if (context->ctx_sub_rse) {
STUFF(blr_aggregate);
STUFF(context->ctx_context);
gen_rse(context->ctx_sub_rse, request);
STUFF(blr_group_by);
if (list = node->nod_arg[e_rse_group_by]) {
request->req_flags |= REQ_group_by;
STUFF(list->nod_count);
qli_nod** ptr = list->nod_arg;
for (const qli_nod* const* const end = ptr + list->nod_count;
ptr < end; ++ptr)
{
gen_expression(*ptr, request);
}
request->req_flags &= ~REQ_group_by;
}
else
STUFF(0);
gen_map(context->ctx_map, request);
if (list = node->nod_arg[e_rse_having]) {
STUFF(blr_boolean);
gen_expression(list, request);
}
if (list = node->nod_arg[e_rse_sort])
gen_sort(list, request, blr_sort);
STUFF(blr_end);
return;
}
// Make relation clauses for all relations
qli_nod** ptr = &node->nod_arg[e_rse_count];
for (const qli_nod* const* const end = ptr + node->nod_count; ptr < end; ++ptr)
{
context = (qli_ctx*) *ptr;
if (context->ctx_stream)
gen_rse(context->ctx_stream, request);
else {
const qli_rel* relation = context->ctx_relation;
STUFF(blr_rid);
STUFF_WORD(relation->rel_id);
STUFF(context->ctx_context);
}
}
// Handle various clauses
if (list = node->nod_arg[e_rse_first]) {
STUFF(blr_first);
gen_expression(list, request);
}
if (list = node->nod_arg[e_rse_boolean]) {
STUFF(blr_boolean);
gen_expression(list, request);
}
if (list = node->nod_arg[e_rse_sort])
gen_sort(list, request, blr_sort);
if (list = node->nod_arg[e_rse_reduced])
gen_sort(list, request, blr_project);
const NOD_T join_type = (NOD_T) (IPTR) node->nod_arg[e_rse_join_type];
if (join_type != (NOD_T) 0 && join_type != nod_join_inner) {
STUFF(blr_join_type);
if (join_type == nod_join_left)
STUFF(blr_left);
else if (join_type == nod_join_right)
STUFF(blr_right);
else
STUFF(blr_full);
}
STUFF(blr_end);
}
static void gen_send_receive( const qli_msg* message, USHORT operatr)
{
/**************************************
*
* g e n _ s e n d _ r e c e i v e
*
**************************************
*
* Functional description
* Generate a send or receive statement, excluding body.
*
**************************************/
qli_req* request = message->msg_request;
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(operatr);
STUFF(message->msg_number);
}
static void gen_sort( qli_nod* node, qli_req* request, const UCHAR operatr)
{
/**************************************
*
* g e n _ s o r t
*
**************************************
*
* Functional description
* Generate sort or reduced clause.
*
**************************************/
qli_rlb* rlb = CHECK_RLB(request->req_blr);
STUFF(operatr);
STUFF(node->nod_count);
request->req_flags |= REQ_project;
qli_nod** ptr = node->nod_arg;
for (qli_nod** const end = ptr + node->nod_count * 2; ptr < end;
ptr += 2)
{
if (operatr == blr_sort)
STUFF((ptr[1]) ? blr_descending : blr_ascending);
gen_expression(ptr[0], request);
}
request->req_flags &= ~REQ_project;
}
static void gen_statement( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ s t a t e m e n t
*
**************************************
*
* Functional description
* Generate BLR for statement.
*
**************************************/
if (request)
CHECK_RLB(request->req_blr);
switch (node->nod_type) {
case nod_abort:
if (node->nod_count)
gen_expression(node->nod_arg[0], 0);
return;
case nod_assign:
gen_assignment(node, request);
return;
case nod_commit_retaining:
return;
case nod_erase:
gen_erase(node, request);
return;
case nod_for:
case nod_report_loop:
gen_for(node, request);
return;
case nod_list:
{
qli_nod** ptr = node->nod_arg;
for (const qli_nod* const* const end = ptr + node->nod_count;
ptr < end; ++ptr)
{
gen_statement(*ptr, request);
}
return;
}
case nod_modify:
gen_modify(node, request);
return;
case nod_output:
gen_statement(node->nod_arg[e_out_statement], request);
return;
case nod_print:
gen_print_list(node->nod_arg[e_prt_list], request);
return;
case nod_repeat:
gen_statement(node->nod_arg[e_rpt_statement], request);
return;
case nod_report:
gen_report(node, request);
return;
case nod_store:
gen_store(node, request);
return;
case nod_if:
gen_if(node, request);
return;
default:
ERRQ_bugcheck(354); // Msg354 gen_statement: not yet implemented
}
}
static void gen_statistical( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ s t a t i s t i c a l
*
**************************************
*
* Functional description
* Generate the BLR for a statistical expresionn.
*
**************************************/
USHORT operatr;
switch (node->nod_type) {
case nod_average:
operatr = blr_average;
break;
case nod_count:
/* count2
operatr = node->nod_arg [e_stt_value] ? blr_count2 : blr_count;
*/
operatr = blr_count;
break;
case nod_max:
operatr = blr_maximum;
break;
case nod_min:
operatr = blr_minimum;
break;
case nod_total:
operatr = blr_total;
break;
case nod_agg_average:
operatr = blr_agg_average;
break;
case nod_agg_count:
/* count2
operatr = node->nod_arg [e_stt_value] ? blr_agg_count2 : blr_agg_count;
*/
operatr = blr_agg_count;
break;
case nod_agg_max:
operatr = blr_agg_max;
break;
case nod_agg_min:
operatr = blr_agg_min;
break;
case nod_agg_total:
operatr = blr_agg_total;
break;
case nod_from:
operatr = (node->nod_arg[e_stt_default]) ? blr_via : blr_from;
break;
default:
ERRQ_bugcheck(355); // Msg355 gen_statistical: not understood
}
/* If there is a request associated with the statement, prepare to
generate BLR. Otherwise assume that a request is alrealdy initialized. */
qli_rlb* rlb;
qli_req* new_request = (qli_req*) node->nod_arg[e_stt_request];
if (new_request) {
request = new_request;
gen_request(request);
const qli_msg* receive = (qli_msg*) node->nod_arg[e_stt_send];
if (receive)
gen_send_receive(receive, blr_receive);
const qli_msg* send = (qli_msg*) node->nod_arg[e_stt_receive];
gen_send_receive(send, blr_send);
rlb = CHECK_RLB(request->req_blr);
STUFF(blr_assignment);
}
else
rlb = CHECK_RLB(request->req_blr);
STUFF(operatr);
if (node->nod_arg[e_stt_rse])
gen_rse(node->nod_arg[e_stt_rse], request);
/*count 2
if (node->nod_arg [e_stt_value])
*/
if (node->nod_arg[e_stt_value] && node->nod_type != nod_agg_count)
gen_expression(node->nod_arg[e_stt_value], request);
if (node->nod_arg[e_stt_default])
gen_expression(node->nod_arg[e_stt_default], request);
if (new_request) {
gen_parameter(node->nod_import, request);
gen_compile(request);
}
}
static void gen_store( qli_nod* node, qli_req* request)
{
/**************************************
*
* g e n _ s t o r e
*
**************************************
*
* Functional description
* Generate code for STORE statement.
*
**************************************/
/* If there is a request associated with the statement, prepare to
generate BLR. Otherwise assume that a request is alrealdy initialized. */
if (node->nod_arg[e_sto_request]) {
request = (qli_req*) node->nod_arg[e_sto_request];
gen_request(request);
}
qli_rlb* rlb = CHECK_RLB(request->req_blr);
// If there is a message to be sent, build a receive for it
if (node->nod_arg[e_sto_send])
gen_send_receive((qli_msg*) node->nod_arg[e_sto_send], blr_receive);
// Generate the STORE statement proper.
STUFF(blr_store);
qli_ctx* context = (qli_ctx*) node->nod_arg[e_sto_context];
qli_rel* relation = context->ctx_relation;
STUFF(blr_rid);
STUFF_WORD(relation->rel_id);
STUFF(context->ctx_context);
// Build the body of the loop.
STUFF(blr_begin);
gen_statement(node->nod_arg[e_sto_statement], request);
STUFF(blr_end);
// If this is our request, compile it.
if (node->nod_arg[e_sto_request])
gen_compile(request);
}