8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-31 12:43:02 +01:00
firebird-mirror/src/lock/print.cpp

1495 lines
41 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Lock Manager
* MODULE: print.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Lock Table 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): ______________________________________.
*
* 2002.10.27 Sean Leyne - Completed removal of obsolete "DELTA" port
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "Ultrix" port
*
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "SGI" port
*
2002-10-30 07:40:58 +01:00
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
2008-12-05 02:20:14 +01:00
*
* 2008.04.04 Roman Simakov - Added html output support
2002-10-30 07:40:58 +01:00
*
2001-05-23 15:26:42 +02:00
*/
#include "firebird.h"
2004-04-29 00:36:29 +02:00
#include <stdio.h>
2001-05-23 15:26:42 +02:00
#include <stdlib.h>
#include <string.h>
#include "../jrd/common.h"
#include "../jrd/file_params.h"
#include "../jrd/jrd.h"
#include "../jrd/lck.h"
#include "../jrd/isc.h"
#include "../jrd/gdsassert.h"
#include "../jrd/db_alias.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/gds_proto.h"
#include "../jrd/isc_proto.h"
#include "../jrd/isc_s_proto.h"
2001-07-12 07:46:06 +02:00
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
2001-05-23 15:26:42 +02:00
#include <sys/stat.h>
#ifdef HAVE_SYS_PARAM_H
# include <sys/param.h>
2001-05-23 15:26:42 +02:00
#endif
#ifdef HAVE_IO_H
#include <io.h>
2009-01-28 15:13:56 +01:00
#endif
#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifndef O_BINARY
#define O_BINARY 0
#endif
2001-05-23 15:26:42 +02:00
#ifndef FPRINTF
2004-04-29 00:36:29 +02:00
#define FPRINTF fprintf
2001-05-23 15:26:42 +02:00
#endif
2004-04-29 00:36:29 +02:00
typedef FILE* OUTFILE;
2001-05-23 15:26:42 +02:00
2004-05-17 17:14:10 +02:00
const USHORT SW_I_ACQUIRE = 1;
const USHORT SW_I_OPERATION = 2;
const USHORT SW_I_TYPE = 4;
const USHORT SW_I_WAIT = 8;
2001-05-23 15:26:42 +02:00
2004-06-09 20:57:07 +02:00
#define SRQ_BASE ((UCHAR*) LOCK_header)
struct waitque
{
2001-05-23 15:26:42 +02:00
USHORT waitque_depth;
SRQ_PTR waitque_entry[30];
2001-05-23 15:26:42 +02:00
};
2003-09-10 13:43:31 +02:00
static void prt_lock_activity(OUTFILE, const lhb*, USHORT, USHORT, USHORT);
static void prt_lock_init(void*, sh_mem*, bool);
static void prt_history(OUTFILE, const lhb*, SRQ_PTR, const SCHAR*);
2008-01-16 08:40:12 +01:00
static void prt_lock(OUTFILE, const lhb*, lbl*, USHORT);
static void prt_owner(OUTFILE, const lhb*, const own*, bool, bool);
static void prt_owner_wait_cycle(OUTFILE, const lhb*, const own*, USHORT, waitque*);
static void prt_request(OUTFILE, const lhb*, const lrq*);
static void prt_que(OUTFILE, const lhb*, const SCHAR*, const srq*, USHORT, const TEXT* prefix = NULL);
static void prt_que2(OUTFILE, const lhb*, const SCHAR*, const srq*, USHORT, const TEXT* prefix = NULL);
2001-05-23 15:26:42 +02:00
// HTML print functions
bool sw_html_format = false;
static void prt_html_begin(OUTFILE);
static void prt_html_end(OUTFILE);
static const TEXT preOwn[] = "own";
static const TEXT preRequest[] = "request";
static const TEXT preLock[] = "lock";
#ifdef WIN_NT
static struct mtx shmemMutex;
#define MUTEX &shmemMutex
#else
#define MUTEX &LOCK_header->lhb_mutex
#endif
class HtmlLink
{
public:
HtmlLink(const TEXT* prefix, const SLONG value)
{
if (sw_html_format && value && prefix)
2008-12-05 02:20:14 +01:00
sprintf(strBuffer, "<a href=\"#%s%"SLONGFORMAT"\">%6"SLONGFORMAT"</a>", prefix, value, value);
else
2008-12-05 02:20:14 +01:00
sprintf(strBuffer, "%6"SLONGFORMAT, value);
}
operator const TEXT*()
{
return strBuffer;
}
private:
TEXT strBuffer[256];
2009-06-27 09:00:34 +02:00
HtmlLink(const HtmlLink&) {}
};
2009-06-27 09:00:34 +02:00
static const TEXT history_names[][10] =
{
2001-05-23 15:26:42 +02:00
"n/a", "ENQ", "DEQ", "CONVERT", "SIGNAL", "POST", "WAIT",
"DEL_PROC", "DEL_LOCK", "DEL_REQ", "DENY", "GRANT", "LEAVE",
"SCAN", "DEAD", "ENTER", "BUG", "ACTIVE", "CLEANUP", "DEL_OWNER"
};
2008-12-05 02:20:14 +01:00
static const TEXT valid_switches[] =
"Valid switches are: -o, -p, -l, -r, -a, -h, -n, -s <n>, -c, -i <n> <n>, -m \n";
2008-02-02 18:06:24 +01:00
// The same table is in lock.cpp, maybe worth moving to a common file?
static const UCHAR compatibility[LCK_max][LCK_max] =
{
2001-05-23 15:26:42 +02:00
/* Shared Prot Shared Prot
none null Read Read Write Write Exclusive */
2008-02-02 18:06:24 +01:00
/* none */ {true, true, true, true, true, true, true},
/* null */ {true, true, true, true, true, true, true},
/* SR */ {true, true, true, true, true, true, false},
/* PR */ {true, true, true, true, false, false, false},
/* SW */ {true, true, true, false, true, false, false},
/* PW */ {true, true, true, false, false, false, false},
/* EX */ {true, true, false, false, false, false, false}
2001-05-23 15:26:42 +02:00
};
2008-02-02 18:06:24 +01:00
//#define COMPATIBLE(st1, st2) compatibility [st1 * LCK_max + st2]
2001-05-23 15:26:42 +02:00
int CLIB_ROUTINE main( int argc, char *argv[])
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a i n
*
**************************************
*
* Functional description
* Check switches passed in and prepare to dump the lock table
2004-04-29 00:36:29 +02:00
* to stdout.
2001-05-23 15:26:42 +02:00
*
**************************************/
2004-04-29 00:36:29 +02:00
OUTFILE outfile = stdout;
2001-05-23 15:26:42 +02:00
2009-08-14 09:25:09 +02:00
// Perform some special handling when run as a Firebird service. The
// first switch can be "-svc" (lower case!) or it can be "-svc_re" followed
// by 3 file descriptors to use in re-directing stdin, stdout, and stderr.
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (argc > 1 && !strcmp(argv[1], "-svc"))
{
2001-05-23 15:26:42 +02:00
argv++;
argc--;
}
2008-12-31 06:06:08 +01:00
else if (argc > 4 && !strcmp(argv[1], "-svc_re"))
{
long redir_in = atol(argv[2]);
long redir_out = atol(argv[3]);
long redir_err = atol(argv[4]);
2001-05-23 15:26:42 +02:00
#ifdef WIN_NT
redir_in = _open_osfhandle(redir_in, 0);
redir_out = _open_osfhandle(redir_out, 0);
redir_err = _open_osfhandle(redir_err, 0);
#endif
if (redir_in != 0)
2009-08-13 16:23:38 +02:00
{
2001-05-23 15:26:42 +02:00
if (dup2((int) redir_in, 0))
close((int) redir_in);
2009-08-13 16:23:38 +02:00
}
2001-05-23 15:26:42 +02:00
if (redir_out != 1)
2009-08-13 16:23:38 +02:00
{
2001-05-23 15:26:42 +02:00
if (dup2((int) redir_out, 1))
close((int) redir_out);
2009-08-13 16:23:38 +02:00
}
2001-05-23 15:26:42 +02:00
if (redir_err != 2)
2009-08-13 16:23:38 +02:00
{
2001-05-23 15:26:42 +02:00
if (dup2((int) redir_err, 2))
close((int) redir_err);
2009-08-13 16:23:38 +02:00
}
2001-05-23 15:26:42 +02:00
argv += 4;
argc -= 4;
}
2003-09-10 13:43:31 +02:00
//const int orig_argc = argc;
//SCHAR** orig_argv = argv;
2001-05-23 15:26:42 +02:00
2009-08-14 09:25:09 +02:00
// Handle switches, etc.
2001-05-23 15:26:42 +02:00
argv++;
2003-09-10 13:43:31 +02:00
bool sw_consistency = false;
bool sw_waitlist = false;
bool sw_requests = false;
bool sw_locks = false;
bool sw_history = false;
bool sw_owners = true;
2008-12-05 02:20:14 +01:00
2003-09-14 02:46:05 +02:00
USHORT sw_interactive;
// Those variables should be signed to accept negative values from atoi
SSHORT sw_series;
SSHORT sw_intervals;
SSHORT sw_seconds;
2001-05-23 15:26:42 +02:00
sw_series = sw_interactive = sw_intervals = sw_seconds = 0;
const TEXT* lock_file = NULL;
const TEXT* db_file = NULL;
2001-05-23 15:26:42 +02:00
2008-12-31 06:06:08 +01:00
while (--argc)
{
2003-09-10 13:43:31 +02:00
SCHAR* p = *argv++;
2009-06-27 09:00:34 +02:00
if (*p++ != '-')
{
FPRINTF(outfile, valid_switches);
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
}
2003-09-10 13:43:31 +02:00
SCHAR c;
2001-05-23 15:26:42 +02:00
while (c = *p++)
2008-12-31 06:06:08 +01:00
switch (c)
{
2001-05-23 15:26:42 +02:00
case 'o':
case 'p':
2003-09-04 23:26:15 +02:00
sw_owners = true;
2001-05-23 15:26:42 +02:00
break;
case 'c':
2003-09-04 23:26:15 +02:00
sw_consistency = true;
2001-05-23 15:26:42 +02:00
break;
case 'l':
2003-09-04 23:26:15 +02:00
sw_locks = true;
2001-05-23 15:26:42 +02:00
break;
case 'r':
2003-09-04 23:26:15 +02:00
sw_requests = true;
2001-05-23 15:26:42 +02:00
break;
case 'a':
2003-09-04 23:26:15 +02:00
sw_locks = true;
sw_owners = true;
sw_requests = true;
sw_history = true;
2001-05-23 15:26:42 +02:00
break;
case 'h':
2003-09-04 23:26:15 +02:00
sw_history = true;
2001-05-23 15:26:42 +02:00
break;
case 's':
if (argc > 1)
sw_series = atoi(*argv++);
2009-06-27 09:00:34 +02:00
if (sw_series <= 0)
{
FPRINTF(outfile, "Please specify a positive value following option -s\n");
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
}
--argc;
break;
case 'i':
while (c = *p++)
2008-12-31 06:06:08 +01:00
switch (c)
{
2001-05-23 15:26:42 +02:00
case 'a':
sw_interactive |= SW_I_ACQUIRE;
break;
case 'o':
sw_interactive |= SW_I_OPERATION;
break;
case 't':
sw_interactive |= SW_I_TYPE;
break;
case 'w':
sw_interactive |= SW_I_WAIT;
break;
default:
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "Valid interactive switches are: a, o, t, w\n");
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
break;
}
if (!sw_interactive)
2008-12-31 06:06:08 +01:00
sw_interactive = (SW_I_ACQUIRE | SW_I_OPERATION | SW_I_TYPE | SW_I_WAIT);
2001-05-23 15:26:42 +02:00
sw_seconds = sw_intervals = 1;
2009-06-27 09:00:34 +02:00
if (argc > 1)
{
2001-05-23 15:26:42 +02:00
sw_seconds = atoi(*argv++);
--argc;
2009-06-27 09:00:34 +02:00
if (argc > 1)
{
2001-05-23 15:26:42 +02:00
sw_intervals = atoi(*argv++);
--argc;
}
2009-06-27 09:00:34 +02:00
if (!(sw_seconds > 0) || (sw_intervals < 0))
{
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "Please specify 2 positive values for option -i\n");
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
}
}
--p;
break;
case 'w':
2003-09-04 23:26:15 +02:00
sw_waitlist = true;
2001-05-23 15:26:42 +02:00
break;
case 'f':
2009-06-27 09:00:34 +02:00
if (argc > 1)
{
2001-05-23 15:26:42 +02:00
lock_file = *argv++;
--argc;
}
2009-06-27 09:00:34 +02:00
else
{
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "Usage: -f <filename>\n");
exit(FINI_OK);
}
2001-05-23 15:26:42 +02:00
break;
2008-12-05 02:20:14 +01:00
case 'd':
2009-06-27 09:00:34 +02:00
if (argc > 1)
{
db_file = *argv++;
--argc;
}
2009-06-27 09:00:34 +02:00
else
{
FPRINTF(outfile, "Usage: -d <filename>\n");
exit(FINI_OK);
}
break;
case 'm':
sw_html_format = true;
break;
2001-05-23 15:26:42 +02:00
default:
FPRINTF(outfile, valid_switches);
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
break;
}
}
Firebird::PathName filename;
if (db_file && lock_file)
{
FPRINTF(outfile, "Switches -d and -f cannot be specified together\n");
exit(FINI_OK);
2001-05-23 15:26:42 +02:00
}
else if (db_file)
{
Firebird::PathName org_name = db_file;
Firebird::PathName db_name;
if (!ResolveDatabaseAlias(org_name, db_name))
{
db_name = org_name;
}
2001-05-23 15:26:42 +02:00
// Below code mirrors the one in JRD (PIO modules and Database class).
// Maybe it's worth putting it into common, if no better solution is found.
#ifdef WIN_NT
const HANDLE h = CreateFile(db_name.c_str(), GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, 0, 0);
if (h == INVALID_HANDLE_VALUE)
{
FPRINTF(outfile, "Unable to open the database file (%d).\n", GetLastError());
exit(FINI_OK);
}
BY_HANDLE_FILE_INFORMATION file_info;
GetFileInformationByHandle(h, &file_info);
const size_t len1 = sizeof(file_info.dwVolumeSerialNumber);
const size_t len2 = sizeof(file_info.nFileIndexHigh);
const size_t len3 = sizeof(file_info.nFileIndexLow);
UCHAR buffer[len1 + len2 + len3], *p = buffer;
memcpy(p, &file_info.dwVolumeSerialNumber, len1);
p += len1;
memcpy(p, &file_info.nFileIndexHigh, len2);
p += len2;
memcpy(p, &file_info.nFileIndexLow, len3);
CloseHandle(h);
#else
struct stat statistics;
if (stat(db_name.c_str(), &statistics) == -1)
{
FPRINTF(outfile, "Unable to open the database file.\n");
exit(FINI_OK);
}
const size_t len1 = sizeof(statistics.st_dev);
const size_t len2 = sizeof(statistics.st_ino);
UCHAR buffer[len1 + len2], *p = buffer;
memcpy(p, &statistics.st_dev, len1);
p += len1;
memcpy(p, &statistics.st_ino, len2);
#endif
2001-05-23 15:26:42 +02:00
Firebird::string file_id;
for (size_t i = 0; i < sizeof(buffer); i++)
{
TEXT hex[3];
sprintf(hex, "%02x", (int) buffer[i]);
file_id.append(hex);
}
2001-05-23 15:26:42 +02:00
filename.printf(LOCK_FILE, file_id.c_str());
}
else if (lock_file)
{
filename = lock_file;
}
else
{
FPRINTF(outfile, "Please specify either -d <database name> or -f <lock file name>\n");
exit(FINI_OK);
}
2008-12-05 02:20:14 +01:00
ISC_STATUS_ARRAY status_vector;
sh_mem shmem_data;
Firebird::AutoPtr<UCHAR, Firebird::ArrayDelete<UCHAR> > buffer;
lhb* LOCK_header = NULL;
2003-09-10 13:43:31 +02:00
if (db_file)
{
LOCK_header = (lhb*) ISC_map_file(status_vector, filename.c_str(),
prt_lock_init, NULL, 0, &shmem_data);
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (!LOCK_header)
{
FPRINTF(outfile, "Unable to access lock table.\n");
gds__print_status(status_vector);
exit(FINI_OK);
}
2001-05-23 15:26:42 +02:00
2009-08-14 09:25:09 +02:00
// Make sure the lock file is valid - if it's a zero length file we
// can't look at the header without causing a BUS error by going
// off the end of the mapped region.
2009-06-27 09:00:34 +02:00
if (shmem_data.sh_mem_length_mapped < sizeof(lhb))
{
2009-08-14 09:25:09 +02:00
// Mapped file is obviously too small to really be a lock file
FPRINTF(outfile, "Unable to access lock table - file too small.\n");
exit(FINI_OK);
}
2009-08-21 14:04:53 +02:00
if (sw_consistency)
{
#ifdef WIN_NT
ISC_mutex_init(MUTEX, shmem_data.sh_mem_name);
#endif
ISC_mutex_lock(MUTEX);
}
#ifdef USE_SHMEM_EXT
ULONG extentSize = shmem_data.sh_mem_length_mapped;
ULONG totalSize = LOCK_header->lhb_length;
ULONG extentsCount = totalSize / extentSize + (totalSize % extentSize == 0 ? 0 : 1);
try
{
buffer = new UCHAR[extentsCount * extentSize];
}
catch (const Firebird::BadAlloc&)
{
FPRINTF(outfile, "Insufficient memory for lock statistics.\n");
exit(FINI_OK);
}
2009-09-29 09:33:30 +02:00
memcpy((UCHAR*) buffer, LOCK_header, extentSize);
for (ULONG extent = 1; extent < extentsCount; ++extent)
{
Firebird::PathName extName;
sh_mem extData;
extName.printf("%s.ext%d", filename.c_str(), extent);
UCHAR* ext = (UCHAR*) ISC_map_file(status_vector, extName.c_str(),
prt_lock_init, NULL, 0, &extData);
if (! ext)
{
FPRINTF(outfile, "Could not map extent number %d, file %s.\n", extent, extName.c_str());
exit(FINI_OK);
}
memcpy(((UCHAR*) buffer) + extent * extentSize, ext, extentSize);
ISC_unmap_file(status_vector, &extData);
}
LOCK_header = (lhb*)(UCHAR*) buffer;
#elif (defined HAVE_MMAP || defined WIN_NT)
if (LOCK_header->lhb_length > shmem_data.sh_mem_length_mapped)
{
const ULONG length = LOCK_header->lhb_length;
LOCK_header = (lhb*) ISC_remap_file(status_vector, &shmem_data, length, false);
}
#endif
if (sw_consistency)
{
#ifndef USE_SHMEM_EXT
// To avoid changes in the lock file while we are dumping it - make
// a local buffer, lock the lock file, copy it, then unlock the
// lock file to let processing continue. Printing of the lock file
// will continue from the in-memory copy.
try
{
buffer = new UCHAR[LOCK_header->lhb_length];
}
catch (const Firebird::BadAlloc&)
{
FPRINTF(outfile, "Insufficient memory for consistent lock statistics.\n");
FPRINTF(outfile, "Try omitting the -c switch.\n");
exit(FINI_OK);
}
memcpy((UCHAR*) buffer, LOCK_header, LOCK_header->lhb_length);
LOCK_header = (lhb*)(UCHAR*) buffer;
#endif
ISC_mutex_unlock(MUTEX);
2009-08-21 14:04:53 +02:00
#ifdef WIN_NT
ISC_mutex_fini(MUTEX);
2009-08-21 14:04:53 +02:00
#endif
}
2001-05-23 15:26:42 +02:00
}
else if (lock_file)
{
const int fd = open(filename.c_str(), O_RDONLY | O_BINARY);
2009-06-27 09:00:34 +02:00
if (fd == -1)
{
FPRINTF(outfile, "Unable to open lock file.\n");
exit(FINI_OK);
}
struct stat file_stat;
2009-06-27 09:00:34 +02:00
if (fstat(fd, &file_stat) == -1)
{
close(fd);
FPRINTF(outfile, "Unable to retrieve lock file size.\n");
exit(FINI_OK);
}
2009-06-27 09:00:34 +02:00
if (!file_stat.st_size)
{
close(fd);
FPRINTF(outfile, "Lock file is empty.\n");
exit(FINI_OK);
}
try
{
buffer = new UCHAR[file_stat.st_size];
}
catch (const Firebird::BadAlloc&)
2009-06-27 09:00:34 +02:00
{
FPRINTF(outfile, "Insufficient memory to read lock file.\n");
exit(FINI_OK);
}
LOCK_header = (lhb*)(UCHAR*) buffer;
read(fd, LOCK_header, file_stat.st_size);
close(fd);
#ifdef USE_SHMEM_EXT
ULONG extentSize = file_stat.st_size;
ULONG totalSize = LOCK_header->lhb_length;
2009-09-29 09:33:30 +02:00
const ULONG extentsCount = totalSize / extentSize + (totalSize % extentSize == 0 ? 0 : 1);
UCHAR* newBuf = NULL;
try
{
newBuf = new UCHAR[extentsCount * extentSize];
}
catch (const Firebird::BadAlloc&)
{
FPRINTF(outfile, "Insufficient memory for lock statistics.\n");
exit(FINI_OK);
}
memcpy(newBuf, LOCK_header, extentSize);
buffer = newBuf;
for (ULONG extent = 1; extent < extentsCount; ++extent)
{
Firebird::PathName extName;
extName.printf("%s.ext%d", filename.c_str(), extent);
const int fd = open(extName.c_str(), O_RDONLY | O_BINARY);
if (fd == -1)
{
FPRINTF(outfile, "Unable to open lock file extent number %d, file %s.\n",
extent, extName.c_str());
exit(FINI_OK);
}
if (read(fd, ((UCHAR*) buffer) + extent * extentSize, extentSize) != extentSize)
{
FPRINTF(outfile, "Could not read lock file extent number %d, file %s.\n",
extent, extName.c_str());
exit(FINI_OK);
}
close(fd);
}
LOCK_header = (lhb*)(UCHAR*) buffer;
#endif
}
else
{
fb_assert(false);
2001-05-23 15:26:42 +02:00
}
fb_assert(LOCK_header);
2001-05-23 15:26:42 +02:00
2009-08-14 09:25:09 +02:00
// if we can't read this version - admit there's nothing to say and return.
2001-05-23 15:26:42 +02:00
if (LOCK_header->lhb_version != LHB_VERSION)
2003-09-10 13:43:31 +02:00
{
2008-12-05 02:20:14 +01:00
if (LOCK_header->lhb_type == 0 && LOCK_header->lhb_version == 0)
{
FPRINTF(outfile, "\tLock table is empty.\n");
}
2008-12-05 02:20:14 +01:00
else
{
FPRINTF(outfile, "\tUnable to read lock table version %d.\n",
2001-05-23 15:26:42 +02:00
LOCK_header->lhb_version);
2007-02-20 09:40:38 +01:00
}
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
}
2009-08-14 09:25:09 +02:00
// Print lock activity report
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (sw_interactive)
{
sw_html_format = false;
2003-09-14 02:46:05 +02:00
prt_lock_activity(outfile, LOCK_header, sw_interactive, (USHORT) sw_seconds,
(USHORT) sw_intervals);
2001-05-23 15:26:42 +02:00
exit(FINI_OK);
}
2009-08-14 09:25:09 +02:00
// Print lock header block
prt_html_begin(outfile);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "LOCK_HEADER BLOCK\n");
FPRINTF(outfile,
"\tVersion: %d, Active owner: %s, Length: %6"SLONGFORMAT
2003-09-14 02:46:05 +02:00
", Used: %6"SLONGFORMAT"\n",
LOCK_header->lhb_version, (const TEXT*)HtmlLink(preOwn, LOCK_header->lhb_active_owner),
2001-05-23 15:26:42 +02:00
LOCK_header->lhb_length, LOCK_header->lhb_used);
2008-01-16 08:40:12 +01:00
FPRINTF(outfile, "\tFlags: 0x%04X\n",
LOCK_header->lhb_flags);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile,
"\tEnqs: %6"UQUADFORMAT", Converts: %6"UQUADFORMAT
", Rejects: %6"UQUADFORMAT", Blocks: %6"UQUADFORMAT"\n",
2001-05-23 15:26:42 +02:00
LOCK_header->lhb_enqs, LOCK_header->lhb_converts,
LOCK_header->lhb_denies, LOCK_header->lhb_blocks);
FPRINTF(outfile,
"\tDeadlock scans: %6"UQUADFORMAT", Deadlocks: %6"UQUADFORMAT
2003-09-14 02:46:05 +02:00
", Scan interval: %3"ULONGFORMAT"\n",
2001-05-23 15:26:42 +02:00
LOCK_header->lhb_scans, LOCK_header->lhb_deadlocks,
LOCK_header->lhb_scan_interval);
FPRINTF(outfile,
"\tAcquires: %6"UQUADFORMAT", Acquire blocks: %6"UQUADFORMAT
2003-09-14 02:46:05 +02:00
", Spin count: %3"ULONGFORMAT"\n",
2001-05-23 15:26:42 +02:00
LOCK_header->lhb_acquires, LOCK_header->lhb_acquire_blocks,
LOCK_header->lhb_acquire_spins);
2009-06-27 09:00:34 +02:00
if (LOCK_header->lhb_acquire_blocks)
{
2007-11-12 15:26:44 +01:00
// CVC: MSVC up to v6 couldn't convert FB_UINT64 to double.
const float bottleneck =
(float) ((100. * (SINT64) LOCK_header->lhb_acquire_blocks) /
(SINT64) LOCK_header->lhb_acquires);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\tMutex wait: %3.1f%%\n", bottleneck);
}
else
FPRINTF(outfile, "\tMutex wait: 0.0%%\n");
2003-09-10 13:43:31 +02:00
SLONG hash_total_count = 0;
SLONG hash_max_count = 0;
SLONG hash_min_count = 10000000;
USHORT i = 0;
2008-12-31 06:06:08 +01:00
for (const srq* slot = LOCK_header->lhb_hash; i < LOCK_header->lhb_hash_slots; slot++, i++)
2003-09-10 13:43:31 +02:00
{
SLONG hash_lock_count = 0;
for (const srq* que_inst = (SRQ) SRQ_ABS_PTR(slot->srq_forward); que_inst != slot;
que_inst = (SRQ) SRQ_ABS_PTR(que_inst->srq_forward))
2003-09-10 13:43:31 +02:00
{
2001-05-23 15:26:42 +02:00
++hash_total_count;
++hash_lock_count;
}
if (hash_lock_count < hash_min_count)
hash_min_count = hash_lock_count;
if (hash_lock_count > hash_max_count)
hash_max_count = hash_lock_count;
}
FPRINTF(outfile, "\tHash slots: %4d, ", LOCK_header->lhb_hash_slots);
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "Hash lengths (min/avg/max): %4"SLONGFORMAT"/%4"SLONGFORMAT"/%4"SLONGFORMAT"\n",
2001-05-23 15:26:42 +02:00
hash_min_count, (hash_total_count / LOCK_header->lhb_hash_slots),
hash_max_count);
2008-01-16 08:40:12 +01:00
const shb* a_shb = (shb*) SRQ_ABS_PTR(LOCK_header->lhb_secondary);
FPRINTF(outfile,
"\tRemove node: %6"SLONGFORMAT", Insert queue: %6"SLONGFORMAT
", Insert prior: %6"SLONGFORMAT"\n",
a_shb->shb_remove_node, a_shb->shb_insert_que,
a_shb->shb_insert_prior);
2001-05-23 15:26:42 +02:00
prt_que(outfile, LOCK_header, "\tOwners", &LOCK_header->lhb_owners,
OFFSET(own*, own_lhb_owners), preOwn);
2001-05-23 15:26:42 +02:00
prt_que(outfile, LOCK_header, "\tFree owners",
&LOCK_header->lhb_free_owners, OFFSET(own*, own_lhb_owners));
2001-05-23 15:26:42 +02:00
prt_que(outfile, LOCK_header, "\tFree locks",
2008-01-16 08:40:12 +01:00
&LOCK_header->lhb_free_locks, OFFSET(lbl*, lbl_lhb_hash));
2001-05-23 15:26:42 +02:00
prt_que(outfile, LOCK_header, "\tFree requests",
2008-01-16 08:40:12 +01:00
&LOCK_header->lhb_free_requests, OFFSET(lrq*, lrq_lbl_requests));
2001-05-23 15:26:42 +02:00
2009-08-14 09:25:09 +02:00
// Print lock ordering option
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\tLock Ordering: %s\n",
2008-12-31 06:06:08 +01:00
(LOCK_header->lhb_flags & LHB_lock_ordering) ? "Enabled" : "Disabled");
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\n");
2009-08-14 09:25:09 +02:00
// Print known owners
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (sw_owners)
{
const srq* que_inst;
2009-06-27 09:00:34 +02:00
SRQ_LOOP(LOCK_header->lhb_owners, que_inst)
{
2001-05-23 15:26:42 +02:00
prt_owner(outfile, LOCK_header,
(own*) ((UCHAR*) que_inst - OFFSET(own*, own_lhb_owners)),
2001-05-23 15:26:42 +02:00
sw_requests, sw_waitlist);
}
}
2009-08-14 09:25:09 +02:00
// Print known locks
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (sw_locks || sw_series)
{
2004-05-12 21:23:17 +02:00
USHORT i2 = 0;
2008-12-31 06:06:08 +01:00
for (const srq* slot = LOCK_header->lhb_hash; i2 < LOCK_header->lhb_hash_slots; slot++, i2++)
2003-09-10 13:43:31 +02:00
{
for (const srq* que_inst = (SRQ) SRQ_ABS_PTR(slot->srq_forward); que_inst != slot;
que_inst = (SRQ) SRQ_ABS_PTR(que_inst->srq_forward))
2003-09-10 13:43:31 +02:00
{
2001-05-23 15:26:42 +02:00
prt_lock(outfile, LOCK_header,
2008-12-31 06:06:08 +01:00
(lbl*) ((UCHAR *) que_inst - OFFSET(lbl*, lbl_lhb_hash)), sw_series);
2003-09-10 13:43:31 +02:00
}
}
}
2001-05-23 15:26:42 +02:00
if (sw_history)
2008-12-31 06:06:08 +01:00
prt_history(outfile, LOCK_header, LOCK_header->lhb_history, "History");
2001-05-23 15:26:42 +02:00
2008-01-16 08:40:12 +01:00
prt_history(outfile, LOCK_header, a_shb->shb_history, "Event log");
2001-05-23 15:26:42 +02:00
prt_html_end(outfile);
2008-12-05 02:20:14 +01:00
if (db_file)
{
ISC_unmap_file(status_vector, &shmem_data);
}
2006-04-08 05:29:17 +02:00
return FINI_OK;
2001-05-23 15:26:42 +02:00
}
2008-12-31 06:06:08 +01:00
static void prt_lock_activity(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* header,
2001-05-23 15:26:42 +02:00
USHORT flag, USHORT seconds, USHORT intervals)
{
/**************************************
*
* p r t _ l o c k _ a c t i v i t y
*
**************************************
*
* Functional description
2008-12-05 02:20:14 +01:00
* Print a time-series lock activity report
2001-05-23 15:26:42 +02:00
*
**************************************/
2003-09-10 13:43:31 +02:00
time_t clock = time(NULL);
2003-10-08 10:42:48 +02:00
tm d = *localtime(&clock);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "%02d:%02d:%02d ", d.tm_hour, d.tm_min, d.tm_sec);
if (flag & SW_I_ACQUIRE)
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "acquire/s acqwait/s %%acqwait acqrtry/s rtrysuc/s ");
2001-05-23 15:26:42 +02:00
if (flag & SW_I_OPERATION)
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "enqueue/s convert/s downgrd/s dequeue/s readata/s wrtdata/s qrydata/s ");
2001-05-23 15:26:42 +02:00
if (flag & SW_I_TYPE)
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, " dblop/s rellop/s pagelop/s tranlop/s relxlop/s idxxlop/s misclop/s ");
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (flag & SW_I_WAIT)
{
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, " wait/s reject/s timeout/s blckast/s dirsig/s indsig/s ");
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, " wakeup/s dlkscan/s deadlck/s avlckwait(msec)");
}
FPRINTF(outfile, "\n");
2003-09-10 13:43:31 +02:00
lhb base = *header;
lhb prior = *header;
2001-05-23 15:26:42 +02:00
if (intervals == 0)
memset(&base, 0, sizeof(base));
2008-12-31 06:06:08 +01:00
for (ULONG i = 0; i < intervals; i++)
{
fflush(outfile);
2001-05-23 15:26:42 +02:00
#ifdef WIN_NT
Sleep((DWORD) seconds * 1000);
#else
sleep(seconds);
#endif
clock = time(NULL);
d = *localtime(&clock);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "%02d:%02d:%02d ", d.tm_hour, d.tm_min, d.tm_sec);
2009-06-27 09:00:34 +02:00
if (flag & SW_I_ACQUIRE)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_acquires - prior.lhb_acquires) / seconds,
2008-12-31 06:06:08 +01:00
(header->lhb_acquire_blocks - prior.lhb_acquire_blocks) / seconds,
(header->lhb_acquires - prior.lhb_acquires) ?
(100 * (header->lhb_acquire_blocks - prior.lhb_acquire_blocks)) /
(header->lhb_acquires - prior.lhb_acquires) : 0,
2001-05-23 15:26:42 +02:00
(header->lhb_acquire_retries -
prior.lhb_acquire_retries) / seconds,
(header->lhb_retry_success -
prior.lhb_retry_success) / seconds);
prior.lhb_acquires = header->lhb_acquires;
prior.lhb_acquire_blocks = header->lhb_acquire_blocks;
prior.lhb_acquire_retries = header->lhb_acquire_retries;
prior.lhb_retry_success = header->lhb_retry_success;
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_OPERATION)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_enqs - prior.lhb_enqs) / seconds,
(header->lhb_converts - prior.lhb_converts) / seconds,
(header->lhb_downgrades - prior.lhb_downgrades) / seconds,
(header->lhb_deqs - prior.lhb_deqs) / seconds,
(header->lhb_read_data - prior.lhb_read_data) / seconds,
(header->lhb_write_data - prior.lhb_write_data) / seconds,
2008-12-31 06:06:08 +01:00
(header->lhb_query_data - prior.lhb_query_data) / seconds);
2001-05-23 15:26:42 +02:00
prior.lhb_enqs = header->lhb_enqs;
prior.lhb_converts = header->lhb_converts;
prior.lhb_downgrades = header->lhb_downgrades;
prior.lhb_deqs = header->lhb_deqs;
prior.lhb_read_data = header->lhb_read_data;
prior.lhb_write_data = header->lhb_write_data;
prior.lhb_query_data = header->lhb_query_data;
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_TYPE)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" ",
(header->lhb_operations[Jrd::LCK_database] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_database]) / seconds,
(header->lhb_operations[Jrd::LCK_relation] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_relation]) / seconds,
(header->lhb_operations[Jrd::LCK_bdb] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_bdb]) / seconds,
(header->lhb_operations[Jrd::LCK_tra] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_tra]) / seconds,
(header->lhb_operations[Jrd::LCK_rel_exist] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_rel_exist]) / seconds,
(header->lhb_operations[Jrd::LCK_idx_exist] -
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_idx_exist]) / seconds,
(header->lhb_operations[0] - prior.lhb_operations[0]) / seconds);
prior.lhb_operations[Jrd::LCK_database] = header->lhb_operations[Jrd::LCK_database];
prior.lhb_operations[Jrd::LCK_relation] = header->lhb_operations[Jrd::LCK_relation];
prior.lhb_operations[Jrd::LCK_bdb] = header->lhb_operations[Jrd::LCK_bdb];
prior.lhb_operations[Jrd::LCK_tra] = header->lhb_operations[Jrd::LCK_tra];
2008-12-31 06:06:08 +01:00
prior.lhb_operations[Jrd::LCK_rel_exist] = header->lhb_operations[Jrd::LCK_rel_exist];
prior.lhb_operations[Jrd::LCK_idx_exist] = header->lhb_operations[Jrd::LCK_idx_exist];
2001-05-23 15:26:42 +02:00
prior.lhb_operations[0] = header->lhb_operations[0];
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_WAIT)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_waits - prior.lhb_waits) / seconds,
(header->lhb_denies - prior.lhb_denies) / seconds,
(header->lhb_timeouts - prior.lhb_timeouts) / seconds,
(header->lhb_blocks - prior.lhb_blocks) / seconds,
(header->lhb_wakeups - prior.lhb_wakeups) / seconds,
(header->lhb_scans - prior.lhb_scans) / seconds,
2008-01-16 08:40:12 +01:00
(header->lhb_deadlocks - prior.lhb_deadlocks) / seconds);
2001-05-23 15:26:42 +02:00
prior.lhb_waits = header->lhb_waits;
prior.lhb_denies = header->lhb_denies;
prior.lhb_timeouts = header->lhb_timeouts;
prior.lhb_blocks = header->lhb_blocks;
prior.lhb_wakeups = header->lhb_wakeups;
prior.lhb_scans = header->lhb_scans;
prior.lhb_deadlocks = header->lhb_deadlocks;
}
FPRINTF(outfile, "\n");
}
2003-09-10 13:43:31 +02:00
ULONG factor = seconds * intervals;
2001-05-23 15:26:42 +02:00
if (factor < 1)
factor = 1;
FPRINTF(outfile, "\nAverage: ");
2009-06-27 09:00:34 +02:00
if (flag & SW_I_ACQUIRE)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_acquires - base.lhb_acquires) / (factor),
(header->lhb_acquire_blocks -
base.lhb_acquire_blocks) / (factor),
2008-12-31 06:06:08 +01:00
(header->lhb_acquires - base.lhb_acquires) ?
(100 * (header->lhb_acquire_blocks - base.lhb_acquire_blocks)) /
(header->lhb_acquires - base.lhb_acquires) : 0,
(header->lhb_acquire_retries - base.lhb_acquire_retries) / (factor),
(header->lhb_retry_success - base.lhb_retry_success) / (factor));
2001-05-23 15:26:42 +02:00
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_OPERATION)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"
UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_enqs - base.lhb_enqs) / (factor),
(header->lhb_converts - base.lhb_converts) / (factor),
(header->lhb_downgrades - base.lhb_downgrades) / (factor),
(header->lhb_deqs - base.lhb_deqs) / (factor),
(header->lhb_read_data - base.lhb_read_data) / (factor),
(header->lhb_write_data - base.lhb_write_data) / (factor),
(header->lhb_query_data - base.lhb_query_data) / (factor));
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_TYPE)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" ",
(header->lhb_operations[Jrd::LCK_database] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_database]) / (factor),
(header->lhb_operations[Jrd::LCK_relation] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_relation]) / (factor),
(header->lhb_operations[Jrd::LCK_bdb] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_bdb]) / (factor),
(header->lhb_operations[Jrd::LCK_tra] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_tra]) / (factor),
(header->lhb_operations[Jrd::LCK_rel_exist] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_rel_exist]) / (factor),
(header->lhb_operations[Jrd::LCK_idx_exist] -
2008-12-31 06:06:08 +01:00
base.lhb_operations[Jrd::LCK_idx_exist]) / (factor),
(header->lhb_operations[0] - base.lhb_operations[0]) / (factor));
2001-05-23 15:26:42 +02:00
}
2009-06-27 09:00:34 +02:00
if (flag & SW_I_WAIT)
{
FPRINTF(outfile, "%9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" %9"UQUADFORMAT" %9"UQUADFORMAT
" %9"UQUADFORMAT" ",
2001-05-23 15:26:42 +02:00
(header->lhb_waits - base.lhb_waits) / (factor),
(header->lhb_denies - base.lhb_denies) / (factor),
(header->lhb_timeouts - base.lhb_timeouts) / (factor),
(header->lhb_blocks - base.lhb_blocks) / (factor),
(header->lhb_wakeups - base.lhb_wakeups) / (factor),
(header->lhb_scans - base.lhb_scans) / (factor),
2008-01-16 08:40:12 +01:00
(header->lhb_deadlocks - base.lhb_deadlocks) / (factor));
2001-05-23 15:26:42 +02:00
}
FPRINTF(outfile, "\n");
}
static void prt_lock_init(void*, sh_mem*, bool)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o c k _ i n i t
*
**************************************
*
* Functional description
* Initialize a lock table to looking -- i.e. don't do
* nuthin.
*
**************************************/
}
2008-12-31 06:06:08 +01:00
static void prt_history(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* LOCK_header,
SRQ_PTR history_header,
2003-09-10 13:43:31 +02:00
const SCHAR* title)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ h i s t o r y
*
**************************************
*
* Functional description
* Print history list of lock table.
*
**************************************/
FPRINTF(outfile, "%s:\n", title);
2008-01-16 08:40:12 +01:00
for (const his* history = (his*) SRQ_ABS_PTR(history_header); true;
history = (his*) SRQ_ABS_PTR(history->his_next))
2003-09-10 13:43:31 +02:00
{
2001-05-23 15:26:42 +02:00
if (history->his_operation)
FPRINTF(outfile,
" %s:\towner = %s, lock = %s, request = %s\n",
2001-05-23 15:26:42 +02:00
history_names[history->his_operation],
2008-12-05 02:20:14 +01:00
(const TEXT*)HtmlLink(preOwn, history->his_process),
(const TEXT*)HtmlLink(preLock, history->his_lock),
(const TEXT*)HtmlLink(preRequest, history->his_request));
2001-05-23 15:26:42 +02:00
if (history->his_next == history_header)
break;
}
}
2009-08-13 16:23:38 +02:00
static void prt_lock(OUTFILE outfile, const lhb* LOCK_header, lbl* lock, USHORT sw_series)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ l o c k
*
**************************************
*
* Functional description
2008-12-05 02:20:14 +01:00
* Print a formatted lock block
2001-05-23 15:26:42 +02:00
*
**************************************/
if (sw_series && lock->lbl_series != sw_series)
return;
if (!sw_html_format)
FPRINTF(outfile, "LOCK BLOCK %6"SLONGFORMAT"\n", SRQ_REL_PTR(lock));
else
{
const SLONG rel_lock = SRQ_REL_PTR(lock);
FPRINTF(outfile, "<a name=\"%s%"SLONGFORMAT"\">LOCK BLOCK %6"SLONGFORMAT"</a>\n",
preLock, rel_lock, rel_lock);
}
2001-05-23 15:26:42 +02:00
FPRINTF(outfile,
"\tSeries: %d, Parent: %s, State: %d, size: %d length: %d data: %"ULONGFORMAT"\n",
lock->lbl_series, (const TEXT*)HtmlLink(preLock, lock->lbl_parent), lock->lbl_state,
2001-05-23 15:26:42 +02:00
lock->lbl_size, lock->lbl_length, lock->lbl_data);
2008-12-05 02:20:14 +01:00
if ((lock->lbl_series == Jrd::LCK_bdb || lock->lbl_series == Jrd::LCK_btr_dont_gc) &&
lock->lbl_length == Jrd::PageNumber::getLockLen())
{
// Since fb 2.1 lock keys for page numbers (series == 3) contains
2007-09-07 09:41:07 +02:00
// page space number in high long of two-longs key. Lets print it
// in <page_space>:<page_number> format
2007-08-30 04:40:29 +02:00
const UCHAR* q = lock->lbl_key;
2008-12-05 02:20:14 +01:00
SLONG key;
2007-09-07 09:41:07 +02:00
memcpy(&key, q, sizeof(SLONG));
q += sizeof(SLONG);
2007-09-07 09:41:07 +02:00
ULONG pg_space;
memcpy(&pg_space, q, sizeof(SLONG));
2007-09-07 09:41:07 +02:00
FPRINTF(outfile, "\tKey: %04"ULONGFORMAT":%06"SLONGFORMAT",", pg_space, key);
}
2009-06-27 09:00:34 +02:00
else if (lock->lbl_length == 4)
{
2001-05-23 15:26:42 +02:00
SLONG key;
2007-09-02 11:52:42 +02:00
memcpy(&key, lock->lbl_key, 4);
2003-09-14 02:46:05 +02:00
FPRINTF(outfile, "\tKey: %06"SLONGFORMAT",", key);
2001-05-23 15:26:42 +02:00
}
2009-06-27 09:00:34 +02:00
else
{
2003-09-10 13:43:31 +02:00
UCHAR temp[512];
2007-09-02 11:52:42 +02:00
fb_assert(sizeof(temp) >= lock->lbl_length + 1); // Not enough, see <%d> below.
2003-09-10 13:43:31 +02:00
UCHAR* p = temp;
2007-09-02 11:52:42 +02:00
const UCHAR* end_temp = p + sizeof(temp) - 1;
2003-09-10 13:43:31 +02:00
const UCHAR* q = lock->lbl_key;
const UCHAR* const end = q + lock->lbl_length;
2008-12-31 06:06:08 +01:00
for (; q < end && p < end_temp; q++)
{
const UCHAR c = *q;
2008-12-31 06:06:08 +01:00
if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9') || c == '/')
{
2001-05-23 15:26:42 +02:00
*p++ = c;
}
2007-09-02 11:52:42 +02:00
else
{
char buf[6] = "";
int n = sprintf(buf, "<%d>", c);
if (n < 1 || p + n >= end_temp)
{
while (p < end_temp)
*p++ = '.';
break;
}
memcpy(p, buf, n);
p += n;
2001-05-23 15:26:42 +02:00
}
}
*p = 0;
FPRINTF(outfile, "\tKey: %s,", temp);
}
FPRINTF(outfile, " Flags: 0x%02X, Pending request count: %6d\n",
lock->lbl_flags, lock->lbl_pending_lrq_count);
2008-12-31 06:06:08 +01:00
prt_que(outfile, LOCK_header, "\tHash que", &lock->lbl_lhb_hash, OFFSET(lbl*, lbl_lhb_hash));
2001-05-23 15:26:42 +02:00
prt_que(outfile, LOCK_header, "\tRequests", &lock->lbl_requests,
OFFSET(lrq*, lrq_lbl_requests), preRequest);
2001-05-23 15:26:42 +02:00
const srq* que_inst;
2009-06-27 09:00:34 +02:00
SRQ_LOOP(lock->lbl_requests, que_inst)
{
2008-01-16 08:40:12 +01:00
const lrq* request = (lrq*) ((UCHAR*) que_inst - OFFSET(lrq*, lrq_lbl_requests));
2001-05-23 15:26:42 +02:00
FPRINTF(outfile,
"\t\tRequest %s, Owner: %s, State: %d (%d), Flags: 0x%02X\n",
2008-12-31 06:06:08 +01:00
(const TEXT*) HtmlLink(preRequest, SRQ_REL_PTR(request)),
(const TEXT*) HtmlLink(preOwn, request->lrq_owner), request->lrq_state,
2001-05-23 15:26:42 +02:00
request->lrq_requested, request->lrq_flags);
}
FPRINTF(outfile, "\n");
}
2003-09-04 23:26:15 +02:00
static void prt_owner(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* LOCK_header,
const own* owner,
2003-09-04 23:26:15 +02:00
bool sw_requests,
bool sw_waitlist)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ o w n e r
*
**************************************
*
* Functional description
* Print a formatted owner block.
*
**************************************/
const prc* process = (prc*) SRQ_ABS_PTR(owner->own_process);
if (!sw_html_format)
FPRINTF(outfile, "OWNER BLOCK %6"SLONGFORMAT"\n", SRQ_REL_PTR(owner));
else
{
const SLONG rel_owner = SRQ_REL_PTR(owner);
2008-12-05 02:20:14 +01:00
FPRINTF(outfile, "<a name=\"%s%"SLONGFORMAT"\">OWNER BLOCK %6"SLONGFORMAT"</a>\n",
preOwn, rel_owner, rel_owner);
}
FPRINTF(outfile, "\tOwner id: %6"QUADFORMAT"d, type: %1d, pending: %s\n",
2001-05-23 15:26:42 +02:00
owner->own_owner_id, owner->own_owner_type,
(const TEXT*)HtmlLink(preRequest, owner->own_pending_request));
2008-12-05 02:20:14 +01:00
2009-04-03 12:49:07 +02:00
FPRINTF(outfile, "\tProcess id: %6d (%s), thread id: %6"SIZEFORMAT"\n",
process->prc_process_id,
ISC_check_process_existence(process->prc_process_id) ? "Alive" : "Dead",
owner->own_thread_id);
2001-05-23 15:26:42 +02:00
{
const USHORT flags = owner->own_flags;
FPRINTF(outfile, "\tFlags: 0x%02X ", flags);
FPRINTF(outfile, " %s", (flags & OWN_blocking) ? "blkg" : " ");
FPRINTF(outfile, " %s", (flags & OWN_wakeup) ? "wake" : " ");
FPRINTF(outfile, " %s", (flags & OWN_scanned) ? "scan" : " ");
FPRINTF(outfile, " %s", (flags & OWN_waiting) ? "wait" : " ");
FPRINTF(outfile, " %s", (flags & OWN_waiting) ? ((flags & OWN_timeout) ? "tout" : "infn") : " ");
FPRINTF(outfile, " %s", (flags & OWN_signaled) ? "sgnl" : " ");
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\n");
}
prt_que(outfile, LOCK_header, "\tRequests", &owner->own_requests,
OFFSET(lrq*, lrq_own_requests), preRequest);
2008-12-31 06:06:08 +01:00
prt_que(outfile, LOCK_header, "\tBlocks", &owner->own_blocks, OFFSET(lrq*, lrq_own_blocks));
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (sw_waitlist)
{
2003-09-18 12:24:03 +02:00
waitque owner_list;
2001-05-23 15:26:42 +02:00
owner_list.waitque_depth = 0;
prt_owner_wait_cycle(outfile, LOCK_header, owner, 8, &owner_list);
}
FPRINTF(outfile, "\n");
2009-06-27 09:00:34 +02:00
if (sw_requests)
{
const srq* que_inst;
SRQ_LOOP(owner->own_requests, que_inst)
2001-05-23 15:26:42 +02:00
prt_request(outfile, LOCK_header,
2008-12-31 06:06:08 +01:00
(lrq*) ((UCHAR *) que_inst - OFFSET(lrq*, lrq_own_requests)));
}
2001-05-23 15:26:42 +02:00
}
2008-12-31 06:06:08 +01:00
static void prt_owner_wait_cycle(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* LOCK_header,
const own* owner,
2003-09-18 12:24:03 +02:00
USHORT indent, waitque *waiters)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ o w n e r _ w a i t _ c y c l e
*
**************************************
*
* Functional description
* For the given owner, print out the list of owners
* being waited on. The printout is recursive, up to
2008-12-05 02:20:14 +01:00
* a limit. It is recommended this be used with
2001-05-23 15:26:42 +02:00
* the -c consistency mode.
*
**************************************/
2009-06-27 08:23:36 +02:00
for (USHORT i = indent; i; i--)
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, " ");
2009-08-14 09:25:09 +02:00
// Check to see if we're in a cycle of owners - this might be
// a deadlock, or might not, if the owners haven't processed
// their blocking queues
2001-05-23 15:26:42 +02:00
2009-06-27 08:23:36 +02:00
for (USHORT i = 0; i < waiters->waitque_depth; i++)
2009-06-27 09:00:34 +02:00
if (SRQ_REL_PTR(owner) == waiters->waitque_entry[i])
{
FPRINTF(outfile, "%s (potential deadlock).\n",
2008-12-31 06:06:08 +01:00
(const TEXT*) HtmlLink(preOwn, SRQ_REL_PTR(owner)));
2001-05-23 15:26:42 +02:00
return;
}
2001-05-23 15:26:42 +02:00
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "%s waits on ", (const TEXT*) HtmlLink(preOwn, SRQ_REL_PTR(owner)));
2001-05-23 15:26:42 +02:00
if (!owner->own_pending_request)
FPRINTF(outfile, "nothing.\n");
2009-06-27 09:00:34 +02:00
else
{
if (waiters->waitque_depth >= FB_NELEM(waiters->waitque_entry))
{
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "Dependency too deep\n");
return;
}
2001-05-23 15:26:42 +02:00
waiters->waitque_entry[waiters->waitque_depth++] = SRQ_REL_PTR(owner);
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\n");
2008-01-16 08:40:12 +01:00
const lrq* owner_request = (lrq*) SRQ_ABS_PTR(owner->own_pending_request);
2003-11-04 00:59:24 +01:00
fb_assert(owner_request->lrq_type == type_lrq);
const bool owner_conversion = (owner_request->lrq_state > LCK_null);
2001-05-23 15:26:42 +02:00
2008-01-16 08:40:12 +01:00
const lbl* lock = (lbl*) SRQ_ABS_PTR(owner_request->lrq_lock);
2003-11-04 00:59:24 +01:00
fb_assert(lock->lbl_type == type_lbl);
2001-05-23 15:26:42 +02:00
int counter = 0;
const srq* que_inst;
2008-12-31 06:06:08 +01:00
SRQ_LOOP(lock->lbl_requests, que_inst)
{
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (counter++ > 50)
{
2009-06-27 08:23:36 +02:00
for (USHORT i = indent + 6; i; i--)
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, " ");
2008-12-31 06:06:08 +01:00
FPRINTF(outfile, "printout stopped after %d owners\n", counter - 1);
2001-05-23 15:26:42 +02:00
break;
}
2008-12-31 06:06:08 +01:00
const lrq* lock_request = (lrq*) ((UCHAR *) que_inst - OFFSET(lrq*, lrq_lbl_requests));
2003-11-04 00:59:24 +01:00
fb_assert(lock_request->lrq_type == type_lrq);
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (LOCK_header->lhb_flags & LHB_lock_ordering && !owner_conversion)
{
2009-08-14 09:25:09 +02:00
// Requests AFTER our request can't block us
2001-05-23 15:26:42 +02:00
if (owner_request == lock_request)
break;
2008-02-02 18:06:24 +01:00
if (compatibility[owner_request->lrq_requested]
[MAX(lock_request->lrq_state, lock_request->lrq_requested)])
{
2004-05-17 17:14:10 +02:00
continue;
}
2001-05-23 15:26:42 +02:00
}
2009-06-27 09:00:34 +02:00
else
{
2009-08-14 09:25:09 +02:00
// Requests AFTER our request CAN block us
2001-05-23 15:26:42 +02:00
if (lock_request == owner_request)
continue;
2008-02-02 18:06:24 +01:00
if (compatibility[owner_request->lrq_requested][lock_request->lrq_state])
2004-05-17 17:14:10 +02:00
continue;
}
const own* lock_owner = (own*) SRQ_ABS_PTR(lock_request->lrq_owner);
2008-12-31 06:06:08 +01:00
prt_owner_wait_cycle(outfile, LOCK_header, lock_owner, indent + 4, waiters);
2001-05-23 15:26:42 +02:00
}
waiters->waitque_depth--;
}
}
static void prt_request(OUTFILE outfile, const lhb* LOCK_header, const lrq* request)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ r e q u e s t
*
**************************************
*
* Functional description
* Print a format request block.
*
**************************************/
if (!sw_html_format)
FPRINTF(outfile, "REQUEST BLOCK %6"SLONGFORMAT"\n", SRQ_REL_PTR(request));
else
{
const SLONG rel_request = SRQ_REL_PTR(request);
2008-12-05 02:20:14 +01:00
FPRINTF(outfile, "<a name=\"%s%"SLONGFORMAT"\">REQUEST BLOCK %6"SLONGFORMAT"</a>\n",
preRequest, rel_request, rel_request);
}
FPRINTF(outfile, "\tOwner: %s, Lock: %s, State: %d, Mode: %d, Flags: 0x%02X\n",
2008-12-31 06:06:08 +01:00
(const TEXT*) HtmlLink(preOwn, request->lrq_owner),
(const TEXT*) HtmlLink(preLock, request->lrq_lock), request->lrq_state,
2001-05-23 15:26:42 +02:00
request->lrq_requested, request->lrq_flags);
2003-09-14 02:46:05 +02:00
FPRINTF(outfile, "\tAST: 0x%p, argument: 0x%p\n",
2001-05-23 15:26:42 +02:00
request->lrq_ast_routine, request->lrq_ast_argument);
prt_que2(outfile, LOCK_header, "\tlrq_own_requests",
&request->lrq_own_requests, OFFSET(lrq*, lrq_own_requests), preRequest);
2001-05-23 15:26:42 +02:00
prt_que2(outfile, LOCK_header, "\tlrq_lbl_requests",
&request->lrq_lbl_requests, OFFSET(lrq*, lrq_lbl_requests), preRequest);
2001-05-23 15:26:42 +02:00
prt_que2(outfile, LOCK_header, "\tlrq_own_blocks ",
2008-01-16 08:40:12 +01:00
&request->lrq_own_blocks, OFFSET(lrq*, lrq_own_blocks));
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "\n");
}
2008-12-31 06:06:08 +01:00
static void prt_que(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* LOCK_header,
const SCHAR* string, const srq* que_inst, USHORT que_offset,
const TEXT* prefix)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ q u e
*
**************************************
*
* Functional description
* Print the contents of a self-relative que.
*
**************************************/
const SLONG offset = SRQ_REL_PTR(que_inst);
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (offset == que_inst->srq_forward && offset == que_inst->srq_backward)
{
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "%s: *empty*\n", string);
return;
}
SLONG count = 0;
const srq* next;
SRQ_LOOP((*que_inst), next)
++count;
2001-05-23 15:26:42 +02:00
2009-04-03 12:49:07 +02:00
FPRINTF(outfile, "%s (%"SLONGFORMAT"):\tforward: %s, backward: %s\n", string, count,
2008-12-31 06:06:08 +01:00
(const TEXT*) HtmlLink(prefix, que_inst->srq_forward - que_offset),
(const TEXT*) HtmlLink(prefix, que_inst->srq_backward - que_offset));
2001-05-23 15:26:42 +02:00
}
2008-12-31 06:06:08 +01:00
static void prt_que2(OUTFILE outfile,
2003-09-10 13:43:31 +02:00
const lhb* LOCK_header,
const SCHAR* string, const srq* que_inst, USHORT que_offset,
const TEXT* prefix)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* p r t _ q u e 2
*
**************************************
*
* Functional description
* Print the contents of a self-relative que.
* But don't try to count the entries, as they might be invalid
*
**************************************/
const SLONG offset = SRQ_REL_PTR(que_inst);
2001-05-23 15:26:42 +02:00
2009-06-27 09:00:34 +02:00
if (offset == que_inst->srq_forward && offset == que_inst->srq_backward)
{
2001-05-23 15:26:42 +02:00
FPRINTF(outfile, "%s: *empty*\n", string);
return;
}
FPRINTF(outfile, "%s:\tforward: %s, backward: %s\n", string,
2008-12-31 06:06:08 +01:00
(const TEXT*) HtmlLink(prefix, que_inst->srq_forward - que_offset),
(const TEXT*) HtmlLink(prefix, que_inst->srq_backward - que_offset));
2001-05-23 15:26:42 +02:00
}
static void prt_html_begin(OUTFILE outfile)
{
2009-08-13 16:23:38 +02:00
/**************************************
*
* p r t _ h t m l _ b e g i n
*
**************************************
*
* Functional description
* Print the html header if heeded
*
**************************************/
if (!sw_html_format)
return;
2008-12-05 02:20:14 +01:00
FPRINTF(outfile, "<html><head><title>%s</title></head><body>", "Lock table");
FPRINTF(outfile, "<pre>");
2008-12-05 02:20:14 +01:00
}
static void prt_html_end(OUTFILE outfile)
{
2009-08-13 16:23:38 +02:00
/**************************************
*
* p r t _ h t m l _ e n d
*
**************************************
*
* Functional description
* Print the html finishing items
*
**************************************/
if (!sw_html_format)
return;
2008-12-05 02:20:14 +01:00
FPRINTF(outfile, "</pre>");
FPRINTF(outfile, "</body></html>");
}