]>
git.saurik.com Git - redis.git/blob - src/debug.c
b364dd1635ad9acef4569d486ef77d331aa5f967
   2 #include "sha1.h"   /* SHA1 is used for DEBUG DIGEST */ 
   6 /* ================================= Debugging ============================== */ 
   8 /* Compute the sha1 of string at 's' with 'len' bytes long. 
   9  * The SHA1 is then xored againt the string pointed by digest. 
  10  * Since xor is commutative, this operation is used in order to 
  11  * "add" digests relative to unordered elements. 
  13  * So digest(a,b,c,d) will be the same of digest(b,a,c,d) */ 
  14 void xorDigest(unsigned char *digest
, void *ptr
, size_t len
) { 
  16     unsigned char hash
[20], *s 
= ptr
; 
  20     SHA1Update(&ctx
,s
,len
); 
  23     for (j 
= 0; j 
< 20; j
++) 
  27 void xorObjectDigest(unsigned char *digest
, robj 
*o
) { 
  28     o 
= getDecodedObject(o
); 
  29     xorDigest(digest
,o
->ptr
,sdslen(o
->ptr
)); 
  33 /* This function instead of just computing the SHA1 and xoring it 
  34  * against diget, also perform the digest of "digest" itself and 
  35  * replace the old value with the new one. 
  37  * So the final digest will be: 
  39  * digest = SHA1(digest xor SHA1(data)) 
  41  * This function is used every time we want to preserve the order so 
  42  * that digest(a,b,c,d) will be different than digest(b,c,d,a) 
  44  * Also note that mixdigest("foo") followed by mixdigest("bar") 
  45  * will lead to a different digest compared to "fo", "obar". 
  47 void mixDigest(unsigned char *digest
, void *ptr
, size_t len
) { 
  51     xorDigest(digest
,s
,len
); 
  53     SHA1Update(&ctx
,digest
,20); 
  54     SHA1Final(digest
,&ctx
); 
  57 void mixObjectDigest(unsigned char *digest
, robj 
*o
) { 
  58     o 
= getDecodedObject(o
); 
  59     mixDigest(digest
,o
->ptr
,sdslen(o
->ptr
)); 
  63 /* Compute the dataset digest. Since keys, sets elements, hashes elements 
  64  * are not ordered, we use a trick: every aggregate digest is the xor 
  65  * of the digests of their elements. This way the order will not change 
  66  * the result. For list instead we use a feedback entering the output digest 
  67  * as input in order to ensure that a different ordered list will result in 
  68  * a different digest. */ 
  69 void computeDatasetDigest(unsigned char *final
) { 
  70     unsigned char digest
[20]; 
  72     dictIterator 
*di 
= NULL
; 
  77     memset(final
,0,20); /* Start with a clean result */ 
  79     for (j 
= 0; j 
< server
.dbnum
; j
++) { 
  80         redisDb 
*db 
= server
.db
+j
; 
  82         if (dictSize(db
->dict
) == 0) continue; 
  83         di 
= dictGetIterator(db
->dict
); 
  85         /* hash the DB id, so the same dataset moved in a different 
  86          * DB will lead to a different digest */ 
  88         mixDigest(final
,&aux
,sizeof(aux
)); 
  90         /* Iterate this DB writing every entry */ 
  91         while((de 
= dictNext(di
)) != NULL
) { 
  96             memset(digest
,0,20); /* This key-val digest */ 
  97             key 
= dictGetEntryKey(de
); 
  98             keyobj 
= createStringObject(key
,sdslen(key
)); 
 100             mixDigest(digest
,key
,sdslen(key
)); 
 102             /* Make sure the key is loaded if VM is active */ 
 103             o 
= lookupKeyRead(db
,keyobj
); 
 105             aux 
= htonl(o
->type
); 
 106             mixDigest(digest
,&aux
,sizeof(aux
)); 
 107             expiretime 
= getExpire(db
,keyobj
); 
 109             /* Save the key and associated value */ 
 110             if (o
->type 
== REDIS_STRING
) { 
 111                 mixObjectDigest(digest
,o
); 
 112             } else if (o
->type 
== REDIS_LIST
) { 
 113                 listTypeIterator 
*li 
= listTypeInitIterator(o
,0,REDIS_TAIL
); 
 115                 while(listTypeNext(li
,&entry
)) { 
 116                     robj 
*eleobj 
= listTypeGet(&entry
); 
 117                     mixObjectDigest(digest
,eleobj
); 
 118                     decrRefCount(eleobj
); 
 120                 listTypeReleaseIterator(li
); 
 121             } else if (o
->type 
== REDIS_SET
) { 
 122                 setTypeIterator 
*si 
= setTypeInitIterator(o
); 
 124                 while((ele 
= setTypeNext(si
)) != NULL
) { 
 125                     xorObjectDigest(digest
,ele
); 
 128                 setTypeReleaseIterator(si
); 
 129             } else if (o
->type 
== REDIS_ZSET
) { 
 131                 dictIterator 
*di 
= dictGetIterator(zs
->dict
); 
 134                 while((de 
= dictNext(di
)) != NULL
) { 
 135                     robj 
*eleobj 
= dictGetEntryKey(de
); 
 136                     double *score 
= dictGetEntryVal(de
); 
 137                     unsigned char eledigest
[20]; 
 139                     snprintf(buf
,sizeof(buf
),"%.17g",*score
); 
 140                     memset(eledigest
,0,20); 
 141                     mixObjectDigest(eledigest
,eleobj
); 
 142                     mixDigest(eledigest
,buf
,strlen(buf
)); 
 143                     xorDigest(digest
,eledigest
,20); 
 145                 dictReleaseIterator(di
); 
 146             } else if (o
->type 
== REDIS_HASH
) { 
 147                 hashTypeIterator 
*hi
; 
 150                 hi 
= hashTypeInitIterator(o
); 
 151                 while (hashTypeNext(hi
) != REDIS_ERR
) { 
 152                     unsigned char eledigest
[20]; 
 154                     memset(eledigest
,0,20); 
 155                     obj 
= hashTypeCurrent(hi
,REDIS_HASH_KEY
); 
 156                     mixObjectDigest(eledigest
,obj
); 
 158                     obj 
= hashTypeCurrent(hi
,REDIS_HASH_VALUE
); 
 159                     mixObjectDigest(eledigest
,obj
); 
 161                     xorDigest(digest
,eledigest
,20); 
 163                 hashTypeReleaseIterator(hi
); 
 165                 redisPanic("Unknown object type"); 
 167             /* If the key has an expire, add it to the mix */ 
 168             if (expiretime 
!= -1) xorDigest(digest
,"!!expire!!",10); 
 169             /* We can finally xor the key-val digest to the final digest */ 
 170             xorDigest(final
,digest
,20); 
 171             decrRefCount(keyobj
); 
 173         dictReleaseIterator(di
); 
 177 void debugCommand(redisClient 
*c
) { 
 178     if (!strcasecmp(c
->argv
[1]->ptr
,"segfault")) { 
 180     } else if (!strcasecmp(c
->argv
[1]->ptr
,"reload")) { 
 181         if (rdbSave(server
.dbfilename
) != REDIS_OK
) { 
 182             addReply(c
,shared
.err
); 
 186         if (rdbLoad(server
.dbfilename
) != REDIS_OK
) { 
 187             addReply(c
,shared
.err
); 
 190         redisLog(REDIS_WARNING
,"DB reloaded by DEBUG RELOAD"); 
 191         addReply(c
,shared
.ok
); 
 192     } else if (!strcasecmp(c
->argv
[1]->ptr
,"loadaof")) { 
 194         if (loadAppendOnlyFile(server
.appendfilename
) != REDIS_OK
) { 
 195             addReply(c
,shared
.err
); 
 198         redisLog(REDIS_WARNING
,"Append Only File loaded by DEBUG LOADAOF"); 
 199         addReply(c
,shared
.ok
); 
 200     } else if (!strcasecmp(c
->argv
[1]->ptr
,"object") && c
->argc 
== 3) { 
 201         dictEntry 
*de 
= dictFind(c
->db
->dict
,c
->argv
[2]->ptr
); 
 205             addReply(c
,shared
.nokeyerr
); 
 208         val 
= dictGetEntryVal(de
); 
 209         if (!server
.vm_enabled 
|| (val
->storage 
== REDIS_VM_MEMORY 
|| 
 210                                    val
->storage 
== REDIS_VM_SWAPPING
)) { 
 213             strenc 
= strEncoding(val
->encoding
); 
 214             addReplyStatusFormat(c
, 
 215                 "Value at:%p refcount:%d " 
 216                 "encoding:%s serializedlength:%lld " 
 217                 "lru:%d lru_seconds_idle:%lu", 
 218                 (void*)val
, val
->refcount
, 
 219                 strenc
, (long long) rdbSavedObjectLen(val
,NULL
), 
 220                 val
->lru
, estimateObjectIdleTime(val
)); 
 222             vmpointer 
*vp 
= (vmpointer
*) val
; 
 223             addReplyStatusFormat(c
, 
 224                 "Value swapped at: page %llu " 
 226                 (unsigned long long) vp
->page
, 
 227                 (unsigned long long) vp
->usedpages
); 
 229     } else if (!strcasecmp(c
->argv
[1]->ptr
,"swapin") && c
->argc 
== 3) { 
 230         lookupKeyRead(c
->db
,c
->argv
[2]); 
 231         addReply(c
,shared
.ok
); 
 232     } else if (!strcasecmp(c
->argv
[1]->ptr
,"swapout") && c
->argc 
== 3) { 
 233         dictEntry 
*de 
= dictFind(c
->db
->dict
,c
->argv
[2]->ptr
); 
 237         if (!server
.vm_enabled
) { 
 238             addReplyError(c
,"Virtual Memory is disabled"); 
 242             addReply(c
,shared
.nokeyerr
); 
 245         val 
= dictGetEntryVal(de
); 
 247         if (val
->storage 
!= REDIS_VM_MEMORY
) { 
 248             addReplyError(c
,"This key is not in memory"); 
 249         } else if (val
->refcount 
!= 1) { 
 250             addReplyError(c
,"Object is shared"); 
 251         } else if ((vp 
= vmSwapObjectBlocking(val
)) != NULL
) { 
 252             dictGetEntryVal(de
) = vp
; 
 253             addReply(c
,shared
.ok
); 
 255             addReply(c
,shared
.err
); 
 257     } else if (!strcasecmp(c
->argv
[1]->ptr
,"populate") && c
->argc 
== 3) { 
 262         if (getLongFromObjectOrReply(c
, c
->argv
[2], &keys
, NULL
) != REDIS_OK
) 
 264         for (j 
= 0; j 
< keys
; j
++) { 
 265             snprintf(buf
,sizeof(buf
),"key:%lu",j
); 
 266             key 
= createStringObject(buf
,strlen(buf
)); 
 267             if (lookupKeyRead(c
->db
,key
) != NULL
) { 
 271             snprintf(buf
,sizeof(buf
),"value:%lu",j
); 
 272             val 
= createStringObject(buf
,strlen(buf
)); 
 273             dbAdd(c
->db
,key
,val
); 
 276         addReply(c
,shared
.ok
); 
 277     } else if (!strcasecmp(c
->argv
[1]->ptr
,"digest") && c
->argc 
== 2) { 
 278         unsigned char digest
[20]; 
 282         computeDatasetDigest(digest
); 
 283         for (j 
= 0; j 
< 20; j
++) 
 284             d 
= sdscatprintf(d
, "%02x",digest
[j
]); 
 289             "Syntax error, try DEBUG [SEGFAULT|OBJECT <key>|SWAPIN <key>|SWAPOUT <key>|RELOAD]"); 
 293 void _redisAssert(char *estr
, char *file
, int line
) { 
 294     redisLog(REDIS_WARNING
,"=== ASSERTION FAILED ==="); 
 295     redisLog(REDIS_WARNING
,"==> %s:%d '%s' is not true",file
,line
,estr
); 
 296 #ifdef HAVE_BACKTRACE 
 297     redisLog(REDIS_WARNING
,"(forcing SIGSEGV in order to print the stack trace)"); 
 302 void _redisPanic(char *msg
, char *file
, int line
) { 
 303     redisLog(REDIS_WARNING
,"!!! Software Failure. Press left mouse button to continue"); 
 304     redisLog(REDIS_WARNING
,"Guru Meditation: %s #%s:%d",msg
,file
,line
); 
 305 #ifdef HAVE_BACKTRACE 
 306     redisLog(REDIS_WARNING
,"(forcing SIGSEGV in order to print the stack trace)");