8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-24 18:03:02 +01:00
firebird-mirror/src/utilities/print_event.cpp

301 lines
8.2 KiB
C++

/*
* PROGRAM: Event manager
* MODULE: print_event.cpp
* DESCRIPTION: Global region print utility
*
* 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"
#include "fb_types.h"
#include "../common/common.h"
#include "../jrd/event.h"
#include "../jrd/event_proto.h"
#include "../yvalve/gds_proto.h"
#include "../common/utils_proto.h"
#define isc_print_status gds__print_status
static void prt_que(const char*, const srq*);
static void event_list();
static void event_dump_list();
static void event_table_dump();
static EVH EVENT_header = NULL;
#define SRQ_BASE ((UCHAR*) EVENT_header)
int main(int argc, char *argv[])
{
/**************************************
*
* m a i n
*
**************************************
*
* Functional description
* This is a simple utility program to print out the current
* contents of the Firebird event table file.
*
**************************************/
ISC_STATUS_ARRAY status_vector;
if (!(EVENT_header = EVENT_init(status_vector)))
{
fprintf(stderr, "Can't access global event region\n");
isc_print_status(status_vector);
return 1;
}
if ((argc == 2) && fb_utils::strnicmp(argv[1], "-dump", MAX(strlen(argv[1]), 2)) == 0)
event_table_dump();
else if (argc == 1)
event_list();
else
fprintf(stderr, "usage: fb_event_print [-dump]\n");
return 0;
}
static void event_list()
{
/**************************************
*
* e v e n t _ l i s t
*
**************************************
*
* Functional description
* List active events in the event table.
* This format is more readable for humans.
*
**************************************/
const srq* database_que;
SRQ_LOOP(EVENT_header->evh_events, database_que)
{
const evnt* database_event = (evnt*) ((UCHAR*) database_que - OFFSET(evnt*, evnt_events));
// Skip non-database entries
if (database_event->evnt_parent)
continue;
// Print out the magic name for the database, this name
// comes from the lock key_id for the database, on Unix
// this is comprised of the device number and inode
printf("Database: ");
const UCHAR* p = (UCHAR *) database_event->evnt_name;
for (ULONG l = database_event->evnt_length; l; --l)
printf("%02x", *p++);
printf(" count: %6ld\n", database_event->evnt_count);
{ // scope
const srq* interest_que;
// Print out the interest list for this event
SRQ_LOOP(database_event->evnt_interests, interest_que)
{
const req_int* interest = (req_int*) ((UCHAR*) interest_que - OFFSET(req_int*, rint_interests));
if (!interest->rint_request)
printf("(0)");
else
{
const evt_req* request = (evt_req*) SRQ_ABS_PTR(interest->rint_request);
const prb* process = (prb*) SRQ_ABS_PTR(request->req_process);
printf("%6d ", process->prb_process_id);
}
}
} // scope
// Print out each event belonging to this database
const srq* que_inst;
SRQ_LOOP(EVENT_header->evh_events, que_inst)
{
const evnt* event = (evnt*) ((UCHAR *) que_inst - OFFSET(evnt*, evnt_events));
fb_assert(event->evnt_header.hdr_type == type_evnt);
if (event->evnt_parent != SRQ_REL_PTR(database_event))
continue;
printf(" \"%-15s\" count: %6ld Interest", event->evnt_name, event->evnt_count);
{ // scope
const srq* interest_que;
// Print out the interest list for this event
SRQ_LOOP(event->evnt_interests, interest_que)
{
const req_int* interest = (req_int*) ((UCHAR*) interest_que - OFFSET(req_int*, rint_interests));
if (!interest->rint_request)
printf("(0)");
else
{
const evt_req* request = (evt_req*) SRQ_ABS_PTR(interest->rint_request);
const prb* process = (prb*) SRQ_ABS_PTR(request->req_process);
printf("%6d ", process->prb_process_id);
}
}
} // scope
printf("\n");
}
}
}
static void event_table_dump()
{
/**************************************
*
* e v e n t _ t a b l e _ d u m p
*
**************************************
*
* Functional description
* Dump the contents of the event table.
* This format exactly matches what's in the event table and
* is not very readable by humans.
*
**************************************/
printf("%.5d GLOBAL REGION HEADER\n", 0);
printf("\tLength: %ld, version: %d, free: %ld, current: %ld, request_id: %ld\n",
EVENT_header->evh_length, EVENT_header->evh_version,
EVENT_header->evh_free, EVENT_header->evh_current_process,
EVENT_header->evh_request_id);
prt_que("\tProcesses", &EVENT_header->evh_processes);
prt_que("\tEvents", &EVENT_header->evh_events);
for (SLONG offset = sizeof(evh); offset < EVENT_header->evh_length; offset += block->hdr_length)
{
printf("\n%.5ld ", offset);
const event_hdr* block = (event_hdr*) SRQ_ABS_PTR(offset);
switch (block->hdr_type)
{
case type_prb:
{
printf("PROCESS_BLOCK (%ld)\n", block->hdr_length);
const prb* process = (prb*) block;
printf("\tFlags: %d, pid: %d\n", process->prb_flags, process->prb_process_id);
prt_que("\tProcesses", &process->prb_processes);
prt_que("\tSessions", &process->prb_sessions);
}
break;
case type_frb:
{
printf("FREE BLOCK (%ld)\n", block->hdr_length);
const frb* free = (frb*) block;
printf("\tNext: %ld\n", free->frb_next);
}
break;
case type_reqb:
{
printf("REQUEST BLOCK (%ld)\n", block->hdr_length);
const evt_req* request = (evt_req*) block;
printf("\tProcess: %ld, interests: %ld, ast: %lx, arg: %lx\n",
request->req_process, request->req_interests,
request->req_ast, request->req_ast_arg);
printf("\tRequest id: %ld\n", request->req_request_id);
prt_que("\tRequests", &request->req_requests);
}
break;
case type_evnt:
{
printf("EVENT (%ld)\n", block->hdr_length);
const evnt* event = (evnt*) block;
printf("\t\"%s\", count: %ld, parent: %ld\n",
event->evnt_name, event->evnt_count, event->evnt_parent);
prt_que("\tInterests", &event->evnt_interests);
prt_que("\tEvents", &event->evnt_events);
}
break;
case type_ses:
{
printf("SESSION (%ld)\n", block->hdr_length);
const ses* session = (ses*) block;
printf("\tInterests: %ld\n", session->ses_interests);
prt_que("\tSessions", &session->ses_sessions);
prt_que("\tRequests", &session->ses_requests);
}
break;
case type_rint:
{
printf("INTEREST (%ld)\n", block->hdr_length);
const req_int* interest = (req_int*) block;
if (interest->rint_event)
{
const evnt* event = (evnt*) SRQ_ABS_PTR(interest->rint_event);
if (event->evnt_parent)
{
const evnt* parent = (evnt*) SRQ_ABS_PTR(event->evnt_parent);
printf("\t\"%s\".\"%s\"\n", parent->evnt_name, event->evnt_name);
}
else
printf("\t\"%s\"\n", event->evnt_name);
}
printf("\tEvent: %ld, request: %ld, next: %ld, count: %ld\n",
interest->rint_event, interest->rint_request,
interest->rint_next, interest->rint_count);
prt_que("\tInterests", &interest->rint_interests);
}
break;
default:
printf("*** UNKNOWN *** (%ld)\n", block->hdr_length);
break;
}
if (!block->hdr_length)
break;
}
}
static void prt_que(const char* string, const srq* que_inst)
{
/**************************************
*
* p r t _ q u e
*
**************************************
*
* Functional description
* Print the contents of a self-relative que.
*
**************************************/
SLONG offset = SRQ_REL_PTR(que_inst);
if (offset == que_inst->srq_forward && offset == que_inst->srq_backward)
printf("%s: *empty*\n", string);
else
printf("%s: forward: %d, backward: %d\n",
string, que_inst->srq_forward, que_inst->srq_backward);
}