mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-24 18:03:02 +01:00
301 lines
8.2 KiB
C++
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);
|
|
}
|