8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-26 07:23:08 +01:00
firebird-mirror/src/jrd/log.cpp

612 lines
15 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
2003-12-31 06:36:12 +01:00
* MODULE: log.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Log all OSRI calls
*
* 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 "../jrd/common.h"
#include <stdarg.h>
#include "../jrd/jrd.h"
#include "../jrd/log.h"
#include "../jrd/ods.h"
#include "../jrd/all_proto.h"
#include "../jrd/gds_proto.h"
#include "../jrd/log_proto.h"
#include "../jrd/pag_proto.h"
#include "../jrd/thd_proto.h"
#ifdef REPLAY_OSRI_API_CALLS_SUBSYSTEM
#include "../jrd/log2.h"
2001-05-23 15:26:42 +02:00
/*
Documentation on the REPLAY_OSRI_API_CALLS_SUBSYSTEM.
This is currently removed as
a) It is currently not used for any customers
b) it doesn't work in 4.0
c) Has a slight performance impact.
d) It needs work to cover the full range of 4.0 API calls
This is a valuable function to re-activate in the product, but
until it is I've #ifdef'ed it out under REPLAY_OSRI_API_CALLS_SUBSYSTEM.
I've also embedded the notes on how it works into log.cpp so they
2001-05-23 15:26:42 +02:00
can serve as a reference to a future completion of implementation.
Not reviewed, removal & documentation only
1996-Nov-07 David Schnepper
================= dialogue history ================
From ccaro@wpo.borland.com Thu Nov 7 10:23 PST 1996
I think we should keep REPLAY in the product..
I think it will be an invaluable tool for debugging complex
SuperServer interactions. While the CLASSIC architecture is well-tested,
multi-user interactions will only become more frenetic as the multi-user
server evolves to higher levels of SMP support.
------------ and ----------
From bkarwin@interbase.borland.com Thu Nov 7 13:19 PST 1996
I do not know of any customers sites who use REPLAY. It is likely that
only Motorola knows of its existance.
This is the first I've heard of this feature, at least that I recall.
I believe the rest of support also did not know of this feature. So it
may be safe to say that none of us use it. :-)
It sounds like a good feature. In fact, a customer asked recently if
there was a server-side feature akin to Delphi's "SQL Monitor" (which
logs all BDE and InterBase API calls), and we had to say no. Even if
we had known about the REPLY feature, I think we would have said no,
since it is uncertified.
I think we should keep it in the product, and even do some development
on it, to eventually make it a supported feature that can be used
for diagnostics.
-Bill Karwin
-----------------------
Documentation on the REPLAY_OSRI_API_CALLS_SUBSYSTEM.
-----------
From deej@borland.com Thu Nov 7 09:48 PST 1996
I'll confess to having written the REPLAY feature.
I used to say that I wrote it to avoid going to Phoenix
in August, when Motorola was having a problem that we
could only reproduce by running their application.
For whatever reason it was impossible to get the
application inhouse.
But I really wrote it as a general purpose diagnostic tool
that would allow us to recreate customer problems without
sending an engineer to a customer site, and without the
necessity of figuring out a complex application.
Here's the way it works: the customer makes a copy of their
database to tape. They start REPLAY logging, then run their
application per usual. When the error occurs, they stop
logging. They send the original database and the REPLAY log
to us, and we re-run the log inhouse. It's like a tape recorder
which records all the API calls their application makes.
The feature worked beautifully in the Motorola case, and I
was able to reproduce two weeks' worth of the customer's
operations in an hour, enabling me to fix the bug within a
day when it might have taken several weeks at the customer
site.
Unfortunately, new API calls or parameters have been added
and the REPLAY feature has not been updated. At some point during
the 4.0 cycle I updated REPLAY to take care of the new calls
added to that point. But REPLAY has never been maintained
as a full-blown feature, meaning it was only tested on one platform,
and certainly never had full testing to certify that it works
for all API calls.
I think it is best if we leave this as a latent feature, which
can be dusted off if really needed in a future customer situation.
At that time we can generate a t-kit with the feature turned on,
so there is no need to maintain it as a fully supported feature.
To be sure, InterBase is a much more reliable database than it
was in 1990 and these types of situations occur less often.
And the cost of fully productizing this particular feature is
frightening.
At 04:58 PM 11/6/96 -0800, David Schnepper wrote:
>
>Does anyone still use the REPLAY facility?
>
>For those that don't know, InterBase had the ability to record all API calls
>in a log file, with the idea being to make an audit trail for debugging
>problems found in the field.
>
>There also was (apparently) the ability to replay the log to see if the
>trouble still reproduced.
>
>The reason I ask is I have found that REPLAY for Blobs (at least) doesn't
>function in 4.O - there is a private API call gds__map_blobs() which was
>used for the replay function that calls into the engine in an unsafe way.
>
>I would like to turn REPLAY off, by turning off the #ifdef in common.h - but
>would like to check on whether anyone in-house is using it - or if
>it is likely that any knowledgable customer is using it.
>
>Aside: we'll get a slight performance gain on each API call by turning
>it off as well.
>
>Dave
>
*/
2003-12-31 06:36:12 +01:00
static void error(const TEXT*);
2001-05-23 15:26:42 +02:00
static void log_char(SCHAR);
static void log_flush(void);
static void log_long(SLONG);
2003-12-31 06:36:12 +01:00
static void log_pointer(const SCHAR*);
2001-05-23 15:26:42 +02:00
static void log_short(SSHORT);
2003-12-31 06:36:12 +01:00
static void log_string(SSHORT, const SCHAR*);
static void log_teb(SSHORT, const TEB*);
static void open_log(const TEXT*, SSHORT, const SCHAR*);
2001-05-23 15:26:42 +02:00
void LOG_call(enum log_t call_type, ...)
{
/**************************************
*
* L O G _ c a l l
*
**************************************
*
* Functional description
* Log an OSRI call for later replay. This is intended
* for use as a debugging utility to ease fixing
* bugs in a production environment.
*
**************************************/
SSHORT number;
SLONG long_number;
SCHAR *pointer;
SLONG *long_pointer;
va_list ptr;
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
if (!dbb || !dbb->dbb_log || !dbb->dbb_log->log_file)
return;
log_char((SCHAR) call_type);
2001-05-23 15:26:42 +02:00
VA_START(ptr, call_type);
/* using the argument types described in the table,
pop the arguments off the call stack and put
them into the log file */
for (const char* arg_type = arg_types1[(int) call_type]; *arg_type; arg_type++)
{
2001-05-23 15:26:42 +02:00
switch (*arg_type) {
case 'o':
case 's':
number = va_arg(ptr, int);
log_short(number);
break;
case 'l':
long_number = va_arg(ptr, SLONG);
log_long(long_number);
break;
case 'p':
pointer = va_arg(ptr, SCHAR *);
log_pointer(pointer);
break;
case 'r':
pointer = va_arg(ptr, SCHAR *);
break;
case 'b':
pointer = va_arg(ptr, SCHAR *);
log_string(number, pointer);
break;
case 'f':
number = 0;
case 'n':
pointer = va_arg(ptr, SCHAR *);
if (!number)
number = strlen(pointer) + 1;
log_string(number, pointer);
break;
case 't':
pointer = va_arg(ptr, SCHAR *);
log_teb(number, pointer);
break;
case 'd':
long_pointer = va_arg(ptr, SLONG *);
log_long(long_pointer[0]);
log_long(long_pointer[1]);
break;
default:
error("argument type not known");
}
}
2001-05-23 15:26:42 +02:00
va_end(ptr);
log_flush();
}
void LOG_disable(void)
{
/**************************************
*
* L O G _ d i s a b l e
*
**************************************
*
* Functional description
* Disable logging.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
/* Get header page and look for logging entry */
PAG_delete_clump_entry(HEADER_PAGE, HDR_log_name);
/* Next, if logging was really enabled, close the file */
if (dbb->dbb_log && dbb->dbb_log->log_file)
LOG_fini();
}
void LOG_enable(const TEXT* log_file_name, USHORT l_length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L O G _ e n a b l e
*
**************************************
*
* Functional description
* Enable replay logging for the database.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
/* if we are already enabled for another file, get rid of it */
if (dbb->dbb_log)
LOG_disable();
/* Get database header page and add the log name */
PAG_add_clump(HEADER_PAGE, HDR_log_name, l_length, log_file_name,
CLUMP_ADD, 0);
/* open the log file for write to make sure that this
is the first entry in the file, and write the version number */
open_log(log_file_name, l_length, MODE_WRITE);
log_short(LOG_VERSION3);
LOG_fini();
/* now reopen for append to be a good citizen */
open_log(log_file_name, l_length, MODE_APPEND);
}
void LOG_fini(void)
{
/**************************************
*
* L O G _ f i n i
*
**************************************
*
* Functional description
* Close down the log file for replay logging.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
fblog* log;
2001-05-23 15:26:42 +02:00
if (dbb && (log = dbb->dbb_log)) {
if (log->log_file) {
log_flush();
2004-04-29 00:43:34 +02:00
fclose(log->log_file);
2001-05-23 15:26:42 +02:00
}
ALL_release(log->log_string);
ALL_release(log);
dbb->dbb_log = NULL;
}
}
void LOG_init(const TEXT* name, USHORT length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L O G _ i n i t
*
**************************************
*
* Functional description
* Initialize the REPLAY logging system.
* Check the database to see if logging
* is enabled. If so, open the file to
* append to.
*
**************************************/
USHORT log_length;
TEXT file_name[256];
/* Get header page and look for logging entry */
if (!PAG_get_clump(HEADER_PAGE, HDR_log_name, &log_length, file_name))
return;
file_name[log_length] = 0;
open_log(file_name, log_length, MODE_APPEND);
}
2003-12-31 06:36:12 +01:00
static void error(const TEXT* error_string)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* e r r o r
*
**************************************
*
* Functional description
* Report a failure of the logging system,
* both to the terminal and to the log.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
2004-04-29 00:43:34 +02:00
printf("ERROR in logging system: %s\n", error_string);
2001-05-23 15:26:42 +02:00
if (dbb->dbb_log && dbb->dbb_log->log_file) {
log_short(log_error);
2003-12-31 06:36:12 +01:00
const SSHORT length = strlen(error_string);
2001-05-23 15:26:42 +02:00
log_short(length);
}
}
static void log_char(SCHAR c)
{
/**************************************
*
* l o g _ c h a r
*
**************************************
*
* Functional description
* Log a character to the fill buffer
* for later flushing to the log.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
fblog* log = dbb->dbb_log;
2001-05-23 15:26:42 +02:00
*log->log_ptr++ = c;
/* this log flush could be done in the middle of an OSRI
call and hence could be non multiuser compatible -- it
is important that the log buffer be large enough that
this doesn't happen very often */
if ((log->log_ptr - log->log_buffer) >= (LOG_BUFFER_LENGTH - 1))
log_flush();
}
static void log_flush(void)
{
/**************************************
*
* l o g _ f l u s h
*
**************************************
*
* Functional description
* Flush the contents of the log buffer to
* the file and do a file flush while we're
* at it.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
fblog* log = dbb->dbb_log;
2001-05-23 15:26:42 +02:00
if (!(log->log_ptr - log->log_buffer))
return;
2003-12-31 06:36:12 +01:00
const UCHAR* buffer = log->log_buffer;
2004-04-29 00:43:34 +02:00
fwrite(buffer, sizeof(*buffer), log->log_ptr - log->log_buffer,
2001-05-23 15:26:42 +02:00
log->log_file);
log->log_ptr = log->log_buffer;
2004-04-29 00:43:34 +02:00
fflush(log->log_file);
2001-05-23 15:26:42 +02:00
}
static void log_long(SLONG number)
{
/**************************************
*
* l o g _ l o n g
*
**************************************
*
* Functional description
* Log a longword.
*
**************************************/
2003-12-31 06:36:12 +01:00
const SLONG vax_number = gds__vax_integer(&number, sizeof(number));
const char* p = (SCHAR *) & vax_number;
for (int i = 0; i < sizeof(number); i++)
log_char(*p++);
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
static void log_pointer(const SCHAR* pointer)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o g _ p o i n t e r
*
**************************************
*
* Functional description
* Log a pointer.
*
**************************************/
2003-12-31 06:36:12 +01:00
const char* p = (SCHAR *) & pointer;
for (int i = 0; i < sizeof(pointer); i++)
log_char(*p++);
2001-05-23 15:26:42 +02:00
}
static void log_short(SSHORT number)
{
/**************************************
*
* l o g _ s h o r t
*
**************************************
*
* Functional description
* Log a shortword.
*
**************************************/
2003-12-31 06:36:12 +01:00
const USHORT vax_number = (USHORT) gds__vax_integer(&number, sizeof(number));
const char* p = (SCHAR *) & vax_number;
for (int i = 0; i < sizeof(number); i++)
log_char(*p++);
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
static void log_string(SSHORT buffer_size, const SCHAR* buffer)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o g _ s t r i n g
*
**************************************
*
* Functional description
* Log a buffer of unknown content.
*
**************************************/
while (buffer_size--)
log_char(*buffer++);
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
static void log_teb(SSHORT count, const TEB* vector)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o g _ t e b
*
**************************************
*
* Functional description
* Log a transaction element block.
*
**************************************/
2003-12-31 06:36:12 +01:00
for (TEB* const end = vector + count; vector < end; vector++) {
2001-05-23 15:26:42 +02:00
log_pointer(*vector->teb_database);
log_long((SLONG) vector->teb_tpb_length);
log_string(vector->teb_tpb_length, vector->teb_tpb);
}
}
2003-12-31 06:36:12 +01:00
static void open_log(const TEXT* file_name, SSHORT file_length,
const SCHAR* mode)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* o p e n _ l o g
*
**************************************
*
* Functional description
* Open the log file.
*
**************************************/
2004-03-07 08:58:55 +01:00
Database* dbb = GET_DBB;
2001-05-23 15:26:42 +02:00
if (dbb->dbb_log)
LOG_fini();
2003-12-31 06:36:12 +01:00
SCHAR buffer[MAXPATHLEN];
const SCHAR* log_name;
2001-05-23 15:26:42 +02:00
if (file_length)
log_name = file_name;
2003-12-31 06:36:12 +01:00
else {
2001-05-23 15:26:42 +02:00
gds__prefix(buffer, LOG_FILE_NAME);
2003-12-31 06:36:12 +01:00
log_name = buffer;
}
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
const int mask = umask(0111);
2004-04-29 00:43:34 +02:00
void* log_file = fopen(log_name, mode);
2001-05-23 15:26:42 +02:00
umask(mask);
if (!log_file)
error("can't open log file");
else {
fblog* log = FB_NEW(*dbb->dbb_permanent) fblog();
2003-12-31 06:36:12 +01:00
dbb->dbb_log = log;
2001-05-23 15:26:42 +02:00
log->log_file = log_file;
log->log_string = FB_NEW_RPT(*dbb->dbb_permanent, LOG_BUFFER_LENGTH) str();
2001-05-23 15:26:42 +02:00
log->log_ptr = log->log_buffer = log->log_string->str_data;
}
}
#endif // REPLAY_OSRI_API_CALLS_SUBSYSTEM