mirror of
https://github.com/MariaDB/server.git
synced 2025-01-16 03:52:35 +01:00
963b19861f
TO DO: Enable this in CMake-based builds. ------------------------------------------------------------ revno: 3474 revision-id: marko.makela@oracle.com-20100520104042-ma2nsscqdvwoph8k parent: marko.makela@oracle.com-20100519081618-h38q02qxuvcowbtk committer: Marko Mäkelä <marko.makela@oracle.com> branch nick: 5.1-innodb timestamp: Thu 2010-05-20 13:40:42 +0300 message: Bug#53593: Add some instrumentation to improve Valgrind sensitivity BUILD/*: Add valgrind_configs=--with-valgrind. BUILD/*: Remove -USAFEMALLOC from valgrind_flags. configure.in: Add AC_ARG_WITH(valgrind) and HAVE_VALGRIND. include/my_sys.h: Define a number of MEM_ wrappers for VALGRIND_ functions. include/my_sys.h: Make TRASH do MEM_UNDEFINED(). include/m_string.h: Remove unused macro bzero_if_purify(A,B). _mymalloc(): Declare MEM_UNDEFINED() on the allocated memory. _myfree(): Declare MEM_NOACCESS() on the freed memory. storage/innobase/include/univ.i: Enable UNIV_DEBUG_VALGRIND based on HAVE_VALGRIND rather than HAVE_purify. Possible things to do: * In my_global.h, remove the defined(HAVE_purify) condition from the _WIN32 uint3korr(). * In my_global.h *int*korr(), use | instead of + in order to keep the Valgrind V bits accurate * Consider replacing HAVE_purify with HAVE_VALGRIND * Use VALGRIND_CREATE_BLOCK, VALGRIND_DISCARD in mem_root and similar places
367 lines
12 KiB
C
367 lines
12 KiB
C
/* Copyright (C) 2000 MySQL AB
|
|
|
|
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, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
/* There may be prolems include all of theese. Try to test in
|
|
configure with ones are needed? */
|
|
|
|
/* This is needed for the definitions of strchr... on solaris */
|
|
|
|
#ifndef _m_string_h
|
|
#define _m_string_h
|
|
|
|
#include "my_global.h" /* HAVE_* */
|
|
|
|
#ifndef __USE_GNU
|
|
#define __USE_GNU /* We want to use stpcpy */
|
|
#endif
|
|
#if defined(HAVE_STRINGS_H)
|
|
#include <strings.h>
|
|
#endif
|
|
#if defined(HAVE_STRING_H)
|
|
#include <string.h>
|
|
#endif
|
|
|
|
/* need by my_vsnprintf */
|
|
#include <stdarg.h>
|
|
|
|
#ifdef _AIX
|
|
#undef HAVE_BCMP
|
|
#endif
|
|
|
|
/* This is needed for the definitions of bzero... on solaris */
|
|
#if defined(HAVE_STRINGS_H)
|
|
#include <strings.h>
|
|
#endif
|
|
|
|
/* This is needed for the definitions of memcpy... on solaris */
|
|
#if defined(HAVE_MEMORY_H) && !defined(__cplusplus)
|
|
#include <memory.h>
|
|
#endif
|
|
|
|
#if !defined(HAVE_MEMCPY) && !defined(HAVE_MEMMOVE)
|
|
# define memcpy(d, s, n) bcopy ((s), (d), (n))
|
|
# define memset(A,C,B) bfill((A),(B),(C))
|
|
# define memmove(d, s, n) bmove ((d), (s), (n))
|
|
#elif defined(HAVE_MEMMOVE)
|
|
# define bmove(d, s, n) memmove((d), (s), (n))
|
|
#else
|
|
# define memmove(d, s, n) bmove((d), (s), (n)) /* our bmove */
|
|
#endif
|
|
|
|
/* Unixware 7 */
|
|
#if !defined(HAVE_BFILL)
|
|
# define bfill(A,B,C) memset((A),(C),(B))
|
|
# define bmove_align(A,B,C) memcpy((A),(B),(C))
|
|
#endif
|
|
|
|
#if !defined(HAVE_BCMP)
|
|
# define bcopy(s, d, n) memcpy((d), (s), (n))
|
|
# define bcmp(A,B,C) memcmp((A),(B),(C))
|
|
# define bzero(A,B) memset((A),0,(B))
|
|
# define bmove_align(A,B,C) memcpy((A),(B),(C))
|
|
#endif
|
|
|
|
#if defined(__cplusplus)
|
|
extern "C" {
|
|
#endif
|
|
|
|
/*
|
|
my_str_malloc() and my_str_free() are assigned to implementations in
|
|
strings/alloc.c, but can be overridden in the calling program.
|
|
*/
|
|
extern void *(*my_str_malloc)(size_t);
|
|
extern void (*my_str_free)(void *);
|
|
|
|
#if defined(HAVE_STPCPY)
|
|
#define strmov(A,B) stpcpy((A),(B))
|
|
#ifndef stpcpy
|
|
extern char *stpcpy(char *, const char *); /* For AIX with gcc 2.95.3 */
|
|
#endif
|
|
#endif
|
|
|
|
/* Declared in int2str() */
|
|
extern char NEAR _dig_vec_upper[];
|
|
extern char NEAR _dig_vec_lower[];
|
|
|
|
#ifndef strmov
|
|
#define strmov_overlapp(A,B) strmov(A,B)
|
|
#define strmake_overlapp(A,B,C) strmake(A,B,C)
|
|
#endif
|
|
|
|
#ifdef BAD_MEMCPY /* Problem with gcc on Alpha */
|
|
#define memcpy_fixed(A,B,C) bmove((A),(B),(C))
|
|
#else
|
|
#define memcpy_fixed(A,B,C) memcpy((A),(B),(C))
|
|
#endif
|
|
|
|
#if (!defined(USE_BMOVE512) || defined(HAVE_purify)) && !defined(bmove512)
|
|
#define bmove512(A,B,C) memcpy(A,B,C)
|
|
#endif
|
|
|
|
/* Prototypes for string functions */
|
|
|
|
#if !defined(bfill) && !defined(HAVE_BFILL)
|
|
extern void bfill(uchar *dst,size_t len,pchar fill);
|
|
#endif
|
|
|
|
#if !defined(bzero) && !defined(HAVE_BZERO)
|
|
extern void bzero(uchar * dst,size_t len);
|
|
#endif
|
|
|
|
#if !defined(bcmp) && !defined(HAVE_BCMP)
|
|
extern size_t bcmp(const uchar *s1,const uchar *s2,size_t len);
|
|
#endif
|
|
#ifdef HAVE_purify
|
|
extern size_t my_bcmp(const uchar *s1,const uchar *s2,size_t len);
|
|
#undef bcmp
|
|
#define bcmp(A,B,C) my_bcmp((A),(B),(C))
|
|
#endif /* HAVE_purify */
|
|
|
|
#ifndef bmove512
|
|
extern void bmove512(uchar *dst,const uchar *src,size_t len);
|
|
#endif
|
|
|
|
#if !defined(HAVE_BMOVE) && !defined(bmove)
|
|
extern void bmove(uuchar *dst, const uchar *src,size_t len);
|
|
#endif
|
|
|
|
extern void bmove_upp(uchar *dst,const uchar *src,size_t len);
|
|
extern void bchange(uchar *dst,size_t old_len,const uchar *src,
|
|
size_t new_len,size_t tot_len);
|
|
extern void strappend(char *s,size_t len,pchar fill);
|
|
extern char *strend(const char *s);
|
|
extern char *strcend(const char *, pchar);
|
|
extern char *strfield(char *src,int fields,int chars,int blanks,
|
|
int tabch);
|
|
extern char *strfill(char * s,size_t len,pchar fill);
|
|
extern size_t strinstr(const char *str,const char *search);
|
|
extern size_t r_strinstr(const char *str, size_t from, const char *search);
|
|
extern char *strkey(char *dst,char *head,char *tail,char *flags);
|
|
extern char *strmake(char *dst,const char *src,size_t length);
|
|
|
|
#ifndef strmov
|
|
extern char *strmov(char *dst,const char *src);
|
|
#else
|
|
extern char *strmov_overlapp(char *dst,const char *src);
|
|
#endif
|
|
extern char *strnmov(char *dst,const char *src,size_t n);
|
|
extern char *strsuff(const char *src,const char *suffix);
|
|
extern char *strcont(const char *src,const char *set);
|
|
extern char *strxcat _VARARGS((char *dst,const char *src, ...));
|
|
extern char *strxmov _VARARGS((char *dst,const char *src, ...));
|
|
extern char *strxcpy _VARARGS((char *dst,const char *src, ...));
|
|
extern char *strxncat _VARARGS((char *dst,size_t len, const char *src, ...));
|
|
extern char *strxnmov _VARARGS((char *dst,size_t len, const char *src, ...));
|
|
extern char *strxncpy _VARARGS((char *dst,size_t len, const char *src, ...));
|
|
|
|
/* Prototypes of normal stringfunctions (with may ours) */
|
|
|
|
#ifdef WANT_STRING_PROTOTYPES
|
|
extern char *strcat(char *, const char *);
|
|
extern char *strchr(const char *, pchar);
|
|
extern char *strrchr(const char *, pchar);
|
|
extern char *strcpy(char *, const char *);
|
|
extern int strcmp(const char *, const char *);
|
|
#ifndef __GNUC__
|
|
extern size_t strlen(const char *);
|
|
#endif
|
|
#endif
|
|
#ifndef HAVE_STRNLEN
|
|
extern size_t strnlen(const char *s, size_t n);
|
|
#endif
|
|
|
|
#if !defined(__cplusplus)
|
|
#ifndef HAVE_STRPBRK
|
|
extern char *strpbrk(const char *, const char *);
|
|
#endif
|
|
#ifndef HAVE_STRSTR
|
|
extern char *strstr(const char *, const char *);
|
|
#endif
|
|
#endif
|
|
extern int is_prefix(const char *, const char *);
|
|
|
|
/* Conversion routines */
|
|
typedef enum {
|
|
MY_GCVT_ARG_FLOAT,
|
|
MY_GCVT_ARG_DOUBLE
|
|
} my_gcvt_arg_type;
|
|
|
|
double my_strtod(const char *str, char **end, int *error);
|
|
double my_atof(const char *nptr);
|
|
size_t my_fcvt(double x, int precision, char *to, my_bool *error);
|
|
size_t my_gcvt(double x, my_gcvt_arg_type type, int width, char *to,
|
|
my_bool *error);
|
|
|
|
#define NOT_FIXED_DEC 31
|
|
|
|
/*
|
|
The longest string my_fcvt can return is 311 + "precision" bytes.
|
|
Here we assume that we never cal my_fcvt() with precision >= NOT_FIXED_DEC
|
|
(+ 1 byte for the terminating '\0').
|
|
*/
|
|
#define FLOATING_POINT_BUFFER (311 + NOT_FIXED_DEC)
|
|
|
|
/*
|
|
We want to use the 'e' format in some cases even if we have enough space
|
|
for the 'f' one just to mimic sprintf("%.15g") behavior for large integers,
|
|
and to improve it for numbers < 10^(-4).
|
|
That is, for |x| < 1 we require |x| >= 10^(-15), and for |x| > 1 we require
|
|
it to be integer and be <= 10^DBL_DIG for the 'f' format to be used.
|
|
We don't lose precision, but make cases like "1e200" or "0.00001" look nicer.
|
|
*/
|
|
#define MAX_DECPT_FOR_F_FORMAT DBL_DIG
|
|
|
|
/*
|
|
The maximum possible field width for my_gcvt() conversion.
|
|
(DBL_DIG + 2) significant digits + sign + "." + ("e-NNN" or
|
|
MAX_DECPT_FOR_F_FORMAT zeros for cases when |x|<1 and the 'f' format is used).
|
|
*/
|
|
#define MY_GCVT_MAX_FIELD_WIDTH (DBL_DIG + 4 + max(5, MAX_DECPT_FOR_F_FORMAT)) \
|
|
|
|
extern char *llstr(longlong value,char *buff);
|
|
extern char *ullstr(longlong value,char *buff);
|
|
#ifndef HAVE_STRTOUL
|
|
extern long strtol(const char *str, char **ptr, int base);
|
|
extern ulong strtoul(const char *str, char **ptr, int base);
|
|
#endif
|
|
|
|
extern char *int2str(long val, char *dst, int radix, int upcase);
|
|
extern char *int10_to_str(long val,char *dst,int radix);
|
|
extern char *str2int(const char *src,int radix,long lower,long upper,
|
|
long *val);
|
|
longlong my_strtoll10(const char *nptr, char **endptr, int *error);
|
|
#if SIZEOF_LONG == SIZEOF_LONG_LONG
|
|
#define ll2str(A,B,C,D) int2str((A),(B),(C),(D))
|
|
#define longlong10_to_str(A,B,C) int10_to_str((A),(B),(C))
|
|
#undef strtoll
|
|
#define strtoll(A,B,C) strtol((A),(B),(C))
|
|
#define strtoull(A,B,C) strtoul((A),(B),(C))
|
|
#ifndef HAVE_STRTOULL
|
|
#define HAVE_STRTOULL
|
|
#endif
|
|
#ifndef HAVE_STRTOLL
|
|
#define HAVE_STRTOLL
|
|
#endif
|
|
#else
|
|
#ifdef HAVE_LONG_LONG
|
|
extern char *ll2str(longlong val,char *dst,int radix, int upcase);
|
|
extern char *longlong10_to_str(longlong val,char *dst,int radix);
|
|
#if (!defined(HAVE_STRTOULL) || defined(NO_STRTOLL_PROTO))
|
|
extern longlong strtoll(const char *str, char **ptr, int base);
|
|
extern ulonglong strtoull(const char *str, char **ptr, int base);
|
|
#endif
|
|
#endif
|
|
#endif
|
|
#define longlong2str(A,B,C) ll2str((A),(B),(C),1)
|
|
|
|
#if defined(__cplusplus)
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
LEX_STRING -- a pair of a C-string and its length.
|
|
(it's part of the plugin API as a MYSQL_LEX_STRING)
|
|
*/
|
|
|
|
#include <mysql/plugin.h>
|
|
typedef struct st_mysql_lex_string LEX_STRING;
|
|
|
|
#define STRING_WITH_LEN(X) (X), ((size_t) (sizeof(X) - 1))
|
|
#define USTRING_WITH_LEN(X) ((uchar*) X), ((size_t) (sizeof(X) - 1))
|
|
#define C_STRING_WITH_LEN(X) ((char *) (X)), ((size_t) (sizeof(X) - 1))
|
|
|
|
struct st_mysql_const_lex_string
|
|
{
|
|
const char *str;
|
|
size_t length;
|
|
};
|
|
typedef struct st_mysql_const_lex_string LEX_CSTRING;
|
|
|
|
/* SPACE_INT is a word that contains only spaces */
|
|
#if SIZEOF_INT == 4
|
|
#define SPACE_INT 0x20202020
|
|
#elif SIZEOF_INT == 8
|
|
#define SPACE_INT 0x2020202020202020
|
|
#else
|
|
#error define the appropriate constant for a word full of spaces
|
|
#endif
|
|
|
|
/**
|
|
Skip trailing space.
|
|
|
|
On most systems reading memory in larger chunks (ideally equal to the size of
|
|
the chinks that the machine physically reads from memory) causes fewer memory
|
|
access loops and hence increased performance.
|
|
This is why the 'int' type is used : it's closest to that (according to how
|
|
it's defined in C).
|
|
So when we determine the amount of whitespace at the end of a string we do
|
|
the following :
|
|
1. We divide the string into 3 zones :
|
|
a) from the start of the string (__start) to the first multiple
|
|
of sizeof(int) (__start_words)
|
|
b) from the end of the string (__end) to the last multiple of sizeof(int)
|
|
(__end_words)
|
|
c) a zone that is aligned to sizeof(int) and can be safely accessed
|
|
through an int *
|
|
2. We start comparing backwards from (c) char-by-char. If all we find is
|
|
space then we continue
|
|
3. If there are elements in zone (b) we compare them as unsigned ints to a
|
|
int mask (SPACE_INT) consisting of all spaces
|
|
4. Finally we compare the remaining part (a) of the string char by char.
|
|
This covers for the last non-space unsigned int from 3. (if any)
|
|
|
|
This algorithm works well for relatively larger strings, but it will slow
|
|
the things down for smaller strings (because of the additional calculations
|
|
and checks compared to the naive method). Thus the barrier of length 20
|
|
is added.
|
|
|
|
@param ptr pointer to the input string
|
|
@param len the length of the string
|
|
@return the last non-space character
|
|
*/
|
|
|
|
static inline const uchar *skip_trailing_space(const uchar *ptr,size_t len)
|
|
{
|
|
const uchar *end= ptr + len;
|
|
|
|
if (len > 20)
|
|
{
|
|
const uchar *end_words= (const uchar *)(intptr)
|
|
(((ulonglong)(intptr)end) / SIZEOF_INT * SIZEOF_INT);
|
|
const uchar *start_words= (const uchar *)(intptr)
|
|
((((ulonglong)(intptr)ptr) + SIZEOF_INT - 1) / SIZEOF_INT * SIZEOF_INT);
|
|
|
|
DBUG_ASSERT(((ulonglong)(intptr)ptr) >= SIZEOF_INT);
|
|
if (end_words > ptr)
|
|
{
|
|
while (end > end_words && end[-1] == 0x20)
|
|
end--;
|
|
if (end[-1] == 0x20 && start_words < end_words)
|
|
while (end > start_words && ((unsigned *)end)[-1] == SPACE_INT)
|
|
end -= SIZEOF_INT;
|
|
}
|
|
}
|
|
while (end > ptr && end[-1] == 0x20)
|
|
end--;
|
|
return (end);
|
|
}
|
|
|
|
static inline void lex_string_set(LEX_STRING *lex_str, const char *c_str)
|
|
{
|
|
lex_str->str= (char *) c_str;
|
|
lex_str->length= strlen(c_str);
|
|
}
|
|
|
|
#endif
|