mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 06:44:16 +01:00
221 lines
6 KiB
C
221 lines
6 KiB
C
/* Copyright (c) 2011, 2012, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA */
|
|
|
|
#ifndef PFS_DIGEST_H
|
|
#define PFS_DIGEST_H
|
|
|
|
/**
|
|
@file storage/perfschema/pfs_digest.h
|
|
Statement Digest data structures (declarations).
|
|
*/
|
|
|
|
#include "pfs_column_types.h"
|
|
#include "lf.h"
|
|
#include "pfs_stat.h"
|
|
|
|
#define PFS_SIZE_OF_A_TOKEN 2
|
|
|
|
extern bool flag_statements_digest;
|
|
extern ulong digest_max;
|
|
extern ulong digest_lost;
|
|
struct PFS_thread;
|
|
|
|
/* Fixed, per MD5 hash. */
|
|
#define PFS_MD5_SIZE 16
|
|
|
|
/**
|
|
Structure to store a MD5 hash value (digest) for a statement.
|
|
*/
|
|
struct PFS_digest_hash
|
|
{
|
|
unsigned char m_md5[PFS_MD5_SIZE];
|
|
};
|
|
|
|
/** A statement digest stat record. */
|
|
struct PFS_statements_digest_stat
|
|
{
|
|
/**
|
|
Digest MD5 Hash.
|
|
*/
|
|
PFS_digest_hash m_digest_hash;
|
|
|
|
/**
|
|
Digest Storage.
|
|
*/
|
|
PSI_digest_storage m_digest_storage;
|
|
|
|
/**
|
|
Statement stat.
|
|
*/
|
|
PFS_statement_stat m_stat;
|
|
|
|
/**
|
|
First Seen/last seen.
|
|
*/
|
|
ulonglong m_first_seen;
|
|
ulonglong m_last_seen;
|
|
|
|
/** Reset data for this record. */
|
|
void reset_data();
|
|
/** Reset data and remove index for this record. */
|
|
void reset_index(PFS_thread *thread);
|
|
};
|
|
|
|
int init_digest(const PFS_global_param *param);
|
|
void cleanup_digest();
|
|
|
|
int init_digest_hash(void);
|
|
void cleanup_digest_hash(void);
|
|
PFS_statement_stat* find_or_create_digest(PFS_thread*,
|
|
PSI_digest_storage*);
|
|
|
|
void get_digest_text(char* digest_text, PSI_digest_storage*);
|
|
|
|
void reset_esms_by_digest();
|
|
|
|
/* Exposing the data directly, for iterators. */
|
|
extern PFS_statements_digest_stat *statements_digest_stat_array;
|
|
|
|
/* Instrumentation callbacks for pfs.cc */
|
|
|
|
struct PSI_digest_locker* pfs_digest_start_v1(PSI_statement_locker *locker);
|
|
PSI_digest_locker* pfs_digest_add_token_v1(PSI_digest_locker *locker,
|
|
uint token,
|
|
OPAQUE_LEX_YYSTYPE *yylval);
|
|
|
|
static inline void digest_reset(PSI_digest_storage *digest)
|
|
{
|
|
digest->m_full= false;
|
|
digest->m_byte_count= 0;
|
|
}
|
|
|
|
static inline void digest_copy(PSI_digest_storage *to, const PSI_digest_storage *from)
|
|
{
|
|
if (from->m_byte_count > 0)
|
|
{
|
|
to->m_full= from->m_full;
|
|
to->m_byte_count= from->m_byte_count;
|
|
DBUG_ASSERT(to->m_byte_count <= PSI_MAX_DIGEST_STORAGE_SIZE);
|
|
memcpy(to->m_token_array, from->m_token_array, to->m_byte_count);
|
|
}
|
|
else
|
|
{
|
|
DBUG_ASSERT(! from->m_full);
|
|
DBUG_ASSERT(from->m_byte_count == 0);
|
|
to->m_full= false;
|
|
to->m_byte_count= 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Function to read a single token from token array.
|
|
*/
|
|
inline int read_token(PSI_digest_storage *digest_storage,
|
|
int index, uint *tok)
|
|
{
|
|
DBUG_ASSERT(index <= digest_storage->m_byte_count);
|
|
DBUG_ASSERT(digest_storage->m_byte_count <= PSI_MAX_DIGEST_STORAGE_SIZE);
|
|
|
|
if (index + PFS_SIZE_OF_A_TOKEN <= digest_storage->m_byte_count)
|
|
{
|
|
unsigned char *src= & digest_storage->m_token_array[index];
|
|
*tok= src[0] | (src[1] << 8);
|
|
return index + PFS_SIZE_OF_A_TOKEN;
|
|
}
|
|
|
|
/* The input byte stream is exhausted. */
|
|
*tok= 0;
|
|
return PSI_MAX_DIGEST_STORAGE_SIZE + 1;
|
|
}
|
|
|
|
/**
|
|
Function to store a single token in token array.
|
|
*/
|
|
inline void store_token(PSI_digest_storage* digest_storage, uint token)
|
|
{
|
|
DBUG_ASSERT(digest_storage->m_byte_count >= 0);
|
|
DBUG_ASSERT(digest_storage->m_byte_count <= PSI_MAX_DIGEST_STORAGE_SIZE);
|
|
|
|
if (digest_storage->m_byte_count + PFS_SIZE_OF_A_TOKEN <= PSI_MAX_DIGEST_STORAGE_SIZE)
|
|
{
|
|
unsigned char* dest= & digest_storage->m_token_array[digest_storage->m_byte_count];
|
|
dest[0]= token & 0xff;
|
|
dest[1]= (token >> 8) & 0xff;
|
|
digest_storage->m_byte_count+= PFS_SIZE_OF_A_TOKEN;
|
|
}
|
|
else
|
|
{
|
|
digest_storage->m_full= true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Function to read an identifier from token array.
|
|
*/
|
|
inline int read_identifier(PSI_digest_storage* digest_storage,
|
|
int index, char ** id_string, int *id_length)
|
|
{
|
|
int new_index;
|
|
DBUG_ASSERT(index <= digest_storage->m_byte_count);
|
|
DBUG_ASSERT(digest_storage->m_byte_count <= PSI_MAX_DIGEST_STORAGE_SIZE);
|
|
|
|
/*
|
|
token + length + string are written in an atomic way,
|
|
so we do always expect a length + string here
|
|
*/
|
|
unsigned char *src= & digest_storage->m_token_array[index];
|
|
uint length= src[0] | (src[1] << 8);
|
|
*id_string= (char *) (src + 2);
|
|
*id_length= length;
|
|
|
|
new_index= index + PFS_SIZE_OF_A_TOKEN + length;
|
|
DBUG_ASSERT(new_index <= digest_storage->m_byte_count);
|
|
return new_index;
|
|
}
|
|
|
|
/**
|
|
Function to store an identifier in token array.
|
|
*/
|
|
inline void store_token_identifier(PSI_digest_storage* digest_storage,
|
|
uint token,
|
|
uint id_length, const char *id_name)
|
|
{
|
|
DBUG_ASSERT(digest_storage->m_byte_count >= 0);
|
|
DBUG_ASSERT(digest_storage->m_byte_count <= PSI_MAX_DIGEST_STORAGE_SIZE);
|
|
|
|
uint bytes_needed= 2 * PFS_SIZE_OF_A_TOKEN + id_length;
|
|
if (digest_storage->m_byte_count + bytes_needed <= PSI_MAX_DIGEST_STORAGE_SIZE)
|
|
{
|
|
unsigned char* dest= & digest_storage->m_token_array[digest_storage->m_byte_count];
|
|
/* Write the token */
|
|
dest[0]= token & 0xff;
|
|
dest[1]= (token >> 8) & 0xff;
|
|
/* Write the string length */
|
|
dest[2]= id_length & 0xff;
|
|
dest[3]= (id_length >> 8) & 0xff;
|
|
/* Write the string data */
|
|
if (id_length > 0)
|
|
{
|
|
strncpy((char *)(dest + 4), id_name, id_length);
|
|
}
|
|
digest_storage->m_byte_count+= bytes_needed;
|
|
}
|
|
else
|
|
{
|
|
digest_storage->m_full= true;
|
|
}
|
|
}
|
|
|
|
#endif
|