2012-11-08 12:25:23 -05:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2009-2012, Pieter Noordhuis <pcnoordhuis at gmail dot com>
|
|
|
|
* Copyright (c) 2009-2012, Salvatore Sanfilippo <antirez at gmail dot com>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* * Neither the name of Redis nor the names of its contributors may be used
|
|
|
|
* to endorse or promote products derived from this software without
|
|
|
|
* specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2015-07-26 09:14:57 -04:00
|
|
|
#include "server.h"
|
2014-05-12 12:12:48 -04:00
|
|
|
#include "rdb.h"
|
2010-03-13 09:55:42 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include "lzf.h"
|
2012-11-29 08:20:08 -05:00
|
|
|
#include "crc64.h"
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
#define ERROR(...) { \
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, __VA_ARGS__); \
|
2010-03-13 09:55:42 -05:00
|
|
|
exit(1); \
|
|
|
|
}
|
|
|
|
|
|
|
|
/* data type to hold offset in file and size */
|
|
|
|
typedef struct {
|
|
|
|
void *data;
|
2010-08-31 07:06:26 -04:00
|
|
|
size_t size;
|
|
|
|
size_t offset;
|
2010-03-13 09:55:42 -05:00
|
|
|
} pos;
|
|
|
|
|
|
|
|
static unsigned char level = 0;
|
|
|
|
static pos positions[16];
|
|
|
|
|
|
|
|
#define CURR_OFFSET (positions[level].offset)
|
|
|
|
|
|
|
|
/* Hold a stack of errors */
|
|
|
|
typedef struct {
|
|
|
|
char error[16][1024];
|
2010-08-31 07:06:26 -04:00
|
|
|
size_t offset[16];
|
|
|
|
size_t level;
|
2010-03-13 09:55:42 -05:00
|
|
|
} errors_t;
|
|
|
|
static errors_t errors;
|
|
|
|
|
|
|
|
#define SHIFT_ERROR(provided_offset, ...) { \
|
|
|
|
sprintf(errors.error[errors.level], __VA_ARGS__); \
|
|
|
|
errors.offset[errors.level] = provided_offset; \
|
|
|
|
errors.level++; \
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Data type to hold opcode with optional key name an success status */
|
|
|
|
typedef struct {
|
|
|
|
char* key;
|
|
|
|
int type;
|
|
|
|
char success;
|
|
|
|
} entry;
|
|
|
|
|
2014-04-13 14:09:54 -04:00
|
|
|
#define MAX_TYPES_NUM 256
|
|
|
|
#define MAX_TYPE_NAME_LEN 16
|
2010-03-13 09:55:42 -05:00
|
|
|
/* store string types for output */
|
2014-04-13 14:09:54 -04:00
|
|
|
static char types[MAX_TYPES_NUM][MAX_TYPE_NAME_LEN];
|
2010-03-13 09:55:42 -05:00
|
|
|
|
2012-10-22 05:44:20 -04:00
|
|
|
/* Return true if 't' is a valid object type. */
|
2014-05-12 12:12:48 -04:00
|
|
|
static int rdbCheckType(unsigned char t) {
|
2014-06-26 12:48:40 -04:00
|
|
|
/* In case a new object type is added, update the following
|
2012-10-22 05:44:20 -04:00
|
|
|
* condition as necessary. */
|
|
|
|
return
|
2015-07-27 03:41:48 -04:00
|
|
|
(t >= RDB_TYPE_HASH_ZIPMAP && t <= RDB_TYPE_HASH_ZIPLIST) ||
|
|
|
|
t <= RDB_TYPE_HASH ||
|
|
|
|
t >= RDB_OPCODE_EXPIRETIME_MS;
|
2012-10-22 05:44:20 -04:00
|
|
|
}
|
|
|
|
|
2010-03-13 09:55:42 -05:00
|
|
|
/* when number of bytes to read is negative, do a peek */
|
2014-05-09 12:06:06 -04:00
|
|
|
static int readBytes(void *target, long num) {
|
2010-03-13 09:55:42 -05:00
|
|
|
char peek = (num < 0) ? 1 : 0;
|
|
|
|
num = (num < 0) ? -num : num;
|
|
|
|
|
|
|
|
pos p = positions[level];
|
|
|
|
if (p.offset + num > p.size) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
2010-08-31 07:06:26 -04:00
|
|
|
memcpy(target, (void*)((size_t)p.data + p.offset), num);
|
2010-03-13 09:55:42 -05:00
|
|
|
if (!peek) positions[level].offset += num;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-03-15 19:51:54 -04:00
|
|
|
int processHeader(void) {
|
2010-03-13 09:55:42 -05:00
|
|
|
char buf[10] = "_________";
|
|
|
|
int dump_version;
|
|
|
|
|
|
|
|
if (!readBytes(buf, 9)) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Cannot read header");
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* expect the first 5 bytes to equal REDIS */
|
|
|
|
if (memcmp(buf,"REDIS",5) != 0) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Wrong signature in header");
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
dump_version = (int)strtol(buf + 5, NULL, 10);
|
2012-04-24 13:05:27 -04:00
|
|
|
if (dump_version < 1 || dump_version > 6) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Unknown RDB format version: %d", dump_version);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
2012-04-24 13:05:27 -04:00
|
|
|
return dump_version;
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static int loadType(entry *e) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint32_t offset = CURR_OFFSET;
|
|
|
|
|
|
|
|
/* this byte needs to qualify as type */
|
|
|
|
unsigned char t;
|
|
|
|
if (readBytes(&t, 1)) {
|
2014-05-12 12:12:48 -04:00
|
|
|
if (rdbCheckType(t)) {
|
2010-03-13 09:55:42 -05:00
|
|
|
e->type = t;
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
SHIFT_ERROR(offset, "Unknown type (0x%02x)", t);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SHIFT_ERROR(offset, "Could not read type");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* failure */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static int peekType() {
|
2010-03-13 09:55:42 -05:00
|
|
|
unsigned char t;
|
2014-05-12 12:12:48 -04:00
|
|
|
if (readBytes(&t, -1) && (rdbCheckType(t)))
|
2011-10-14 10:52:11 -04:00
|
|
|
return t;
|
2010-03-13 09:55:42 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* discard time, just consume the bytes */
|
2014-05-09 12:06:06 -04:00
|
|
|
static int processTime(int type) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint32_t offset = CURR_OFFSET;
|
2012-10-22 05:44:20 -04:00
|
|
|
unsigned char t[8];
|
2015-07-27 03:41:48 -04:00
|
|
|
int timelen = (type == RDB_OPCODE_EXPIRETIME_MS) ? 8 : 4;
|
2012-10-22 05:44:20 -04:00
|
|
|
|
|
|
|
if (readBytes(t,timelen)) {
|
2010-03-13 09:55:42 -05:00
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
SHIFT_ERROR(offset, "Could not read time");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* failure */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-06-01 14:18:28 -04:00
|
|
|
static uint64_t loadLength(int *isencoded) {
|
2010-03-13 09:55:42 -05:00
|
|
|
unsigned char buf[2];
|
|
|
|
uint32_t len;
|
|
|
|
int type;
|
|
|
|
|
|
|
|
if (isencoded) *isencoded = 0;
|
2015-07-27 03:41:48 -04:00
|
|
|
if (!readBytes(buf, 1)) return RDB_LENERR;
|
2010-03-13 09:55:42 -05:00
|
|
|
type = (buf[0] & 0xC0) >> 6;
|
2015-07-27 03:41:48 -04:00
|
|
|
if (type == RDB_6BITLEN) {
|
2010-03-13 09:55:42 -05:00
|
|
|
/* Read a 6 bit len */
|
|
|
|
return buf[0] & 0x3F;
|
2015-07-27 03:41:48 -04:00
|
|
|
} else if (type == RDB_ENCVAL) {
|
2010-03-13 09:55:42 -05:00
|
|
|
/* Read a 6 bit len encoding type */
|
|
|
|
if (isencoded) *isencoded = 1;
|
|
|
|
return buf[0] & 0x3F;
|
2015-07-27 03:41:48 -04:00
|
|
|
} else if (type == RDB_14BITLEN) {
|
2010-03-13 09:55:42 -05:00
|
|
|
/* Read a 14 bit len */
|
2015-07-27 03:41:48 -04:00
|
|
|
if (!readBytes(buf+1,1)) return RDB_LENERR;
|
2010-03-13 09:55:42 -05:00
|
|
|
return ((buf[0] & 0x3F) << 8) | buf[1];
|
2016-06-01 14:18:28 -04:00
|
|
|
} else if (buf[0] == RDB_32BITLEN) {
|
2010-03-13 09:55:42 -05:00
|
|
|
/* Read a 32 bit len */
|
2015-07-27 03:41:48 -04:00
|
|
|
if (!readBytes(&len, 4)) return RDB_LENERR;
|
2016-06-01 14:18:28 -04:00
|
|
|
return ntohl(len);
|
|
|
|
} else if (buf[0] == RDB_64BITLEN) {
|
|
|
|
/* Read a 64 bit len */
|
|
|
|
if (!readBytes(&len, 8)) return RDB_LENERR;
|
|
|
|
return ntohu64(len);
|
|
|
|
} else {
|
|
|
|
return RDB_LENERR;
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static char *loadIntegerObject(int enctype) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint32_t offset = CURR_OFFSET;
|
|
|
|
unsigned char enc[4];
|
|
|
|
long long val;
|
|
|
|
|
2015-07-27 03:41:48 -04:00
|
|
|
if (enctype == RDB_ENC_INT8) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint8_t v;
|
|
|
|
if (!readBytes(enc, 1)) return NULL;
|
|
|
|
v = enc[0];
|
|
|
|
val = (int8_t)v;
|
2015-07-27 03:41:48 -04:00
|
|
|
} else if (enctype == RDB_ENC_INT16) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint16_t v;
|
|
|
|
if (!readBytes(enc, 2)) return NULL;
|
|
|
|
v = enc[0]|(enc[1]<<8);
|
|
|
|
val = (int16_t)v;
|
2015-07-27 03:41:48 -04:00
|
|
|
} else if (enctype == RDB_ENC_INT32) {
|
2010-03-13 09:55:42 -05:00
|
|
|
uint32_t v;
|
|
|
|
if (!readBytes(enc, 4)) return NULL;
|
|
|
|
v = enc[0]|(enc[1]<<8)|(enc[2]<<16)|(enc[3]<<24);
|
|
|
|
val = (int32_t)v;
|
|
|
|
} else {
|
|
|
|
SHIFT_ERROR(offset, "Unknown integer encoding (0x%02x)", enctype);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convert val into string */
|
|
|
|
char *buf;
|
2014-05-12 12:12:48 -04:00
|
|
|
buf = zmalloc(sizeof(char) * 128);
|
2010-03-13 09:55:42 -05:00
|
|
|
sprintf(buf, "%lld", val);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static char* loadLzfStringObject() {
|
2016-06-01 14:18:28 -04:00
|
|
|
uint64_t slen, clen;
|
2010-03-13 09:55:42 -05:00
|
|
|
char *c, *s;
|
|
|
|
|
2015-07-27 03:41:48 -04:00
|
|
|
if ((clen = loadLength(NULL)) == RDB_LENERR) return NULL;
|
|
|
|
if ((slen = loadLength(NULL)) == RDB_LENERR) return NULL;
|
2010-03-13 09:55:42 -05:00
|
|
|
|
2014-05-12 12:12:48 -04:00
|
|
|
c = zmalloc(clen);
|
2010-03-13 09:55:42 -05:00
|
|
|
if (!readBytes(c, clen)) {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(c);
|
2010-03-13 09:55:42 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-05-12 12:12:48 -04:00
|
|
|
s = zmalloc(slen+1);
|
2010-03-13 09:55:42 -05:00
|
|
|
if (lzf_decompress(c,clen,s,slen) == 0) {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(c); zfree(s);
|
2010-03-13 09:55:42 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(c);
|
2010-03-13 09:55:42 -05:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* returns NULL when not processable, char* when valid */
|
2014-05-09 12:06:06 -04:00
|
|
|
static char* loadStringObject() {
|
2016-06-01 14:18:28 -04:00
|
|
|
uint64_t offset = CURR_OFFSET;
|
|
|
|
uint64_t len;
|
2010-03-13 09:55:42 -05:00
|
|
|
int isencoded;
|
|
|
|
|
|
|
|
len = loadLength(&isencoded);
|
|
|
|
if (isencoded) {
|
|
|
|
switch(len) {
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_ENC_INT8:
|
|
|
|
case RDB_ENC_INT16:
|
|
|
|
case RDB_ENC_INT32:
|
2010-03-13 09:55:42 -05:00
|
|
|
return loadIntegerObject(len);
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_ENC_LZF:
|
2010-03-13 09:55:42 -05:00
|
|
|
return loadLzfStringObject();
|
|
|
|
default:
|
|
|
|
/* unknown encoding */
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Unknown string encoding (0x%02llx)", len);
|
2010-03-13 09:55:42 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-27 03:41:48 -04:00
|
|
|
if (len == RDB_LENERR) return NULL;
|
2010-03-13 09:55:42 -05:00
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
char *buf = zmalloc(sizeof(char) * (len+1));
|
2013-07-11 07:45:15 -04:00
|
|
|
if (buf == NULL) return NULL;
|
2010-03-13 09:55:42 -05:00
|
|
|
buf[len] = '\0';
|
|
|
|
if (!readBytes(buf, len)) {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(buf);
|
2010-03-13 09:55:42 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static int processStringObject(char** store) {
|
2010-03-13 09:55:42 -05:00
|
|
|
unsigned long offset = CURR_OFFSET;
|
|
|
|
char *key = loadStringObject();
|
|
|
|
if (key == NULL) {
|
|
|
|
SHIFT_ERROR(offset, "Error reading string object");
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(key);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (store != NULL) {
|
|
|
|
*store = key;
|
|
|
|
} else {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(key);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static double* loadDoubleValue() {
|
2010-03-13 09:55:42 -05:00
|
|
|
char buf[256];
|
|
|
|
unsigned char len;
|
|
|
|
double* val;
|
|
|
|
|
|
|
|
if (!readBytes(&len,1)) return NULL;
|
|
|
|
|
2014-05-12 12:12:48 -04:00
|
|
|
val = zmalloc(sizeof(double));
|
2010-03-13 09:55:42 -05:00
|
|
|
switch(len) {
|
|
|
|
case 255: *val = R_NegInf; return val;
|
|
|
|
case 254: *val = R_PosInf; return val;
|
|
|
|
case 253: *val = R_Nan; return val;
|
|
|
|
default:
|
|
|
|
if (!readBytes(buf, len)) {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(val);
|
2010-03-13 09:55:42 -05:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
buf[len] = '\0';
|
|
|
|
sscanf(buf, "%lg", val);
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static int processDoubleValue(double** store) {
|
2010-03-13 09:55:42 -05:00
|
|
|
unsigned long offset = CURR_OFFSET;
|
|
|
|
double *val = loadDoubleValue();
|
|
|
|
if (val == NULL) {
|
|
|
|
SHIFT_ERROR(offset, "Error reading double value");
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(val);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (store != NULL) {
|
|
|
|
*store = val;
|
|
|
|
} else {
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(val);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static int loadPair(entry *e) {
|
2016-06-01 14:18:28 -04:00
|
|
|
uint64_t offset = CURR_OFFSET;
|
|
|
|
uint64_t i;
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
/* read key first */
|
|
|
|
char *key;
|
|
|
|
if (processStringObject(&key)) {
|
|
|
|
e->key = key;
|
|
|
|
} else {
|
|
|
|
SHIFT_ERROR(offset, "Error reading entry key");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-06-01 14:18:28 -04:00
|
|
|
uint64_t length = 0;
|
2015-07-27 03:41:48 -04:00
|
|
|
if (e->type == RDB_TYPE_LIST ||
|
|
|
|
e->type == RDB_TYPE_SET ||
|
|
|
|
e->type == RDB_TYPE_ZSET ||
|
|
|
|
e->type == RDB_TYPE_HASH) {
|
|
|
|
if ((length = loadLength(NULL)) == RDB_LENERR) {
|
2010-03-13 09:55:42 -05:00
|
|
|
SHIFT_ERROR(offset, "Error reading %s length", types[e->type]);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(e->type) {
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_TYPE_STRING:
|
|
|
|
case RDB_TYPE_HASH_ZIPMAP:
|
|
|
|
case RDB_TYPE_LIST_ZIPLIST:
|
|
|
|
case RDB_TYPE_SET_INTSET:
|
|
|
|
case RDB_TYPE_ZSET_ZIPLIST:
|
|
|
|
case RDB_TYPE_HASH_ZIPLIST:
|
2010-03-13 09:55:42 -05:00
|
|
|
if (!processStringObject(NULL)) {
|
|
|
|
SHIFT_ERROR(offset, "Error reading entry value");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_TYPE_LIST:
|
|
|
|
case RDB_TYPE_SET:
|
2010-03-13 09:55:42 -05:00
|
|
|
for (i = 0; i < length; i++) {
|
|
|
|
offset = CURR_OFFSET;
|
|
|
|
if (!processStringObject(NULL)) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Error reading element at index %llu (length: %llu)", i, length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_TYPE_ZSET:
|
2010-03-13 09:55:42 -05:00
|
|
|
for (i = 0; i < length; i++) {
|
|
|
|
offset = CURR_OFFSET;
|
|
|
|
if (!processStringObject(NULL)) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Error reading element key at index %llu (length: %llu)", i, length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
offset = CURR_OFFSET;
|
|
|
|
if (!processDoubleValue(NULL)) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Error reading element value at index %llu (length: %llu)", i, length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2015-07-27 03:41:48 -04:00
|
|
|
case RDB_TYPE_HASH:
|
2010-03-13 09:55:42 -05:00
|
|
|
for (i = 0; i < length; i++) {
|
|
|
|
offset = CURR_OFFSET;
|
|
|
|
if (!processStringObject(NULL)) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Error reading element key at index %llu (length: %llu)", i, length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
offset = CURR_OFFSET;
|
|
|
|
if (!processStringObject(NULL)) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset, "Error reading element value at index %llu (length: %llu)", i, length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SHIFT_ERROR(offset, "Type not implemented");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* because we're done, we assume success */
|
|
|
|
e->success = 1;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static entry loadEntry() {
|
2010-03-13 09:55:42 -05:00
|
|
|
entry e = { NULL, -1, 0 };
|
2016-06-01 14:18:28 -04:00
|
|
|
uint64_t length, offset[4];
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
/* reset error container */
|
|
|
|
errors.level = 0;
|
|
|
|
|
|
|
|
offset[0] = CURR_OFFSET;
|
|
|
|
if (!loadType(&e)) {
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
offset[1] = CURR_OFFSET;
|
2015-07-27 03:41:48 -04:00
|
|
|
if (e.type == RDB_OPCODE_SELECTDB) {
|
|
|
|
if ((length = loadLength(NULL)) == RDB_LENERR) {
|
2010-03-13 09:55:42 -05:00
|
|
|
SHIFT_ERROR(offset[1], "Error reading database number");
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
if (length > 63) {
|
2016-06-01 14:18:28 -04:00
|
|
|
SHIFT_ERROR(offset[1], "Database number out of range (%llu)", length);
|
2010-03-13 09:55:42 -05:00
|
|
|
return e;
|
|
|
|
}
|
2015-07-27 03:41:48 -04:00
|
|
|
} else if (e.type == RDB_OPCODE_EOF) {
|
2010-03-13 09:55:42 -05:00
|
|
|
if (positions[level].offset < positions[level].size) {
|
|
|
|
SHIFT_ERROR(offset[0], "Unexpected EOF");
|
|
|
|
} else {
|
|
|
|
e.success = 1;
|
|
|
|
}
|
|
|
|
return e;
|
|
|
|
} else {
|
|
|
|
/* optionally consume expire */
|
2015-07-27 03:41:48 -04:00
|
|
|
if (e.type == RDB_OPCODE_EXPIRETIME ||
|
|
|
|
e.type == RDB_OPCODE_EXPIRETIME_MS) {
|
2012-10-22 05:44:20 -04:00
|
|
|
if (!processTime(e.type)) return e;
|
2010-03-13 09:55:42 -05:00
|
|
|
if (!loadType(&e)) return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
offset[1] = CURR_OFFSET;
|
|
|
|
if (!loadPair(&e)) {
|
|
|
|
SHIFT_ERROR(offset[1], "Error for type %s", types[e.type]);
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* all entries are followed by a valid type:
|
|
|
|
* e.g. a new entry, SELECTDB, EXPIRE, EOF */
|
|
|
|
offset[2] = CURR_OFFSET;
|
|
|
|
if (peekType() == -1) {
|
|
|
|
SHIFT_ERROR(offset[2], "Followed by invalid type");
|
|
|
|
SHIFT_ERROR(offset[0], "Error for type %s", types[e.type]);
|
|
|
|
e.success = 0;
|
|
|
|
} else {
|
|
|
|
e.success = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static void printCentered(int indent, int width, char* body) {
|
2010-03-13 09:55:42 -05:00
|
|
|
char head[256], tail[256];
|
|
|
|
memset(head, '\0', 256);
|
|
|
|
memset(tail, '\0', 256);
|
|
|
|
|
|
|
|
memset(head, '=', indent);
|
|
|
|
memset(tail, '=', width - 2 - indent - strlen(body));
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "%s %s %s", head, body, tail);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static void printValid(uint64_t ops, uint64_t bytes) {
|
2010-03-13 09:55:42 -05:00
|
|
|
char body[80];
|
2010-09-01 12:31:30 -04:00
|
|
|
sprintf(body, "Processed %llu valid opcodes (in %llu bytes)",
|
|
|
|
(unsigned long long) ops, (unsigned long long) bytes);
|
2010-03-13 09:55:42 -05:00
|
|
|
printCentered(4, 80, body);
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static void printSkipped(uint64_t bytes, uint64_t offset) {
|
2010-03-13 09:55:42 -05:00
|
|
|
char body[80];
|
2010-09-01 12:31:30 -04:00
|
|
|
sprintf(body, "Skipped %llu bytes (resuming at 0x%08llx)",
|
|
|
|
(unsigned long long) bytes, (unsigned long long) offset);
|
2010-03-13 09:55:42 -05:00
|
|
|
printCentered(4, 80, body);
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
static void printErrorStack(entry *e) {
|
2010-03-13 09:55:42 -05:00
|
|
|
unsigned int i;
|
|
|
|
char body[64];
|
|
|
|
|
|
|
|
if (e->type == -1) {
|
|
|
|
sprintf(body, "Error trace");
|
|
|
|
} else if (e->type >= 253) {
|
|
|
|
sprintf(body, "Error trace (%s)", types[e->type]);
|
|
|
|
} else if (!e->key) {
|
|
|
|
sprintf(body, "Error trace (%s: (unknown))", types[e->type]);
|
|
|
|
} else {
|
|
|
|
char tmp[41];
|
|
|
|
strncpy(tmp, e->key, 40);
|
|
|
|
|
|
|
|
/* display truncation at the last 3 chars */
|
|
|
|
if (strlen(e->key) > 40) {
|
|
|
|
memset(&tmp[37], '.', 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* display unprintable characters as ? */
|
|
|
|
for (i = 0; i < strlen(tmp); i++) {
|
|
|
|
if (tmp[i] <= 32) tmp[i] = '?';
|
|
|
|
}
|
|
|
|
sprintf(body, "Error trace (%s: %s)", types[e->type], tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
printCentered(4, 80, body);
|
|
|
|
|
|
|
|
/* display error stack */
|
|
|
|
for (i = 0; i < errors.level; i++) {
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "0x%08lx - %s",
|
2010-11-02 06:15:09 -04:00
|
|
|
(unsigned long) errors.offset[i], errors.error[i]);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-15 19:51:54 -04:00
|
|
|
void process(void) {
|
2010-08-31 04:21:35 -04:00
|
|
|
uint64_t num_errors = 0, num_valid_ops = 0, num_valid_bytes = 0;
|
2015-01-30 09:19:39 -05:00
|
|
|
entry entry = { NULL, -1, 0 };
|
2012-04-24 13:05:27 -04:00
|
|
|
int dump_version = processHeader();
|
|
|
|
|
|
|
|
/* Exclude the final checksum for RDB >= 5. Will be checked at the end. */
|
|
|
|
if (dump_version >= 5) {
|
|
|
|
if (positions[0].size < 8) {
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "RDB version >= 5 but no room for checksum.");
|
2012-04-24 13:05:27 -04:00
|
|
|
exit(1);
|
|
|
|
}
|
2014-08-01 15:42:50 -04:00
|
|
|
positions[0].size -= 8;
|
2012-04-24 13:05:27 -04:00
|
|
|
}
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
level = 1;
|
|
|
|
while(positions[0].offset < positions[0].size) {
|
|
|
|
positions[1] = positions[0];
|
|
|
|
|
|
|
|
entry = loadEntry();
|
|
|
|
if (!entry.success) {
|
|
|
|
printValid(num_valid_ops, num_valid_bytes);
|
|
|
|
printErrorStack(&entry);
|
|
|
|
num_errors++;
|
|
|
|
num_valid_ops = 0;
|
|
|
|
num_valid_bytes = 0;
|
|
|
|
|
|
|
|
/* search for next valid entry */
|
2010-08-31 04:21:35 -04:00
|
|
|
uint64_t offset = positions[0].offset + 1;
|
|
|
|
int i = 0;
|
|
|
|
|
2010-03-13 09:55:42 -05:00
|
|
|
while (!entry.success && offset < positions[0].size) {
|
|
|
|
positions[1].offset = offset;
|
|
|
|
|
|
|
|
/* find 3 consecutive valid entries */
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
entry = loadEntry();
|
|
|
|
if (!entry.success) break;
|
|
|
|
}
|
|
|
|
/* check if we found 3 consecutive valid entries */
|
|
|
|
if (i < 3) {
|
|
|
|
offset++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* print how many bytes we have skipped to find a new valid opcode */
|
|
|
|
if (offset < positions[0].size) {
|
|
|
|
printSkipped(offset - positions[0].offset, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
positions[0].offset = offset;
|
|
|
|
} else {
|
|
|
|
num_valid_ops++;
|
|
|
|
num_valid_bytes += positions[1].offset - positions[0].offset;
|
|
|
|
|
|
|
|
/* advance position */
|
|
|
|
positions[0] = positions[1];
|
|
|
|
}
|
2014-05-12 12:12:48 -04:00
|
|
|
zfree(entry.key);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* because there is another potential error,
|
|
|
|
* print how many valid ops we have processed */
|
|
|
|
printValid(num_valid_ops, num_valid_bytes);
|
|
|
|
|
|
|
|
/* expect an eof */
|
2015-07-27 03:41:48 -04:00
|
|
|
if (entry.type != RDB_OPCODE_EOF) {
|
2010-03-13 09:55:42 -05:00
|
|
|
/* last byte should be EOF, add error */
|
|
|
|
errors.level = 0;
|
|
|
|
SHIFT_ERROR(positions[0].offset, "Expected EOF, got %s", types[entry.type]);
|
|
|
|
|
|
|
|
/* this is an EOF error so reset type */
|
|
|
|
entry.type = -1;
|
|
|
|
printErrorStack(&entry);
|
|
|
|
|
|
|
|
num_errors++;
|
|
|
|
}
|
|
|
|
|
2012-04-24 13:05:27 -04:00
|
|
|
/* Verify checksum */
|
|
|
|
if (dump_version >= 5) {
|
|
|
|
uint64_t crc = crc64(0,positions[0].data,positions[0].size);
|
|
|
|
uint64_t crc2;
|
|
|
|
unsigned char *p = (unsigned char*)positions[0].data+positions[0].size;
|
|
|
|
crc2 = ((uint64_t)p[0] << 0) |
|
|
|
|
((uint64_t)p[1] << 8) |
|
|
|
|
((uint64_t)p[2] << 16) |
|
|
|
|
((uint64_t)p[3] << 24) |
|
|
|
|
((uint64_t)p[4] << 32) |
|
|
|
|
((uint64_t)p[5] << 40) |
|
|
|
|
((uint64_t)p[6] << 48) |
|
|
|
|
((uint64_t)p[7] << 56);
|
|
|
|
if (crc != crc2) {
|
|
|
|
SHIFT_ERROR(positions[0].offset, "RDB CRC64 does not match.");
|
|
|
|
} else {
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "CRC64 checksum is OK");
|
2012-04-24 13:05:27 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-13 09:55:42 -05:00
|
|
|
/* print summary on errors */
|
2010-08-31 04:21:35 -04:00
|
|
|
if (num_errors) {
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "Total unprocessable opcodes: %llu",
|
2010-09-01 12:31:30 -04:00
|
|
|
(unsigned long long) num_errors);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
int redis_check_rdb(char *rdbfilename) {
|
2010-03-13 09:55:42 -05:00
|
|
|
int fd;
|
2010-08-31 07:06:26 -04:00
|
|
|
off_t size;
|
2010-03-13 09:55:42 -05:00
|
|
|
struct stat stat;
|
|
|
|
void *data;
|
|
|
|
|
2014-05-09 12:06:06 -04:00
|
|
|
fd = open(rdbfilename, O_RDONLY);
|
2010-03-13 09:55:42 -05:00
|
|
|
if (fd < 1) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Cannot open file: %s", rdbfilename);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
if (fstat(fd, &stat) == -1) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Cannot stat: %s", rdbfilename);
|
2010-03-13 09:55:42 -05:00
|
|
|
} else {
|
|
|
|
size = stat.st_size;
|
|
|
|
}
|
|
|
|
|
2010-08-31 07:06:26 -04:00
|
|
|
if (sizeof(size_t) == sizeof(int32_t) && size >= INT_MAX) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Cannot check dump files >2GB on a 32-bit platform");
|
2010-08-31 07:06:26 -04:00
|
|
|
}
|
|
|
|
|
2010-03-13 09:55:42 -05:00
|
|
|
data = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0);
|
|
|
|
if (data == MAP_FAILED) {
|
2014-05-12 12:12:48 -04:00
|
|
|
ERROR("Cannot mmap: %s", rdbfilename);
|
2010-03-13 09:55:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize static vars */
|
|
|
|
positions[0].data = data;
|
|
|
|
positions[0].size = size;
|
|
|
|
positions[0].offset = 0;
|
|
|
|
errors.level = 0;
|
|
|
|
|
|
|
|
/* Object types */
|
2015-07-27 03:41:48 -04:00
|
|
|
sprintf(types[RDB_TYPE_STRING], "STRING");
|
|
|
|
sprintf(types[RDB_TYPE_LIST], "LIST");
|
|
|
|
sprintf(types[RDB_TYPE_SET], "SET");
|
|
|
|
sprintf(types[RDB_TYPE_ZSET], "ZSET");
|
|
|
|
sprintf(types[RDB_TYPE_HASH], "HASH");
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
/* Object types only used for dumping to disk */
|
2015-07-27 03:41:48 -04:00
|
|
|
sprintf(types[RDB_OPCODE_EXPIRETIME], "EXPIRETIME");
|
|
|
|
sprintf(types[RDB_OPCODE_SELECTDB], "SELECTDB");
|
|
|
|
sprintf(types[RDB_OPCODE_EOF], "EOF");
|
2010-03-13 09:55:42 -05:00
|
|
|
|
|
|
|
process();
|
|
|
|
|
|
|
|
munmap(data, size);
|
|
|
|
close(fd);
|
|
|
|
return 0;
|
|
|
|
}
|
2015-02-03 04:25:01 -05:00
|
|
|
|
|
|
|
/* RDB check main: called form redis.c when Redis is executed with the
|
|
|
|
* redis-check-rdb alias. */
|
|
|
|
int redis_check_rdb_main(char **argv, int argc) {
|
|
|
|
if (argc != 2) {
|
|
|
|
fprintf(stderr, "Usage: %s <rdb-file-name>\n", argv[0]);
|
|
|
|
exit(1);
|
|
|
|
}
|
2015-07-27 03:41:48 -04:00
|
|
|
serverLog(LL_WARNING, "Checking RDB file %s", argv[1]);
|
2015-02-03 04:25:01 -05:00
|
|
|
exit(redis_check_rdb(argv[1]));
|
|
|
|
return 0;
|
|
|
|
}
|