8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 06:03:02 +01:00
firebird-mirror/src/jrd/perf.cpp
2009-09-23 15:43:33 +00:00

408 lines
8.9 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: perf.cpp
* DESCRIPTION: Performance monitoring 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): ______________________________________.
*
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "DELTA" port
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "IMP" port
*
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
*
*/
#include "firebird.h"
#include <stdio.h>
#include <limits.h>
#include "../jrd/common.h"
#include "../jrd/ibase.h"
#include "../jrd/perf.h"
#include "../jrd/gds_proto.h"
#include "../jrd/perf_proto.h"
#include "../jrd/gdsassert.h"
#if defined(TIME_WITH_SYS_TIME)
#include <sys/time.h>
#include <time.h>
#elif defined(HAVE_SYS_TIME_H)
#include <sys/time.h>
#else
#include <time.h>
#endif
#ifdef HAVE_SYS_TIMEB_H
#include <sys/timeb.h>
#endif
static SINT64 get_parameter(const SCHAR**);
#ifndef HAVE_TIMES
static void times(struct tms*);
#endif
static const SCHAR items[] =
{
isc_info_reads,
isc_info_writes,
isc_info_fetches,
isc_info_marks,
isc_info_page_size, isc_info_num_buffers,
isc_info_current_memory, isc_info_max_memory
};
static const SCHAR* report = "elapsed = !e cpu = !u reads = !r writes = !w fetches = !f marks = !m$";
#if defined(WIN_NT) && !defined(CLOCKS_PER_SEC)
#define TICK 100
#endif
/* EKU: TICK (sys/param.h) and CLOCKS_PER_SEC (time.h) may both be defined */
#if !defined(TICK) && defined(CLOCKS_PER_SEC)
#define TICK ((SLONG)CLOCKS_PER_SEC)
#endif
#ifndef TICK
#define TICK ((SLONG)CLK_TCK)
#endif
template<typename P>
static int perf_format(const P* before, const P* after,
const SCHAR* string, SCHAR* buffer, SSHORT* buf_len)
{
/**************************************
*
* P E R F _ f o r m a t
*
**************************************
*
* Functional description
* Format a buffer with statistical stuff under control of formatting
* string. Substitute in appropriate stuff. Return the length of the
* formatting output.
*
**************************************/
SCHAR c;
SLONG buffer_length = buf_len ? *buf_len : 0;
SCHAR* p = buffer;
while ((c = *string++) && c != '$')
{
if (c != '!')
*p++ = c;
else
{
SINT64 delta;
switch (c = *string++)
{
case 'r':
delta = after->perf_reads - before->perf_reads;
break;
case 'w':
delta = after->perf_writes - before->perf_writes;
break;
case 'f':
delta = after->perf_fetches - before->perf_fetches;
break;
case 'm':
delta = after->perf_marks - before->perf_marks;
break;
case 'd':
delta = after->perf_current_memory - before->perf_current_memory;
break;
case 'p':
delta = after->perf_page_size;
break;
case 'b':
delta = after->perf_buffers;
break;
case 'c':
delta = after->perf_current_memory;
break;
case 'x':
delta = after->perf_max_memory;
break;
case 'e':
delta = after->perf_elapsed - before->perf_elapsed;
break;
case 'u':
delta = after->perf_times.tms_utime - before->perf_times.tms_utime;
break;
case 's':
delta = after->perf_times.tms_stime - before->perf_times.tms_stime;
break;
default:
sprintf(p, "?%c?", c);
while (*p)
p++;
}
switch (c)
{
case 'r':
case 'w':
case 'f':
case 'm':
case 'd':
case 'p':
case 'b':
case 'c':
case 'x':
sprintf(p, "%"SQUADFORMAT, delta);
while (*p)
p++;
break;
case 'u':
case 's':
sprintf(p, "%"SQUADFORMAT".%.2"SQUADFORMAT, delta / TICK, (delta % TICK) * 100 / TICK);
while (*p)
p++;
break;
case 'e':
sprintf(p, "%"SQUADFORMAT".%.2"SQUADFORMAT, delta / 100, delta % 100);
while (*p)
p++;
break;
}
}
}
*p = 0;
const int length = p - buffer;
if (buffer_length && (buffer_length -= length) >= 0) {
memset(p, ' ', buffer_length);
}
return length;
}
template<typename P>
static void perf_get_info(FB_API_HANDLE* handle, P* perf)
{
/**************************************
*
* P E R F _ g e t _ i n f o
*
**************************************
*
* Functional description
* Acquire timing and performance information. Some info comes
* from the system and some from the database.
*
**************************************/
SSHORT buffer_length, item_length;
ISC_STATUS_ARRAY jrd_status;
#ifdef HAVE_GETTIMEOFDAY
struct timeval tp;
#else
struct timeb time_buffer;
#define LARGE_NUMBER 696600000 /* to avoid overflow, get rid of decades) */
#endif
/* If there isn't a database, zero everything out */
if (!*handle) {
memset(perf, 0, sizeof(PERF));
}
/* Get system times */
times(&perf->perf_times);
#ifdef HAVE_GETTIMEOFDAY
GETTIMEOFDAY(&tp);
perf->perf_elapsed = tp.tv_sec * 100 + tp.tv_usec / 10000;
#else
ftime(&time_buffer);
perf->perf_elapsed = (time_buffer.time - LARGE_NUMBER) * 100 + (time_buffer.millitm / 10);
#endif
if (!*handle)
return;
SCHAR buffer[256];
buffer_length = sizeof(buffer);
item_length = sizeof(items);
isc_database_info(jrd_status, handle, item_length, items, buffer_length, buffer);
const char* p = buffer;
while (true)
switch (*p++)
{
case isc_info_reads:
perf->perf_reads = get_parameter(&p);
break;
case isc_info_writes:
perf->perf_writes = get_parameter(&p);
break;
case isc_info_marks:
perf->perf_marks = get_parameter(&p);
break;
case isc_info_fetches:
perf->perf_fetches = get_parameter(&p);
break;
case isc_info_num_buffers:
perf->perf_buffers = get_parameter(&p);
break;
case isc_info_page_size:
perf->perf_page_size = get_parameter(&p);
break;
case isc_info_current_memory:
perf->perf_current_memory = get_parameter(&p);
break;
case isc_info_max_memory:
perf->perf_max_memory = get_parameter(&p);
break;
case isc_info_end:
return;
case isc_info_error:
switch (p[2])
{
case isc_info_marks:
perf->perf_marks = 0;
break;
case isc_info_current_memory:
perf->perf_current_memory = 0;
break;
case isc_info_max_memory:
perf->perf_max_memory = 0;
break;
}
{
const SLONG temp = isc_vax_integer(p, 2);
fb_assert(temp <= MAX_SSHORT);
p += temp + 2;
}
perf->perf_marks = 0;
break;
default:
return;
}
}
template<typename P>
static void perf_report(const P* before, const P* after, SCHAR* buffer, SSHORT* buf_len)
{
/**************************************
*
* p e r f _ r e p o r t
*
**************************************
*
* Functional description
* Report a common set of parameters.
*
**************************************/
perf_format<P>(before, after, report, buffer, buf_len);
}
static SINT64 get_parameter(const SCHAR** ptr)
{
/**************************************
*
* g e t _ p a r a m e t e r
*
**************************************
*
* Functional description
* Get a parameter (length is encoded), convert to internal form,
* and return.
*
**************************************/
SSHORT l = *(*ptr)++;
l += (*(*ptr)++) << 8;
const SINT64 parameter = isc_portable_integer(reinterpret_cast<const ISC_UCHAR*>(*ptr), l);
*ptr += l;
return parameter;
}
#ifndef HAVE_TIMES
static void times(struct tms* buffer)
{
/**************************************
*
* t i m e s
*
**************************************
*
* Functional description
* Emulate the good old unix call "times". Only both with user time.
*
**************************************/
buffer->tms_utime = clock();
}
#endif
int API_ROUTINE perf_format(const PERF* before, const PERF* after,
const SCHAR* string, SCHAR* buffer, SSHORT* buf_len)
{
return perf_format<PERF>(before, after, string, buffer, buf_len);
}
int API_ROUTINE perf64_format(const PERF64* before, const PERF64* after,
const SCHAR* string, SCHAR* buffer, SSHORT* buf_len)
{
return perf_format<PERF64>(before, after, string, buffer, buf_len);
}
void API_ROUTINE perf_get_info(FB_API_HANDLE* handle, PERF* perf)
{
perf_get_info<PERF>(handle, perf);
}
void API_ROUTINE perf64_get_info(FB_API_HANDLE* handle, PERF64* perf)
{
perf_get_info<PERF64>(handle, perf);
}
void API_ROUTINE perf_report(const PERF* before, const PERF* after, SCHAR* buffer, SSHORT* buf_len)
{
perf_report<PERF>(before, after, buffer, buf_len);
}
void API_ROUTINE perf64_report(const PERF64* before, const PERF64* after, SCHAR* buffer, SSHORT* buf_len)
{
perf_report<PERF64>(before, after, buffer, buf_len);
}