mirror of
https://github.com/MariaDB/server.git
synced 2025-01-22 06:44:16 +01:00
5d4ed64c3e
Addresses #172 git-svn-id: file:///svn/tokudb@1082 c7de825b-a66e-492c-adef-691d508d4ae1
220 lines
7.3 KiB
C
220 lines
7.3 KiB
C
#if !defined(TOKUDB_COMMON_FUNCS_H)
|
|
#define TOKUDB_COMMON_FUNCS_H
|
|
|
|
/* -*- mode: C; c-basic-offset: 4 -*- */
|
|
#ident "Copyright (c) 2007 Tokutek Inc. All rights reserved."
|
|
|
|
#include "tokudb_common.h"
|
|
|
|
//DB_ENV->err disabled since it does not use db_strerror
|
|
#define ERROR(retval, ...) \
|
|
if (0) g.dbenv->err(g.dbenv, retval, __VA_ARGS__); \
|
|
else { \
|
|
fprintf(stderr, "%s: %s:", g.progname, db_strerror(retval)); \
|
|
fprintf(stderr, __VA_ARGS__); \
|
|
fprintf(stderr, "\n"); \
|
|
}
|
|
|
|
//DB_ENV->err disabled since it does not use db_strerror, errx does not exist.
|
|
#define ERRORX(...) \
|
|
if (0) g.dbenv->err(g.dbenv, 0, __VA_ARGS__); \
|
|
else { \
|
|
fprintf(stderr, "%s: ", g.progname); \
|
|
fprintf(stderr, __VA_ARGS__); \
|
|
fprintf(stderr, "\n"); \
|
|
}
|
|
|
|
int strtoint32 (char* str, int32_t* num, int32_t min, int32_t max, int base);
|
|
int strtouint32 (char* str, uint32_t* num, uint32_t min, uint32_t max, int base);
|
|
int strtoint64 (char* str, int64_t* num, int64_t min, int64_t max, int base);
|
|
int strtouint64 (char* str, uint64_t* num, uint64_t min, uint64_t max, int base);
|
|
|
|
/*
|
|
* Convert a string to an integer of type "type".
|
|
*
|
|
*
|
|
* Sets errno and returns:
|
|
* EINVAL: str == NULL, num == NULL, or string not of the form [ \t]*[+-]?[0-9]+
|
|
* ERANGE: value out of range specified. (Range of [min, max])
|
|
*
|
|
* *num is unchanged on error.
|
|
* Returns:
|
|
*
|
|
*/
|
|
#define DEF_STR_TO(name, type, bigtype, strtofunc, frmt) \
|
|
int name(char* str, type* num, type min, type max, int base) \
|
|
{ \
|
|
char* test; \
|
|
bigtype value; \
|
|
\
|
|
assert(str); \
|
|
assert(num); \
|
|
assert(min <= max); \
|
|
assert(g.dbenv || g.progname); \
|
|
assert(base == 0 || (base >= 2 && base <= 36)); \
|
|
\
|
|
errno = 0; \
|
|
while (isspace(*str)) str++; \
|
|
value = strtofunc(str, &test, base); \
|
|
if ((*test != '\0' && *test != '\n') || test == str) { \
|
|
ERRORX("%s: Invalid numeric argument\n", str); \
|
|
errno = EINVAL; \
|
|
goto error; \
|
|
} \
|
|
if (errno != 0) { \
|
|
ERROR(errno, "%s\n", str); \
|
|
} \
|
|
if (value < min) { \
|
|
ERRORX("%s: Less than minimum value (%" frmt ")\n", str, min); \
|
|
goto error; \
|
|
} \
|
|
if (value > max) { \
|
|
ERRORX("%s: Greater than maximum value (%" frmt ")\n", str, max); \
|
|
goto error; \
|
|
} \
|
|
*num = value; \
|
|
return EXIT_SUCCESS; \
|
|
error: \
|
|
return errno; \
|
|
}
|
|
|
|
DEF_STR_TO(strtoint32, int32_t, int64_t, strtoll, PRId32);
|
|
DEF_STR_TO(strtouint32, uint32_t, uint64_t, strtoull, PRIu32);
|
|
DEF_STR_TO(strtoint64, int64_t, int64_t, strtoll, PRId64);
|
|
DEF_STR_TO(strtouint64, uint64_t, uint64_t, strtoull, PRIu64);
|
|
|
|
void outputbyte(uint8_t ch)
|
|
{
|
|
if (g.plaintext) {
|
|
if (ch == '\\') printf("\\\\");
|
|
else if (isprint(ch)) printf("%c", ch);
|
|
else printf("\\%02x", ch);
|
|
}
|
|
else printf("%02x", ch);
|
|
}
|
|
|
|
void outputstring(char* str)
|
|
{
|
|
char* p;
|
|
|
|
for (p = str; *p != '\0'; p++) {
|
|
outputbyte((uint8_t)*p);
|
|
}
|
|
}
|
|
|
|
void outputplaintextstring(char* str)
|
|
{
|
|
bool old_plaintext = g.plaintext;
|
|
g.plaintext = true;
|
|
outputstring(str);
|
|
g.plaintext = old_plaintext;
|
|
}
|
|
|
|
int hextoint(int ch)
|
|
{
|
|
if (ch >= '0' && ch <= '9') {
|
|
return ch - '0';
|
|
}
|
|
if (ch >= 'a' && ch <= 'z') {
|
|
return ch - 'a' + 10;
|
|
}
|
|
if (ch >= 'A' && ch <= 'Z') {
|
|
return ch - 'A' + 10;
|
|
}
|
|
return EOF;
|
|
}
|
|
|
|
int printabletocstring(char* inputstr, char** poutputstr)
|
|
{
|
|
char highch;
|
|
char lowch;
|
|
char nextch;
|
|
char* cstring;
|
|
|
|
assert(inputstr);
|
|
assert(poutputstr);
|
|
assert(*poutputstr == NULL);
|
|
|
|
cstring = (char*)malloc((strlen(inputstr) + 1) * sizeof(char));
|
|
if (cstring == NULL) {
|
|
ERROR(errno, "printabletocstring");
|
|
goto error;
|
|
}
|
|
|
|
for (*poutputstr = cstring; *inputstr != '\0'; inputstr++) {
|
|
if (*inputstr == '\\') {
|
|
if ((highch = *++inputstr) == '\\') {
|
|
*cstring++ = '\\';
|
|
continue;
|
|
}
|
|
if (highch == '\0' || (lowch = *++inputstr) == '\0') {
|
|
ERROR(0, "unexpected end of input data or key/data pair");
|
|
goto error;
|
|
}
|
|
if (!isxdigit(highch)) {
|
|
ERROR(0, "Unexpected '%c' (non-hex) input.\n", highch);
|
|
goto error;
|
|
}
|
|
if (!isxdigit(lowch)) {
|
|
ERROR(0, "Unexpected '%c' (non-hex) input.\n", lowch);
|
|
goto error;
|
|
}
|
|
nextch = (hextoint(highch) << 4) | hextoint(lowch);
|
|
if (nextch == '\0') {
|
|
/* Database names are c strings, and cannot have extra NULL terminators. */
|
|
ERROR(0, "Unexpected '\\00' in input.\n");
|
|
goto error;
|
|
}
|
|
*cstring++ = nextch;
|
|
}
|
|
else *cstring++ = *inputstr;
|
|
}
|
|
/* Terminate the string. */
|
|
*cstring = '\0';
|
|
return EXIT_SUCCESS;
|
|
|
|
error:
|
|
ERROR(0, "Quitting out due to errors.\n");
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
int verify_library_version()
|
|
{
|
|
int major;
|
|
int minor;
|
|
|
|
db_version(&major, &minor, NULL);
|
|
if (major != DB_VERSION_MAJOR || minor != DB_VERSION_MINOR) {
|
|
ERRORX("version %d.%d doesn't match library version %d.%d\n",
|
|
DB_VERSION_MAJOR, DB_VERSION_MINOR, major, minor);
|
|
return EXIT_FAILURE;
|
|
}
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
static int last_caught = 0;
|
|
|
|
static void catch_signal(int signal) {
|
|
last_caught = signal;
|
|
if (last_caught == 0) last_caught = SIGINT;
|
|
}
|
|
|
|
void init_catch_signals(void) {
|
|
signal(SIGHUP, catch_signal);
|
|
signal(SIGINT, catch_signal);
|
|
signal(SIGPIPE, catch_signal);
|
|
signal(SIGTERM, catch_signal);
|
|
}
|
|
|
|
int caught_any_signals(void) {
|
|
return last_caught != 0;
|
|
}
|
|
|
|
void resend_signals(void) {
|
|
if (last_caught) {
|
|
signal(last_caught, SIG_DFL);
|
|
raise(last_caught);
|
|
}
|
|
}
|
|
|
|
#endif /* #if !defined(TOKUDB_COMMON_H) */
|