#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) */