//____________________________________________________________ // // PROGRAM: BLR Pretty Printer // MODULE: pretty.cpp // DESCRIPTION: BLR Pretty Printer // // 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): ______________________________________. // TMN (Mike Nordell) 11.APR.2001 - Reduce compiler warnings // // //____________________________________________________________ // // $Id: pretty.cpp,v 1.26 2004-05-23 23:24:42 brodsom Exp $ // #include "firebird.h" #include #include "../jrd/common.h" #include #include "../jrd/ibase.h" #include "../jrd/constants.h" #include "../gpre/prett_proto.h" #include "../jrd/gds_proto.h" #define ADVANCE_PTR(ptr) while (*ptr) ptr++; //#define PRINT_VERB if (print_verb (control, level)) return -1 #define PRINT_DYN_VERB if (print_dyn_verb (control, level)) return -1 #define PRINT_SDL_VERB if (print_sdl_verb (control, level)) return -1 #define BLR_BYTE *(control->ctl_blr)++ #define PUT_BYTE(byte) *(control->ctl_ptr)++ = byte #define NEXT_BYTE *(control->ctl_blr) #define CHECK_BUFFER if (control->ctl_ptr > control->ctl_buffer + sizeof (control->ctl_buffer) - 20) print_line(control, (SSHORT)offset) typedef struct ctl { UCHAR *ctl_blr; // Running blr string UCHAR *ctl_blr_start; // Original start of blr string FPTR_PRINT_CALLBACK ctl_routine; // Call back void *ctl_user_arg; // User argument TEXT *ctl_ptr; SSHORT ctl_language; SSHORT ctl_level; TEXT ctl_buffer[PRETTY_BUFFER_SIZE]; } *CTL; static int blr_format(CTL, const char *, ...); static int error(CTL, SSHORT, TEXT *, int); static int indent(CTL, SSHORT); static int print_blr_dtype(CTL, bool); static void print_blr_line(void*, SSHORT, const char*); static int print_byte(CTL, SSHORT); static int print_char(CTL, SSHORT); static int print_dyn_verb(CTL, SSHORT); static int print_line(CTL, SSHORT); static SLONG print_long(CTL, SSHORT); static int print_sdl_verb(CTL, SSHORT); static int print_string(CTL, SSHORT); static int print_word(CTL, SSHORT); const char *dyn_table[] = { #include "../gpre/dyntable.h" NULL }; const char *cdb_table[] = { #include "../gpre/cdbtable.h" NULL }; const char *sdl_table[] = { #include "../gpre/sdltable.h" NULL }; const char *map_strings[] = { "FIELD2", "FIELD1", "MESSAGE", "TERMINATOR", "TERMINATING_FIELD", "OPAQUE", "TRANSPARENT", "TAG", "SUB_FORM", "ITEM_INDEX", "SUB_FIELD" }; //____________________________________________________________ // // Pretty print create database parameter buffer thru callback routine. // int PRETTY_print_cdb( UCHAR* blr, FPTR_PRINT_CALLBACK routine, void* user_arg, SSHORT language) { ctl ctl_buffer; ctl* control = &ctl_buffer; SCHAR temp[32]; SSHORT parameter; SSHORT level = 0; SSHORT length; SSHORT i; SSHORT offset = 0; if (!routine) { routine = gds__default_printer; user_arg = NULL; } control->ctl_routine = routine; control->ctl_user_arg = user_arg; control->ctl_blr = control->ctl_blr_start = blr; control->ctl_ptr = control->ctl_buffer; control->ctl_language = language; indent(control, level); i = BLR_BYTE; if (*control->ctl_blr) sprintf(temp, "gds__dpb_version%d, ", i); else sprintf(temp, "gds__dpb_version%d", i); blr_format(control, temp); print_line(control, offset); while (parameter = BLR_BYTE) { const char *p; if (parameter > FB_NELEM(cdb_table) || !(p = cdb_table[parameter])) { return error(control, 0, "*** cdb parameter %d is undefined ***\n", parameter); } indent(control, level); blr_format(control, p); PUT_BYTE(','); if (length = print_byte(control, offset)) { do { print_char(control, offset); } while (--length); } print_line(control, offset); } return 0; } int PRETTY_print_dyn( UCHAR* blr, //____________________________________________________________ // // Pretty print dynamic DDL thru callback routine. // FPTR_PRINT_CALLBACK routine, void* user_arg, SSHORT language) { ctl ctl_buffer; ctl* control = &ctl_buffer; SSHORT offset = 0; SSHORT version = 0; SSHORT level = 0; if (!routine) { routine = gds__default_printer; user_arg = NULL; } control->ctl_routine = routine; control->ctl_user_arg = user_arg; control->ctl_blr = control->ctl_blr_start = blr; control->ctl_ptr = control->ctl_buffer; control->ctl_language = language; version = BLR_BYTE; if (version != isc_dyn_version_1) return error(control, offset, "*** dyn version %d is not supported ***\n", version); blr_format(control, "gds__dyn_version_1, "); print_line(control, offset); level++; PRINT_DYN_VERB; if (BLR_BYTE != isc_dyn_eoc) return error(control, offset, "*** expected dyn end-of-command ***\n", 0); blr_format(control, "gds__dyn_eoc"); print_line(control, offset); return 0; } int PRETTY_print_sdl(UCHAR* blr, //____________________________________________________________ // // Pretty print slice description language. // FPTR_PRINT_CALLBACK routine, void *user_arg, SSHORT language) { ctl ctl_buffer; ctl* control = &ctl_buffer; SSHORT offset = 0; SSHORT version; SSHORT level = 0; if (!routine) { routine = gds__default_printer; user_arg = NULL; } control->ctl_routine = routine; control->ctl_user_arg = user_arg; control->ctl_blr = control->ctl_blr_start = blr; control->ctl_ptr = control->ctl_buffer; control->ctl_language = language; version = BLR_BYTE; if (version != isc_sdl_version1) return error(control, offset, "*** sdl version %d is not supported ***\n", version); blr_format(control, "gds__sdl_version1, "); print_line(control, offset); level++; while (NEXT_BYTE != isc_sdl_eoc) PRINT_SDL_VERB; offset = control->ctl_blr - control->ctl_blr_start; blr_format(control, "gds__sdl_eoc"); print_line(control, offset); return 0; } //____________________________________________________________ // // Format an utterance. // static int blr_format(CTL control, const char *string, ...) { va_list ptr; va_start(ptr, string); vsprintf(control->ctl_ptr, string, ptr); while (*control->ctl_ptr) control->ctl_ptr++; return 0; } //____________________________________________________________ // // Put out an error msg and punt. // static int error( CTL control, SSHORT offset, TEXT * string, int arg) { print_line(control, offset); sprintf(control->ctl_ptr, string, arg); fprintf(stderr, control->ctl_ptr); ADVANCE_PTR(control->ctl_ptr); print_line(control, offset); return -1; } //____________________________________________________________ // // Indent for pretty printing. // static int indent( CTL control, SSHORT level) { level *= 3; while (--level >= 0) PUT_BYTE(' '); return 0; } //____________________________________________________________ // // Print a datatype sequence and return the length of the // data described. // static int print_blr_dtype(CTL control, bool print_object) { unsigned short dtype; SCHAR *string; SSHORT length; // TMN: FIX FIX Note that offset is not initialized to anything useful // for e.g. print_word(control, (SSHORT)offset). I assume it's better to initialize it to zero // than letting it be random. // SSHORT offset = 0; dtype = BLR_BYTE; // Special case blob (261) to keep down the size of the // jump table switch (dtype) { case blr_short: string = "short"; length = 2; break; case blr_long: string = "long"; length = 4; break; case blr_quad: string = "quad"; length = 8; break; // ** Begin date/time/timestamp * case blr_sql_date: string = "sql_date"; length = sizeof(ISC_DATE); break; case blr_sql_time: string = "sql_time"; length = sizeof(ISC_TIME); break; case blr_timestamp: string = "timestamp"; length = sizeof(ISC_TIMESTAMP); break; // ** End date/time/timestamp * case blr_int64: string = "int64"; length = sizeof(ISC_INT64); break; case blr_float: string = "float"; length = 4; break; case blr_double: string = "double"; length = 8; break; case blr_d_float: string = "d_float"; length = 8; break; case blr_text: string = "text"; break; case blr_cstring: string = "cstring"; break; case blr_varying: string = "varying"; break; case blr_text2: string = "text2"; break; case blr_cstring2: string = "cstring2"; break; case blr_varying2: string = "varying2"; break; case blr_blob_id: string = "blob_id"; length = 8; break; default: error(control, 0, "*** invalid data type ***", 0); } blr_format(control, "blr_%s, ", string); if (!print_object) return length; switch (dtype) { case blr_text: length = print_word(control, offset); break; case blr_varying: length = print_word(control, offset) + 2; break; case blr_text2: print_word(control, offset); length = print_word(control, offset); break; case blr_varying2: print_word(control, offset); length = print_word(control, offset) + 2; break; case blr_short: case blr_long: case blr_int64: case blr_quad: print_byte(control, offset); break; case blr_blob_id: print_word(control, offset); break; default: if (dtype == blr_cstring) length = print_word(control, offset); if (dtype == blr_cstring2) { print_word(control, offset); length = print_word(control, offset); } break; } return length; } //____________________________________________________________ // // Print a line of pretty-printed BLR. // static void print_blr_line(void* arg, SSHORT offset, const char* line) { CTL control = reinterpret_cast(arg); bool comma = false; char c; indent(control, control->ctl_level); while (c = *line++) { PUT_BYTE(c); if (c == ',') comma = true; else if (c != ' ') comma = false; } if (!comma) PUT_BYTE(','); print_line(control, offset); } //____________________________________________________________ // // Print a byte as a numeric value and return same. // static int print_byte( CTL control, SSHORT offset) { UCHAR v; v = BLR_BYTE; sprintf(control->ctl_ptr, (control->ctl_language) ? "chr(%d), " : "%d, ", v); ADVANCE_PTR(control->ctl_ptr); return v; } //____________________________________________________________ // // Print a byte as a numeric value and return same. // static int print_char( CTL control, SSHORT offset) { UCHAR c; SSHORT printable; c = BLR_BYTE; printable = (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9' || c == '$' || c == '_'); sprintf(control->ctl_ptr, (printable) ? "'%c'," : (control->ctl_language) ? "chr(%d)," : "%d,", c); ADVANCE_PTR(control->ctl_ptr); CHECK_BUFFER; return c; } //____________________________________________________________ // // Primary recursive routine to print dynamic DDL. // static int print_dyn_verb( CTL control, SSHORT level) { SSHORT offset = control->ctl_blr - control->ctl_blr_start; const UCHAR dyn_operator = BLR_BYTE; const char* p; const int size = FB_NELEM(dyn_table); if (dyn_operator > size || dyn_operator <= 0 || !(p = dyn_table[dyn_operator])) { return error(control, offset, "*** dyn operator %d is undefined ***\n", (int) dyn_operator); } indent(control, level); blr_format(control, p); PUT_BYTE(','); PUT_BYTE(' '); ++level; int length; switch (dyn_operator) { case isc_dyn_drop_difference: case isc_dyn_begin_backup: case isc_dyn_end_backup: return 0; case isc_dyn_begin: case isc_dyn_mod_database: print_line(control, offset); while (NEXT_BYTE != isc_dyn_end) PRINT_DYN_VERB; PRINT_DYN_VERB; return 0; case isc_dyn_view_blr: case isc_dyn_fld_validation_blr: case isc_dyn_fld_computed_blr: case isc_dyn_trg_blr: case isc_dyn_fld_missing_value: case isc_dyn_prc_blr: case isc_dyn_fld_default_value: length = print_word(control, offset); print_line(control, offset); if (length) { control->ctl_level = level; gds__print_blr((const UCHAR*) control->ctl_blr, print_blr_line, control, control->ctl_language); control->ctl_blr += length; } return 0; case isc_dyn_scl_acl: case isc_dyn_log_check_point_length: case isc_dyn_log_num_of_buffers: case isc_dyn_log_buffer_size: case isc_dyn_log_group_commit_wait: case isc_dyn_idx_inactive: length = print_word(control, offset); while (length--) print_byte(control, offset); print_line(control, offset); return 0; case isc_dyn_view_source: case isc_dyn_fld_validation_source: case isc_dyn_fld_computed_source: case isc_dyn_description: case isc_dyn_prc_source: case isc_dyn_fld_default_source: length = print_word(control, offset); while (length--) print_char(control, offset); print_line(control, offset); return 0; case isc_dyn_del_exception: if (length = print_word(control, offset)) do { print_char(control, offset); } while (--length); return 0; case isc_dyn_fld_not_null: case isc_dyn_sql_object: case isc_dyn_drop_log: case isc_dyn_drop_cache: case isc_dyn_def_default_log: case isc_dyn_log_file_serial: case isc_dyn_log_file_raw: case isc_dyn_log_file_overflow: case isc_dyn_single_validation: case isc_dyn_del_default: case isc_dyn_del_validation: case isc_dyn_idx_statistic: case isc_dyn_foreign_key_delete: case isc_dyn_foreign_key_update: case isc_dyn_foreign_key_cascade: case isc_dyn_foreign_key_default: case isc_dyn_foreign_key_null: case isc_dyn_foreign_key_none: print_line(control, offset); return 0; case isc_dyn_end: print_line(control, offset); return 0; } if (length = print_word(control, offset)) do { print_char(control, offset); } while (--length); print_line(control, offset); switch (dyn_operator) { case isc_dyn_def_database: case isc_dyn_def_dimension: case isc_dyn_def_exception: case isc_dyn_def_file: case isc_dyn_def_log_file: case isc_dyn_def_cache_file: case isc_dyn_def_filter: case isc_dyn_def_function: case isc_dyn_def_function_arg: case isc_dyn_def_generator: case isc_dyn_def_global_fld: case isc_dyn_def_idx: case isc_dyn_def_local_fld: case isc_dyn_def_rel: case isc_dyn_def_procedure: case isc_dyn_def_parameter: case isc_dyn_def_security_class: case isc_dyn_def_shadow: case isc_dyn_def_sql_fld: case isc_dyn_def_trigger: case isc_dyn_def_trigger_msg: case isc_dyn_def_view: case isc_dyn_delete_database: case isc_dyn_delete_dimensions: case isc_dyn_delete_filter: case isc_dyn_delete_function: case isc_dyn_delete_global_fld: case isc_dyn_delete_idx: case isc_dyn_delete_local_fld: case isc_dyn_delete_rel: case isc_dyn_delete_procedure: case isc_dyn_delete_parameter: case isc_dyn_delete_security_class: case isc_dyn_delete_trigger: case isc_dyn_delete_trigger_msg: case isc_dyn_delete_shadow: case isc_dyn_mod_exception: case isc_dyn_mod_global_fld: case isc_dyn_mod_idx: case isc_dyn_mod_local_fld: case isc_dyn_mod_procedure: case isc_dyn_mod_rel: case isc_dyn_mod_security_class: case isc_dyn_mod_trigger: case isc_dyn_mod_trigger_msg: case isc_dyn_rel_constraint: case isc_dyn_mod_view: case isc_dyn_grant: case isc_dyn_revoke: case isc_dyn_view_relation: case isc_dyn_def_sql_role: while (NEXT_BYTE != isc_dyn_end) PRINT_DYN_VERB; PRINT_DYN_VERB; return 0; } return 0; } //____________________________________________________________ // // Invoke callback routine to print (or do something with) a line. // static int print_line( CTL control, SSHORT offset) { *control->ctl_ptr = 0; (*control->ctl_routine) (control->ctl_user_arg, offset, control->ctl_buffer); control->ctl_ptr = control->ctl_buffer; return 0; } //____________________________________________________________ // // Print a VAX word as a numeric value an return same. // static SLONG print_long( CTL control, SSHORT offset) { const UCHAR v1 = BLR_BYTE; const UCHAR v2 = BLR_BYTE; const UCHAR v3 = BLR_BYTE; const UCHAR v4 = BLR_BYTE; sprintf(control->ctl_ptr, (control-> ctl_language) ? "chr(%d),chr(%d),chr(%d),chr(%d) " : "%d,%d,%d,%d, ", v1, v2, v3, v4); ADVANCE_PTR(control->ctl_ptr); return v1 | (v2 << 8) | (v3 << 16) | (v4 << 24); } //____________________________________________________________ // // Primary recursive routine to print slice description language. // static int print_sdl_verb( CTL control, SSHORT level) { const char* p; SSHORT offset = control->ctl_blr - control->ctl_blr_start; const UCHAR sdl_operator = BLR_BYTE; if (sdl_operator > FB_NELEM(sdl_table) || sdl_operator <= 0 || !(p = sdl_table[sdl_operator])) { return error(control, offset, "*** SDL operator %d is undefined ***\n", (int) sdl_operator); } indent(control, level); blr_format(control, p); PUT_BYTE(','); PUT_BYTE(' '); ++level; int n = 0; switch (sdl_operator) { case isc_sdl_begin: print_line(control, offset); while (NEXT_BYTE != isc_sdl_end) PRINT_SDL_VERB; PRINT_SDL_VERB; return 0; case isc_sdl_struct: n = print_byte(control, offset); while (n--) { print_line(control, offset); indent(control, level + 1); offset = control->ctl_blr - control->ctl_blr_start; print_blr_dtype(control, true); } break; case isc_sdl_scalar: print_byte(control, offset); case isc_sdl_element: n = print_byte(control, offset); print_line(control, offset); while (n--) PRINT_SDL_VERB; return 0; case isc_sdl_field: case isc_sdl_relation: print_string(control, offset); break; case isc_sdl_fid: case isc_sdl_rid: case isc_sdl_short_integer: print_word(control, offset); break; case isc_sdl_variable: case isc_sdl_tiny_integer: print_byte(control, offset); break; case isc_sdl_long_integer: print_long(control, offset); break; case isc_sdl_add: case isc_sdl_subtract: case isc_sdl_multiply: case isc_sdl_divide: print_line(control, offset); PRINT_SDL_VERB; PRINT_SDL_VERB; return 0; case isc_sdl_negate: print_line(control, offset); PRINT_SDL_VERB; return 0; case isc_sdl_do3: n++; case isc_sdl_do2: n++; case isc_sdl_do1: n += 2; print_byte(control, offset); print_line(control, offset); while (n--) PRINT_SDL_VERB; return 0; } print_line(control, offset); return 0; } //____________________________________________________________ // // Print a byte-counted string. // static int print_string( CTL control, SSHORT offset) { SSHORT n; n = print_byte(control, offset); while (--n >= 0) print_char(control, offset); PUT_BYTE(' '); return 0; } //____________________________________________________________ // // Print a VAX word as a numeric value an return same. // static int print_word( CTL control, SSHORT offset) { UCHAR v1, v2; v1 = BLR_BYTE; v2 = BLR_BYTE; sprintf(control->ctl_ptr, (control->ctl_language) ? "chr(%d),chr(%d), " : "%d,%d, ", v1, v2); ADVANCE_PTR(control->ctl_ptr); return (v2 << 8) | v1; }