8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 04:43:03 +01:00
firebird-mirror/src/jrd/dbg.cpp

1296 lines
30 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
* MODULE: dbg.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Debugging routines
*
* 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): ______________________________________.
*/
#include "firebird.h"
2004-04-29 00:43:34 +02:00
#include <stdio.h>
2001-05-23 15:26:42 +02:00
#include <string.h>
#include <signal.h>
#include "../jrd/jrd.h"
#include "../jrd/lck.h"
#include "../jrd/ods.h"
#include "../jrd/cch.h"
#include "../jrd/dbg.h"
#include "../jrd/val.h"
#include "../jrd/exe.h"
#include "../jrd/req.h"
#include "../jrd/rse.h"
#include "../jrd/btr.h"
#include "../jrd/sort.h"
#include "../jrd/que.h"
#include "../jrd/cch_proto.h"
#include "../jrd/dbg_proto.h"
2001-07-12 07:46:06 +02:00
#include "../jrd/err_proto.h"
2001-05-23 15:26:42 +02:00
#ifdef SUPERSERVER
#include "../jrd/err_proto.h"
#endif
using namespace Jrd;
2001-05-23 15:26:42 +02:00
/* Given pointer a field in the block, find the block */
#define BLOCK(fld_ptr, type, fld) (type)((SCHAR*) fld_ptr - OFFSET (type, fld))
#ifndef DEBUG
int debug;
#endif
extern int *dbt_blocks[], dbt_window[], dbt_record_param[];
2001-05-23 15:26:42 +02:00
extern SLONG gds_delta_alloc, gds_max_alloc;
typedef int (*DBG_PFN_V) ();
typedef int (*DBG_PFN_I) (int);
int (*dbg_all) () = DBG_all;
int (*dbg_analyze) (int) = DBG_analyze;
int (*dbg_block) (blk *) = DBG_block;
int (*dbg_eval) (int) = DBG_eval;
int (*dbg_open) () = DBG_open;
int (*dbg_close) () = DBG_close;
2008-01-29 11:11:52 +01:00
int (*dbg_pool) (MemoryPool*) = DBG_pool;
int (*dbg_pretty) (const jrd_nod*, int) = DBG_pretty;
2001-05-23 15:26:42 +02:00
int (*dbg_window) (int *) = DBG_window;
int (*dbg_rpb) (record_param*) = DBG_rpb;
2001-05-23 15:26:42 +02:00
static int (*dbg_bdbs) () = DBG_bdbs;
int (*dbg_examine) (int *) = DBG_examine;
int (*dbg_check) (int) = DBG_check;
#ifdef LINUX
2004-04-29 00:43:34 +02:00
FILE *dbg_file = NULL;
2001-05-23 15:26:42 +02:00
#else
2004-04-29 00:43:34 +02:00
FILE *dbg_file = stdout;
2001-05-23 15:26:42 +02:00
#endif
int DBG_supervisor(int);
int *prior_frame;
static SLONG perm_pool_mem;
static SLONG req_pool_mem;
static SLONG trans_pool_mem;
static SLONG other_pool_mem;
static void go_column(int);
static void prt_dsc(DSC *, int);
static int prt_fields(SCHAR *, int *);
static int prt_que(SCHAR *, que*);
static int rsb_pretty(const RecordSource*, int);
2001-05-23 15:26:42 +02:00
/* Pick up node names */
#define NODE(type, name, keyword) "name",
static const TEXT* node_names[] = {
2001-05-23 15:26:42 +02:00
#include "../jrd/nod.h"
0
};
#undef NODE
/* rsb types */
static const TEXT* rsb_names[] =
{
2001-05-23 15:26:42 +02:00
"boolean",
"cross",
"dbkey",
"first",
"indexed",
"merge",
"multiple",
"project",
"sequential",
"sort",
"union",
"aggregate",
"ext_sequential", /* External sequential access */
"ext_indexed", /* External indexed access */
"ext_dbkey",
"navigation",
"bit_sieve",
"left_cross",
"procedure"
};
2009-01-14 12:10:48 +01:00
int DBG_all()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ a l l
*
**************************************
*
* Functional description
* Print all known blocks.
*
**************************************/
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
if (!dbg_file) {
2004-04-29 00:43:34 +02:00
dbg_file = fopen("tt:", "w");
2001-05-23 15:26:42 +02:00
}
2001-12-24 03:51:06 +01:00
// if (!dbb || !(vector = dbb->dbb_pools))
if (!dbb || !dbb->dbb_pools.getCount())
2001-12-24 03:51:06 +01:00
{
2001-05-23 15:26:42 +02:00
return TRUE;
}
2004-03-07 08:58:55 +01:00
Database::pool_vec_type::iterator itr;
Database::pool_vec_type::iterator end = dbb->dbb_pools.end();
2001-12-24 03:51:06 +01:00
for (itr = dbb->dbb_pools.begin(); itr < end; ++itr) {
DBG_pool(*itr);
2001-05-23 15:26:42 +02:00
}
return TRUE;
}
int DBG_analyze(int pool_id)
{
/**************************************
*
* D B G _ a n a l y z e
*
**************************************
*
* Functional description
* Analyze pool by block type and sub-type.
*
**************************************/
SLONG total_length = 0;
2001-05-23 15:26:42 +02:00
struct {
int sum_count;
SLONG sum_length;
} blocks[type_MAX], nodes[nod_MAX], *p, *end;
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
if (!dbb || !dbb->dbb_pools.getCount())
2001-05-23 15:26:42 +02:00
return TRUE;
2008-12-05 02:20:14 +01:00
Database::pool_vec_type* vector = &dbb->dbb_pools;
2001-05-23 15:26:42 +02:00
for (p = blocks, end = p + (int) type_MAX; p < end; p++) {
p->sum_count = 0;
p->sum_length = 0;
}
for (p = nodes, end = p + (int) nod_MAX; p < end; p++) {
p->sum_count = 0;
p->sum_length = 0;
}
Database::pool_ptr pool = (*vector)[pool_id];
/*
2004-03-07 08:58:55 +01:00
if (pool) {
SLONG length = 0;
2001-05-23 15:26:42 +02:00
for (hunk = pool->plb_hunks; hunk; hunk = hunk->hnk_next) {
const char* hunk_end = ((char*)hunk->hnk_address) + hunk->hnk_length;
for (const blk* block = (blk*) hunk->hnk_address; block != (const blk*) hunk_end;
block = (blk*) ((SCHAR *) block + length))
2004-03-07 08:58:55 +01:00
{
const SSHORT type = block->blk_type;
2001-05-23 15:26:42 +02:00
length = block->blk_length << SHIFT;
total_length += length;
if (type <= (SSHORT) type_MIN || type >= (SSHORT) type_MAX) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "***punt***\n");
2001-05-23 15:26:42 +02:00
break;
}
p = blocks + type;
p->sum_count++;
p->sum_length += length;
if (type == (SSHORT) type_nod) {
2004-01-13 10:52:19 +01:00
p = nodes + (int) ((jrd_nod*) block)->nod_type;
2001-05-23 15:26:42 +02:00
p->sum_count++;
p->sum_length += length;
}
}
}
}
int pool_type = 0;
2001-05-23 15:26:42 +02:00
if (pool->plb_pool_id == 0) {
pool_type = 1;
perm_pool_mem = total_length / 1024;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nPool %d (Permanent pool)\n",
2001-05-23 15:26:42 +02:00
pool->plb_pool_id);
}
else {
SSHORT type = 0;
for (p = blocks, end = p + (int) type_MAX; p < end;
2004-03-07 08:58:55 +01:00
p++, type++)
{
2001-05-23 15:26:42 +02:00
if (p->sum_count)
TEXT** fields = reinterpret_cast<char**>(dbt_blocks[type]);
2001-05-23 15:26:42 +02:00
if (!strcmp(*fields, "TRANSACTION") && p->sum_count) {
pool_type = 2;
trans_pool_mem += (total_length / 1024);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nPool %d (Transaction pool)\n",
2001-05-23 15:26:42 +02:00
pool->plb_pool_id);
break;
}
if (!strcmp(*fields, "REQUEST") && p->sum_count) {
pool_type = 3;
req_pool_mem += (total_length / 1024);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nPool %d (Request pool)\n",
2001-05-23 15:26:42 +02:00
pool->plb_pool_id);
break;
}
}
}
if (!pool_type) {
other_pool_mem += (total_length / 1024);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nPool %d\n", pool->plb_pool_id);
2001-05-23 15:26:42 +02:00
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"Summary by block types: (total length of pool = %ldk)\n",
total_length / 1024);
SSHORT type;
2001-05-23 15:26:42 +02:00
for (p = blocks, end = p + (int) type_MAX, type = 0; p < end; p++, type++)
if (p->sum_count) {
int i;
TEXT** fields = reinterpret_cast<char**>(dbt_blocks[type]);
TEXT name_padded[MAX_SQL_IDENTIFIER_SIZE];
for (i = 0; i < MAX_SQL_IDENTIFIER_LEN; name_padded[i++] = ' ');
2001-05-23 15:26:42 +02:00
name_padded[i] = '\0';
for (i = 0; (*fields)[i]; i++)
name_padded[i] = (*fields)[i];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t%s\t%d\t%ld\n", name_padded, p->sum_count,
2001-05-23 15:26:42 +02:00
p->sum_length);
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "Summary by node types:\n");
2001-05-23 15:26:42 +02:00
for (p = nodes, end = p + (int) nod_MAX, type = 0; p < end; p++, type++)
if (p->sum_count) {
int i;
const TEXT* node_name = node_names[type];
TEXT name_padded[MAX_SQL_IDENTIFIER_SIZE];
2001-05-23 15:26:42 +02:00
for (i = 0; i < 31; name_padded[i++] = ' ');
name_padded[i] = '\0';
for (i = 0; node_name[i]; i++)
name_padded[i] = node_name[i];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t%s\t%d\t%ld\n", name_padded, p->sum_count,
2001-05-23 15:26:42 +02:00
p->sum_length);
}
return pool_type;
*/
return 0;
2001-05-23 15:26:42 +02:00
}
2009-01-14 12:10:48 +01:00
int DBG_bdbs()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ b d b s
*
**************************************
*
* Functional description
*
**************************************/
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
BufferControl* bcb = dbb->dbb_bcb;
2001-05-23 15:26:42 +02:00
for (unsigned int i = 0; i < bcb->bcb_count; i++)
DBG_block(bcb->bcb_rpt[i].bcb_bdb);
2001-05-23 15:26:42 +02:00
return TRUE;
}
2009-01-14 12:10:48 +01:00
int DBG_precedence()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ p r e c e d e n c e
*
**************************************
*
* Functional description
*
**************************************/
QUE que_inst;
Precedence* precedence;
BufferDesc* hi_bdb;
BufferDesc* lo_bdb;
2001-05-23 15:26:42 +02:00
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
BufferControl* bcb = dbb->dbb_bcb;
2001-05-23 15:26:42 +02:00
for (unsigned int i = 0; i < bcb->bcb_count; i++) {
const BufferDesc* bdb = bcb->bcb_rpt[i].bcb_bdb;
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags || bdb->bdb_ast_flags) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "BufferDesc %d:\tpage %"SLONGFORMAT"", i, bdb->bdb_page);
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_dirty)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", dirty");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_ast_flags & BDB_blocking)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", blocking");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_writer)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", writer");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_marked)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", marked");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_must_write)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", must_write");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_faked)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", faked");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_system_dirty)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", system_dirty");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_io_error)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", io_error");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_read_pending)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", read_pending");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_free_pending)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", free_pending");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_not_valid)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", not_valid");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_db_dirty)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", db_dirty");
2001-05-23 15:26:42 +02:00
if (bdb->bdb_flags & BDB_checkpoint)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", checkpoint");
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
if (QUE_NOT_EMPTY(bdb->bdb_higher)) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\tdirect higher precedence pages:");
for (que_inst = bdb->bdb_higher.que_forward;
que_inst != &bdb->bdb_higher; que_inst = que_inst->que_forward)
{
precedence = BLOCK(que_inst, Precedence*, pre_higher);
2001-05-23 15:26:42 +02:00
hi_bdb = precedence->pre_hi;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, " %"SLONGFORMAT"", hi_bdb->bdb_page);
2001-05-23 15:26:42 +02:00
if (precedence->pre_flags & PRE_cleared)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "(cleared)");
2001-05-23 15:26:42 +02:00
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
}
if (QUE_NOT_EMPTY(bdb->bdb_lower)) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\tdirect lower precedence pages:");
for (que_inst = bdb->bdb_lower.que_forward; que_inst != &bdb->bdb_lower;
que_inst = que_inst->que_forward)
{
precedence = BLOCK(que_inst, Precedence*, pre_lower);
2001-05-23 15:26:42 +02:00
lo_bdb = precedence->pre_low;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, " %"SLONGFORMAT"", lo_bdb->bdb_page);
2001-05-23 15:26:42 +02:00
if (precedence->pre_flags & PRE_cleared)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "(cleared)");
2001-05-23 15:26:42 +02:00
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
}
}
}
return TRUE;
}
int DBG_block(BLK block)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ b l o c k
*
**************************************
*
* Functional description
* Print a formatted block
*
**************************************/
int *fields;
int i;
2001-05-23 15:26:42 +02:00
SCHAR s[10], string[100], *p;
DSC *desc;
#undef BLOCK
#define BLOCK(struct) ((struct) block)
if (!block) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "*** NULL ***\n");
2001-05-23 15:26:42 +02:00
return FALSE;
}
/*
2008-12-18 11:47:25 +01:00
if (block->blk_type <= (SCHAR) type_MIN || block->blk_type >= (SCHAR) type_MAX)
2004-03-07 08:58:55 +01:00
{
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%X\t*** BAD BLOCK (%d) ***\n", block,
2001-05-23 15:26:42 +02:00
block->blk_type);
return FALSE;
}
*/
2001-05-23 15:26:42 +02:00
if (!block->blk_length) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%X\t*** BAD BLOCK LENGTH (%d) ***\n", block,
2001-05-23 15:26:42 +02:00
block->blk_length);
return FALSE;
}
fields = dbt_blocks[block->blk_type];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n%X\t%s (%d)", block, *fields++,
2001-05-23 15:26:42 +02:00
block->blk_length);
/*
2001-05-23 15:26:42 +02:00
if (block->blk_type == (SCHAR) type_nod)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, " -- %s",
2004-01-13 10:52:19 +01:00
node_names[(int) ((jrd_nod*) block)->nod_type]);
*/
2001-05-23 15:26:42 +02:00
prt_fields(reinterpret_cast<char*>(block), fields);
2001-05-23 15:26:42 +02:00
2009-01-20 09:33:59 +01:00
switch ((enum blk_t) block->blk_type)
{
2001-05-23 15:26:42 +02:00
case type_vec:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t");
2001-05-23 15:26:42 +02:00
p = string;
*p = 0;
for (i = 0; i < ((vec<void**>*) block)->count(); i++) {
sprintf(p, "%X, ", (*((vec<void**>*) block))[i]);
2001-05-23 15:26:42 +02:00
if (strlen(string) > 60) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s\n", string);
2001-05-23 15:26:42 +02:00
strcpy(string, "\t\t");
}
p = string + strlen(string);
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s\n", string);
2001-05-23 15:26:42 +02:00
break;
case type_vcl:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t");
2001-05-23 15:26:42 +02:00
p = string;
*p = 0;
for (i = 0; i < ((vcl*) block)->count(); i++) {
sprintf(p, "%X, ", (*(vcl*) block)[i]);
2001-05-23 15:26:42 +02:00
if (strlen(string) > 60) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s\n", string);
2001-05-23 15:26:42 +02:00
strcpy(string, "\t\t");
}
p = string + strlen(string);
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s\n", string);
2001-05-23 15:26:42 +02:00
break;
case type_bcb:
prt_que("Empty", &BLOCK(BufferControl*)->bcb_empty);
for (i = 0; i < BLOCK(BufferControl*)->bcb_count; i++) {
2001-05-23 15:26:42 +02:00
sprintf(s, "mod %d", i);
prt_que(s, &BLOCK(BufferControl*)->bcb_rpt[i].bcb_page_mod);
2001-05-23 15:26:42 +02:00
}
break;
case type_bdb:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"\tUse count: %d, page: %d, flags: %x, ast flags: %x\n",
((BufferDesc*) block)->bdb_use_count, ((BufferDesc*) block)->bdb_page,
((BufferDesc*) block)->bdb_flags, ((BufferDesc*) block)->bdb_ast_flags);
#ifdef DIRTY_TREE
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"\tParent: %X, left: %X, right: %X, dirty mask: %X\n",
((BufferDesc*) block)->bdb_parent, ((BufferDesc*) block)->bdb_left,
((BufferDesc*) block)->bdb_right, ((BufferDesc*) block)->bdb_transactions);
#else
fprintf(dbg_file,
"\tdirty mask: %X\n",
((BufferDesc*) block)->bdb_transactions);
#endif
prt_que("Que", &BLOCK(BufferDesc*)->bdb_que);
prt_que("Higher", &BLOCK(BufferDesc*)->bdb_higher);
prt_que("Lower", &BLOCK(BufferDesc*)->bdb_lower);
2001-05-23 15:26:42 +02:00
break;
case type_pre:
prt_que("Higher", &BLOCK(Precedence*)->pre_higher);
prt_que("Lower", &BLOCK(Precedence*)->pre_lower);
2001-05-23 15:26:42 +02:00
break;
case type_fmt:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t");
for (i = 0, desc = BLOCK(Format*)->fmt_desc.begin();
2004-03-30 06:10:52 +02:00
i < BLOCK(Format*)->fmt_count; desc++, i++)
{
2001-05-23 15:26:42 +02:00
prt_dsc(desc, (i % 4) * 20);
if (i % 4 == 3)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n\t");
2001-05-23 15:26:42 +02:00
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
break;
2008-01-16 09:31:31 +01:00
default: /* Shut up compiler warnings */
break;
2001-05-23 15:26:42 +02:00
}
return TRUE;
}
int DBG_check(int pool_id)
{
/**************************************
*
* D B G _ c h e c k
*
**************************************
*
* Functional description
* Check pool for integrity.
*
**************************************/
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
2004-03-07 08:58:55 +01:00
int corrupt = 0;
2001-05-23 15:26:42 +02:00
if (!dbb || !dbb->dbb_pools.getCount())
2001-05-23 15:26:42 +02:00
return corrupt;
Database::pool_vec_type* vector = &dbb->dbb_pools;
Database::pool_ptr pool = (*vector)[pool_id];
/*
2004-03-07 08:58:55 +01:00
if (pool) {
for (HNK hunk = pool->plb_hunks; hunk; hunk = hunk->hnk_next) {
const char* hunk_end = ((char*)hunk->hnk_address) + hunk->hnk_length;
for (blk* block = (blk*) hunk->hnk_address; block != (const blk*) hunk_end;
block = (blk*) ((SCHAR *) block + (block->blk_length << SHIFT)))
{
2001-05-23 15:26:42 +02:00
if (block->blk_pool_id != (UCHAR) pool_id) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%X\t*** BAD POOL ID (%d) ***\n",
2001-05-23 15:26:42 +02:00
block, block->blk_pool_id);
++corrupt;
break;
}
2008-12-18 11:47:25 +01:00
if (block->blk_type <= (SCHAR) type_MIN || block->blk_type >= (SCHAR) type_MAX)
{
2008-12-18 11:47:25 +01:00
fprintf(dbg_file, "%X\t*** BAD BLOCK (%d) ***\n", block, block->blk_type);
2001-05-23 15:26:42 +02:00
++corrupt;
break;
}
if (!block->blk_length) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"%X\t*** BAD BLOCK LENGTH (%d) ***\n", block,
block->blk_length);
++corrupt;
break;
}
}
}
}
*/
2001-05-23 15:26:42 +02:00
return corrupt;
}
2009-01-14 12:10:48 +01:00
int DBG_close()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ c l o s e
*
**************************************
*
* Functional description
* Close the debugging file.
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\014\014");
fclose(dbg_file);
dbg_file = stdout;
2001-05-23 15:26:42 +02:00
return TRUE;
}
int DBG_eval(int n)
{
/**************************************
*
* D B G _ e v a l
*
**************************************
*
* Functional description
* Examine a value.
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "octal = %X, decimal = %d, hex = %x\n", n, n, n);
2001-05-23 15:26:42 +02:00
return TRUE;
}
int DBG_examine(int *n)
{
/**************************************
*
* D B G _ e x a m i n e
*
**************************************
*
* Functional description
* Examine a value.
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "octal = %X, decimal = %d, hex = %x\n", *n, *n, *n);
2001-05-23 15:26:42 +02:00
return TRUE;
}
2009-01-14 12:10:48 +01:00
int DBG_init()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ i n i t
*
**************************************
*
* Functional description
*
**************************************/
/*
sigset (2, &DBG_supervisor);
*/
return TRUE;
}
2009-01-14 12:10:48 +01:00
int DBG_open()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ o p e n
*
**************************************
*
* Functional description
* Open a debugging output file.
*
**************************************/
SCHAR filename[64];
2004-04-29 00:43:34 +02:00
printf("Debug file: ");
scanf("%s", filename);
dbg_file = fopen(filename, "w");
2001-05-23 15:26:42 +02:00
return TRUE;
}
2008-01-29 11:11:52 +01:00
int DBG_pool(MemoryPool *pool)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ p o o l
*
**************************************
*
* Functional description
* Print all known blocks.
*
**************************************/
//pool->print_memory_pool_info(dbg_file, 0, DBG_block);
pool->print_contents(dbg_file);
2001-05-23 15:26:42 +02:00
return TRUE;
}
int DBG_pretty(const jrd_nod* node, int column)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ p r e t t y
*
**************************************
*
* Functional description
* Pretty print a node tree.
*
**************************************/
int i;
2001-05-23 15:26:42 +02:00
#define NODE(struct) ((struct) node)
2001-12-24 03:51:06 +01:00
if (node && node->blk_type == (SCHAR) type_rsb)
return rsb_pretty(reinterpret_cast<const RecordSource*>(node), column);
2001-05-23 15:26:42 +02:00
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%8X\t", node);
2001-05-23 15:26:42 +02:00
for (i = 0; i < column; i++)
2004-04-29 00:43:34 +02:00
putc(' ', dbg_file);
2001-05-23 15:26:42 +02:00
if (node == NULL)
2004-04-29 00:43:34 +02:00
return fprintf(dbg_file, "*** null ***\n");
2001-05-23 15:26:42 +02:00
/*
2001-12-24 03:51:06 +01:00
if (node->blk_type != (SCHAR) type_nod)
2004-04-29 00:43:34 +02:00
return fprintf(dbg_file, "*** bad node ***\n");
*/
2001-05-23 15:26:42 +02:00
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s (%"SLONGFORMAT")", node_names[(int) node->nod_type],
2001-05-23 15:26:42 +02:00
node->nod_impure);
column += 4;
const jrd_rel* relation;
const jrd_prc* procedure;
const jrd_nod* const* ptr;
const jrd_nod* const* end;
const IndexRetrieval* retrieval;
2009-01-20 09:33:59 +01:00
switch (node->nod_type)
{
2001-05-23 15:26:42 +02:00
case nod_rse:
{
const RecordSelExpr* recse = (RecordSelExpr*) node;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
if (recse->rse_rsb)
DBG_pretty(reinterpret_cast<const jrd_nod*>(recse->rse_rsb), column);
else {
DBG_pretty(recse->rse_first, column);
DBG_pretty(recse->rse_boolean, column);
DBG_pretty(recse->rse_sorted, column);
DBG_pretty(recse->rse_projection, column);
for (ptr = recse->rse_relation, end = ptr + recse->rse_count;
ptr < end; ptr++)
{
DBG_pretty(*ptr, column);
}
}
break;
2001-05-23 15:26:42 +02:00
}
case nod_argument:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", id: %d, message: %X\n",
2001-05-23 15:26:42 +02:00
node->nod_arg[e_arg_number], node->nod_arg[e_arg_message]);
if (node->nod_arg[e_arg_flag]) {
for (i = 0; i < column + 10; i++)
2004-04-29 00:43:34 +02:00
putc(' ', dbg_file);
fprintf(dbg_file, "flag:\n");
2001-05-23 15:26:42 +02:00
DBG_pretty(node->nod_arg[e_arg_flag], column);
}
if (node->nod_arg[e_arg_indicator]) {
for (i = 0; i < column + 10; i++)
2004-04-29 00:43:34 +02:00
putc(' ', dbg_file);
fprintf(dbg_file, "indicator:\n");
2001-05-23 15:26:42 +02:00
DBG_pretty(node->nod_arg[e_arg_flag], column);
}
return TRUE;
case nod_message:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", number: %d, format: %X\n",
2001-05-23 15:26:42 +02:00
node->nod_arg[e_msg_number], node->nod_arg[e_msg_format]);
return TRUE;
case nod_field:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", stream: %d, id: %d\n",
2001-05-23 15:26:42 +02:00
node->nod_arg[e_fld_stream], node->nod_arg[e_fld_id]);
return TRUE;
case nod_index:
retrieval = (IndexRetrieval*) node->nod_arg[e_idx_retrieval];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", id: %d\n", retrieval->irb_index);
2001-05-23 15:26:42 +02:00
for (ptr = retrieval->irb_value, end =
ptr + retrieval->irb_lower_count; ptr < end; ptr++)
{
2001-05-23 15:26:42 +02:00
DBG_pretty(*ptr, column);
}
2001-05-23 15:26:42 +02:00
for (end = ptr + retrieval->irb_upper_count; ptr < end; ptr++)
DBG_pretty(*ptr, column);
return TRUE;
case nod_relation:
relation = (jrd_rel*) node->nod_arg[e_rel_relation];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", stream: %d, %s (%X)\n",
2008-12-05 02:20:14 +01:00
node->nod_arg[e_rel_stream],
relation->rel_name.c_str(), relation);
2001-05-23 15:26:42 +02:00
return TRUE;
case nod_procedure:
{
const SSHORT procedure_id = (SSHORT)(SLONG) node->nod_arg[e_prc_procedure];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", stream: %d, prc_id: %d\n",
node->nod_arg[e_prc_stream], procedure_id);
if (node->nod_arg[e_prc_inputs])
DBG_pretty(node->nod_arg[e_prc_inputs], column);
return TRUE;
}
2001-05-23 15:26:42 +02:00
case nod_exec_proc:
procedure = (jrd_prc*) node->nod_arg[e_esp_procedure];
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", name: %s (%X)\n",
procedure->prc_name.c_str(), procedure);
2001-05-23 15:26:42 +02:00
for (ptr = node->nod_arg, end = ptr + node->nod_count; ptr < end;
ptr++)
{
2001-05-23 15:26:42 +02:00
DBG_pretty(*ptr, column);
}
2001-05-23 15:26:42 +02:00
return TRUE;
case nod_union:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", stream: %d\n", node->nod_arg[e_uni_stream]);
2001-05-23 15:26:42 +02:00
DBG_pretty(node->nod_arg[e_uni_clauses], column);
return TRUE;
case nod_aggregate:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, ", stream: %d\n", node->nod_arg[e_agg_stream]);
2001-05-23 15:26:42 +02:00
DBG_pretty(node->nod_arg[e_agg_rse], column);
DBG_pretty(node->nod_arg[e_agg_group], column);
DBG_pretty(node->nod_arg[e_agg_map], column);
return TRUE;
case nod_max:
case nod_min:
case nod_average:
case nod_total:
case nod_count:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
DBG_pretty(node->nod_arg[e_stat_rse], column);
DBG_pretty(node->nod_arg[e_stat_value], column);
DBG_pretty(node->nod_arg[e_stat_default], column);
return TRUE;
default:
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
for (ptr = node->nod_arg, end = ptr + node->nod_count; ptr < end;
ptr++)
{
2001-05-23 15:26:42 +02:00
DBG_pretty(*ptr, column);
}
2001-05-23 15:26:42 +02:00
}
if (node->nod_type == nod_for && node->nod_arg[e_for_rsb]) {
rsb_pretty(reinterpret_cast<const RecordSource*>(node->nod_arg[e_for_rsb]),
2001-05-23 15:26:42 +02:00
column);
return TRUE;
}
2001-12-24 03:51:06 +01:00
return FALSE;
2001-05-23 15:26:42 +02:00
}
int DBG_supervisor(int arg)
{
/**************************************
*
* D B G _ s u p e r v i s o r
*
**************************************
*
* Functional description
*
**************************************/
prior_frame = (int *) *(&arg - 2);
debug = 0;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nEntering JRD diagnostic DBG_supervisor\n");
2001-05-23 15:26:42 +02:00
int yyparse();
yyparse();
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nLeaving JRD diagnostic DBG_supervisor\n");
2001-05-23 15:26:42 +02:00
DBG_init();
return TRUE;
}
int DBG_rpb(record_param* rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ r p b
*
**************************************
*
* Functional description
* Print a record paramter block
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n%X\tRECORD PARAMETER BLOCK", rpb);
prt_fields(reinterpret_cast<char*>(rpb), dbt_record_param);
//DBG_window(reinterpret_cast<int*>(&rpb->rpb_window));
// This is the best I can get without causing an AV for having a tdbb* being null.
if (!rpb->rpb_relation->isTemporary())
DBG_window(reinterpret_cast<int*>(&rpb->getWindow(0)));
2001-05-23 15:26:42 +02:00
return TRUE;
}
int DBG_smb(SortMap* smb, int column)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ s m b
*
**************************************
*
* Functional description
* Pretty print an SortMap (Sort Memory Block)
2001-05-23 15:26:42 +02:00
*
**************************************/
int i;
go_column(column);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "SortMap\n");
2001-05-23 15:26:42 +02:00
go_column(column);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"keys = %d, count = %d length = %d, key_length = %d\n",
smb->smb_keys, smb->smb_count, smb->smb_length,
smb->smb_key_length);
for (i = 0; i < smb->smb_keys; i++) {
go_column(column + 2);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"key [%d] dtype = %d flags = %d length = %d offset = %d, vary_offset = %d\n",
i,
smb->smb_key_desc[i].skd_dtype,
smb->smb_key_desc[i].skd_flags,
smb->smb_key_desc[i].skd_length,
smb->smb_key_desc[i].skd_offset,
smb->smb_key_desc[i].skd_vary_offset);
}
for (i = 0; i < smb->smb_count; i++) {
smb_repeat* ptr = &smb->smb_rpt[i];
2001-05-23 15:26:42 +02:00
go_column(column + 2);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "fld [%d] flag = %d stream = %d field = %d\n",
2001-05-23 15:26:42 +02:00
i, ptr->smb_flag_offset, ptr->smb_stream,
ptr->smb_field_id);
prt_dsc(&ptr->smb_desc, column + 4);
DBG_pretty(ptr->smb_node, column + 4);
}
return TRUE;
}
2009-01-14 12:10:48 +01:00
int DBG_verify()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ v e r i f y
*
**************************************
*
* Functional description
* Verify integrity of all pools.
*
**************************************/
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
if (!dbg_file)
2004-04-29 00:43:34 +02:00
dbg_file = fopen("tt:", "w");
2001-05-23 15:26:42 +02:00
if (!dbb || !dbb->dbb_pools.getCount())
2001-05-23 15:26:42 +02:00
return TRUE;
Database::pool_vec_type* vector = &dbb->dbb_pools;
for (int i = 0; i < vector->getCount(); i++)
2001-05-23 15:26:42 +02:00
DBG_check(i);
return TRUE;
}
int DBG_window(int *window)
{
/**************************************
*
* D B G _ w i n d o w
*
**************************************
*
* Functional description
* Print a window paramter block
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n%X\tWINDOW", window);
prt_fields(reinterpret_cast<char*>(window), dbt_window);
2001-05-23 15:26:42 +02:00
return TRUE;
}
2009-01-14 12:10:48 +01:00
int DBG_memory()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D B G _ m e m o r y
*
**************************************
*
* Functional description
* Print memory usage
*
**************************************/
Database* dbb = GET_DBB();
2001-05-23 15:26:42 +02:00
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "MEMORY UTILIZATION for database\n\n");
2001-05-23 15:26:42 +02:00
/* walk through all the pools in the database */
perm_pool_mem = 0;
req_pool_mem = 0;
trans_pool_mem = 0;
other_pool_mem = 0;
2004-03-07 08:58:55 +01:00
int trans_pools = 0;
int req_pools = 0;
int other_pools = 0;
if (!dbb || !dbb->dbb_pools.getCount())
return TRUE;
Database::pool_vec_type* vector = &dbb->dbb_pools;
for (int pool_id = 0; pool_id < vector->getCount(); pool_id++) {
Database::pool_ptr pool = (*vector)[pool_id];
2001-05-23 15:26:42 +02:00
if (!pool)
continue;
2004-03-07 08:58:55 +01:00
const int pool_type = DBG_analyze(pool_id);
2009-01-20 09:33:59 +01:00
switch (pool_type)
{
2001-05-23 15:26:42 +02:00
case 1:
break;
case 2:
trans_pools++;
break;
case 3:
req_pools++;
break;
default:
other_pools++;
break;
}
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nTotal memory used in the PERMANENT pool = %ldk\n",
2001-05-23 15:26:42 +02:00
perm_pool_mem);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"Total memory used in the %d TRANSACTION pools combined = %ldk\n",
trans_pools, trans_pool_mem);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"Total memory used in the %d REQUEST pools combined = %ldk\n",
req_pools, req_pool_mem);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"Total memory used in the %d OTHER pools combined = %ldk\n",
other_pools, other_pool_mem);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file,
2001-05-23 15:26:42 +02:00
"Total memory used in all the pools combined = %ldk\n",
trans_pool_mem + req_pool_mem + other_pool_mem +
perm_pool_mem);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\nMemory malloc-ed = %ldk\n", gds_max_alloc / 1024);
fprintf(dbg_file, "Memory used of the malloc-ed memory = %ldk\n",
2001-05-23 15:26:42 +02:00
gds_delta_alloc / 1024);
return TRUE;
}
static void go_column(int column)
{
/**************************************
*
* g o _ c o l u m n
*
**************************************
*
* Functional description
* Utility function to print a bunch of spaces.
*
**************************************/
while (column-- > 0)
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, " ");
2001-05-23 15:26:42 +02:00
}
static void prt_dsc(DSC * desc, int column)
{
/**************************************
*
2008-12-05 02:20:14 +01:00
* p r t _ d s c
2001-05-23 15:26:42 +02:00
*
**************************************
*
* Functional description
* Pretty print an dsc.
*
**************************************/
go_column(column);
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "(a=%d, t=%d, l=%d, scale=%d, subtype=%d)\n",
2001-05-23 15:26:42 +02:00
desc->dsc_address, desc->dsc_dtype, desc->dsc_length,
desc->dsc_scale, desc->dsc_sub_type);
}
static int prt_fields(SCHAR * block, int *fields)
{
/**************************************
*
* p r t _ f i e l d s
*
**************************************
*
* Functional description
* Print structured block.
*
**************************************/
2004-03-07 08:58:55 +01:00
int length, offset;
2001-05-23 15:26:42 +02:00
TEXT *string, *ptr, *p, s[80];
2004-03-07 08:58:55 +01:00
int column = 99;
2001-05-23 15:26:42 +02:00
2001-12-24 03:51:06 +01:00
while ( (string = (TEXT *) * fields++) ) {
2001-05-23 15:26:42 +02:00
offset = *fields++;
length = *fields++;
ptr = (SCHAR *) block + offset;
2009-01-20 09:33:59 +01:00
switch (length)
{
2001-05-23 15:26:42 +02:00
case 0:
case 1:
sprintf(s, string, *ptr);
break;
case 2:
sprintf(s, string, *(SSHORT *) ptr);
break;
case 4:
sprintf(s, string, *(SLONG *) ptr);
break;
}
for (p = s, length = 0; *p++;)
length++;
if ((column += length + 1) >= 60) {
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n\t");
2001-05-23 15:26:42 +02:00
column = length + 1;
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s ", s);
2001-05-23 15:26:42 +02:00
}
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
return TRUE;
}
static int prt_que(SCHAR * string, QUE que_inst)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ q u e
*
**************************************
*
* Functional description
* Print a formatted que_inst entry.
2001-05-23 15:26:42 +02:00
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\t%X %s forward: %X, backward: %X\n",
que_inst, string, que_inst->que_forward, que_inst->que_backward);
2001-05-23 15:26:42 +02:00
return TRUE;
}
static int rsb_pretty(const RecordSource* rsb, int column)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* r s b _ p r e t t y
*
**************************************
*
* Functional description
* Pretty print an rsb tree.
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%X\t", rsb);
2004-03-07 08:58:55 +01:00
for (int i = 0; i < column; i++)
2004-04-29 00:43:34 +02:00
putc(' ', dbg_file);
2001-05-23 15:26:42 +02:00
if (rsb == NULL)
2004-04-29 00:43:34 +02:00
return fprintf(dbg_file, "*** null ***\n");
2001-05-23 15:26:42 +02:00
//if (rsb->blk_type != (SCHAR) type_rsb)
// return fprintf(dbg_file, "*** bad rsb ***\n");
2001-05-23 15:26:42 +02:00
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s (%d), stream: %d",
2001-05-23 15:26:42 +02:00
rsb_names[(int) rsb->rsb_type], rsb->rsb_impure,
rsb->rsb_stream);
jrd_rel* relation = rsb->rsb_relation;
if (relation) {
fprintf(dbg_file, " %s", relation->rel_name.c_str());
}
2001-05-23 15:26:42 +02:00
column += 4;
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "\n");
2001-05-23 15:26:42 +02:00
const RecordSource* const* ptr = rsb->rsb_arg;
if (rsb->rsb_type == rsb_merge) {
for (const RecordSource* const* const end = ptr + rsb->rsb_count * 2; ptr < end;
2001-05-23 15:26:42 +02:00
ptr += 2)
{
DBG_pretty(reinterpret_cast<const jrd_nod*>(*ptr), column);
}
}
else if (rsb->rsb_type != rsb_left_cross) {
for (const RecordSource* const* const end = ptr + rsb->rsb_count; ptr < end; ptr++)
{
DBG_pretty(reinterpret_cast<const jrd_nod*>(*ptr), column);
}
}
else {
for (const RecordSource* const* const end = ptr + rsb->rsb_count + 1; ptr < end;
2001-05-23 15:26:42 +02:00
ptr++)
{
DBG_pretty(reinterpret_cast<const jrd_nod*>(*ptr), column);
}
}
2001-05-23 15:26:42 +02:00
if (rsb->rsb_next)
DBG_pretty(reinterpret_cast<jrd_nod*>(rsb->rsb_next), column);
2001-05-23 15:26:42 +02:00
return TRUE;
}
void yyerror(const char* string)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* y y e r r o r
*
**************************************
*
* Functional description
* YACC error function. Boring.
*
**************************************/
2004-04-29 00:43:34 +02:00
fprintf(dbg_file, "%s\n", string);
2001-05-23 15:26:42 +02:00
}
2009-01-14 12:10:48 +01:00
int yywrap()
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* y y w r a p
*
**************************************
*
* Functional description
* Wrapup function for YACC.
*
**************************************/
return (1);
}