8 #include "solarisfixes.h" 
  21 #include "ae.h"     /* Event driven programming library */ 
  22 #include "sds.h"    /* Dynamic safe strings */ 
  23 #include "dict.h"   /* Hash tables */ 
  24 #include "adlist.h" /* Linked lists */ 
  25 #include "zmalloc.h" /* total memory usage aware version of malloc/free */ 
  26 #include "anet.h"   /* Networking the easy way */ 
  27 #include "zipmap.h" /* Compact string -> string data structure */ 
  28 #include "ziplist.h" /* Compact list data structure */ 
  35 /* Static server configuration */ 
  36 #define REDIS_SERVERPORT        6379    /* TCP port */ 
  37 #define REDIS_MAXIDLETIME       (60*5)  /* default client timeout */ 
  38 #define REDIS_IOBUF_LEN         1024 
  39 #define REDIS_LOADBUF_LEN       1024 
  40 #define REDIS_STATIC_ARGS       8 
  41 #define REDIS_DEFAULT_DBNUM     16 
  42 #define REDIS_CONFIGLINE_MAX    1024 
  43 #define REDIS_OBJFREELIST_MAX   1000000 /* Max number of objects to cache */ 
  44 #define REDIS_MAX_SYNC_TIME     60      /* Slave can't take more to sync */ 
  45 #define REDIS_EXPIRELOOKUPS_PER_CRON    10 /* lookup 10 expires per loop */ 
  46 #define REDIS_MAX_WRITE_PER_EVENT (1024*64) 
  47 #define REDIS_REQUEST_MAX_SIZE (1024*1024*256) /* max bytes in inline command */ 
  48 #define REDIS_SHARED_INTEGERS 10000 
  50 /* If more then REDIS_WRITEV_THRESHOLD write packets are pending use writev */ 
  51 #define REDIS_WRITEV_THRESHOLD      3 
  52 /* Max number of iovecs used for each writev call */ 
  53 #define REDIS_WRITEV_IOVEC_COUNT    256 
  55 /* Hash table parameters */ 
  56 #define REDIS_HT_MINFILL        10      /* Minimal hash table fill 10% */ 
  59 #define REDIS_CMD_BULK          1       /* Bulk write command */ 
  60 #define REDIS_CMD_INLINE        2       /* Inline command */ 
  61 /* REDIS_CMD_DENYOOM reserves a longer comment: all the commands marked with 
  62    this flags will return an error when the 'maxmemory' option is set in the 
  63    config file and the server is using more than maxmemory bytes of memory. 
  64    In short this commands are denied on low memory conditions. */ 
  65 #define REDIS_CMD_DENYOOM       4 
  66 #define REDIS_CMD_FORCE_REPLICATION 8 /* Force replication even if dirty is 0 */ 
  69 #define REDIS_STRING 0 
  74 #define REDIS_VMPOINTER 8 
  76 /* Objects encoding. Some kind of objects like Strings and Hashes can be 
  77  * internally represented in multiple ways. The 'encoding' field of the object 
  78  * is set to one of this fields for this object. */ 
  79 #define REDIS_ENCODING_RAW 0     /* Raw representation */ 
  80 #define REDIS_ENCODING_INT 1     /* Encoded as integer */ 
  81 #define REDIS_ENCODING_HT 2      /* Encoded as hash table */ 
  82 #define REDIS_ENCODING_ZIPMAP 3  /* Encoded as zipmap */ 
  83 #define REDIS_ENCODING_LINKEDLIST 4 /* Encoded as regular linked list */ 
  84 #define REDIS_ENCODING_ZIPLIST 5 /* Encoded as ziplist */ 
  86 /* Object types only used for dumping to disk */ 
  87 #define REDIS_EXPIRETIME 253 
  88 #define REDIS_SELECTDB 254 
  91 /* Defines related to the dump file format. To store 32 bits lengths for short 
  92  * keys requires a lot of space, so we check the most significant 2 bits of 
  93  * the first byte to interpreter the length: 
  95  * 00|000000 => if the two MSB are 00 the len is the 6 bits of this byte 
  96  * 01|000000 00000000 =>  01, the len is 14 byes, 6 bits + 8 bits of next byte 
  97  * 10|000000 [32 bit integer] => if it's 01, a full 32 bit len will follow 
  98  * 11|000000 this means: specially encoded object will follow. The six bits 
  99  *           number specify the kind of object that follows. 
 100  *           See the REDIS_RDB_ENC_* defines. 
 102  * Lenghts up to 63 are stored using a single byte, most DB keys, and may 
 103  * values, will fit inside. */ 
 104 #define REDIS_RDB_6BITLEN 0 
 105 #define REDIS_RDB_14BITLEN 1 
 106 #define REDIS_RDB_32BITLEN 2 
 107 #define REDIS_RDB_ENCVAL 3 
 108 #define REDIS_RDB_LENERR UINT_MAX 
 110 /* When a length of a string object stored on disk has the first two bits 
 111  * set, the remaining two bits specify a special encoding for the object 
 112  * accordingly to the following defines: */ 
 113 #define REDIS_RDB_ENC_INT8 0        /* 8 bit signed integer */ 
 114 #define REDIS_RDB_ENC_INT16 1       /* 16 bit signed integer */ 
 115 #define REDIS_RDB_ENC_INT32 2       /* 32 bit signed integer */ 
 116 #define REDIS_RDB_ENC_LZF 3         /* string compressed with FASTLZ */ 
 118 /* Virtual memory object->where field. */ 
 119 #define REDIS_VM_MEMORY 0       /* The object is on memory */ 
 120 #define REDIS_VM_SWAPPED 1      /* The object is on disk */ 
 121 #define REDIS_VM_SWAPPING 2     /* Redis is swapping this object on disk */ 
 122 #define REDIS_VM_LOADING 3      /* Redis is loading this object from disk */ 
 124 /* Virtual memory static configuration stuff. 
 125  * Check vmFindContiguousPages() to know more about this magic numbers. */ 
 126 #define REDIS_VM_MAX_NEAR_PAGES 65536 
 127 #define REDIS_VM_MAX_RANDOM_JUMP 4096 
 128 #define REDIS_VM_MAX_THREADS 32 
 129 #define REDIS_THREAD_STACK_SIZE (1024*1024*4) 
 130 /* The following is the *percentage* of completed I/O jobs to process when the 
 131  * handelr is called. While Virtual Memory I/O operations are performed by 
 132  * threads, this operations must be processed by the main thread when completed 
 133  * in order to take effect. */ 
 134 #define REDIS_MAX_COMPLETED_JOBS_PROCESSED 1 
 137 #define REDIS_SLAVE 1       /* This client is a slave server */ 
 138 #define REDIS_MASTER 2      /* This client is a master server */ 
 139 #define REDIS_MONITOR 4     /* This client is a slave monitor, see MONITOR */ 
 140 #define REDIS_MULTI 8       /* This client is in a MULTI context */ 
 141 #define REDIS_BLOCKED 16    /* The client is waiting in a blocking operation */ 
 142 #define REDIS_IO_WAIT 32    /* The client is waiting for Virtual Memory I/O */ 
 143 #define REDIS_DIRTY_CAS 64  /* Watched keys modified. EXEC will fail. */ 
 145 /* Slave replication state - slave side */ 
 146 #define REDIS_REPL_NONE 0   /* No active replication */ 
 147 #define REDIS_REPL_CONNECT 1    /* Must connect to master */ 
 148 #define REDIS_REPL_CONNECTED 2  /* Connected to master */ 
 150 /* Slave replication state - from the point of view of master 
 151  * Note that in SEND_BULK and ONLINE state the slave receives new updates 
 152  * in its output queue. In the WAIT_BGSAVE state instead the server is waiting 
 153  * to start the next background saving in order to send updates to it. */ 
 154 #define REDIS_REPL_WAIT_BGSAVE_START 3 /* master waits bgsave to start feeding it */ 
 155 #define REDIS_REPL_WAIT_BGSAVE_END 4 /* master waits bgsave to start bulk DB transmission */ 
 156 #define REDIS_REPL_SEND_BULK 5 /* master is sending the bulk DB */ 
 157 #define REDIS_REPL_ONLINE 6 /* bulk DB already transmitted, receive updates */ 
 159 /* List related stuff */ 
 163 /* Sort operations */ 
 164 #define REDIS_SORT_GET 0 
 165 #define REDIS_SORT_ASC 1 
 166 #define REDIS_SORT_DESC 2 
 167 #define REDIS_SORTKEY_MAX 1024 
 170 #define REDIS_DEBUG 0 
 171 #define REDIS_VERBOSE 1 
 172 #define REDIS_NOTICE 2 
 173 #define REDIS_WARNING 3 
 175 /* Anti-warning macro... */ 
 176 #define REDIS_NOTUSED(V) ((void) V) 
 178 #define ZSKIPLIST_MAXLEVEL 32 /* Should be enough for 2^32 elements */ 
 179 #define ZSKIPLIST_P 0.25      /* Skiplist P = 1/4 */ 
 181 /* Append only defines */ 
 182 #define APPENDFSYNC_NO 0 
 183 #define APPENDFSYNC_ALWAYS 1 
 184 #define APPENDFSYNC_EVERYSEC 2 
 186 /* Zip structure related defaults */ 
 187 #define REDIS_HASH_MAX_ZIPMAP_ENTRIES 64 
 188 #define REDIS_HASH_MAX_ZIPMAP_VALUE 512 
 189 #define REDIS_LIST_MAX_ZIPLIST_ENTRIES 1024 
 190 #define REDIS_LIST_MAX_ZIPLIST_VALUE 32 
 192 /* Sets operations codes */ 
 193 #define REDIS_OP_UNION 0 
 194 #define REDIS_OP_DIFF 1 
 195 #define REDIS_OP_INTER 2 
 197 /* We can print the stacktrace, so our assert is defined this way: */ 
 198 #define redisAssert(_e) ((_e)?(void)0 : (_redisAssert(#_e,__FILE__,__LINE__),_exit(1))) 
 199 #define redisPanic(_e) _redisPanic(#_e,__FILE__,__LINE__),_exit(1) 
 200 void _redisAssert(char *estr
, char *file
, int line
); 
 201 void _redisPanic(char *msg
, char *file
, int line
); 
 203 /*----------------------------------------------------------------------------- 
 205  *----------------------------------------------------------------------------*/ 
 207 /* A redis object, that is a type able to hold a string / list / set */ 
 209 /* The actual Redis Object */ 
 210 typedef struct redisObject 
{ 
 212     unsigned storage
:2;     /* REDIS_VM_MEMORY or REDIS_VM_SWAPPING */ 
 214     unsigned lru
:22;        /* lru time (relative to server.lruclock) */ 
 217     /* VM fields are only allocated if VM is active, otherwise the 
 218      * object allocation function will just allocate 
 219      * sizeof(redisObjct) minus sizeof(redisObjectVM), so using 
 220      * Redis without VM active will not have any overhead. */ 
 223 /* The VM pointer structure - identifies an object in the swap file. 
 225  * This object is stored in place of the value 
 226  * object in the main key->value hash table representing a database. 
 227  * Note that the first fields (type, storage) are the same as the redisObject 
 228  * structure so that vmPointer strucuters can be accessed even when casted 
 229  * as redisObject structures. 
 231  * This is useful as we don't know if a value object is or not on disk, but we 
 232  * are always able to read obj->storage to check this. For vmPointer 
 233  * structures "type" is set to REDIS_VMPOINTER (even if without this field 
 234  * is still possible to check the kind of object from the value of 'storage').*/ 
 235 typedef struct vmPointer 
{ 
 237     unsigned storage
:2; /* REDIS_VM_SWAPPED or REDIS_VM_LOADING */ 
 239     unsigned int vtype
; /* type of the object stored in the swap file */ 
 240     off_t page
;         /* the page at witch the object is stored on disk */ 
 241     off_t usedpages
;    /* number of pages used on disk */ 
 244 /* Macro used to initalize a Redis object allocated on the stack. 
 245  * Note that this macro is taken near the structure definition to make sure 
 246  * we'll update it when the structure is changed, to avoid bugs like 
 247  * bug #85 introduced exactly in this way. */ 
 248 #define initStaticStringObject(_var,_ptr) do { \ 
 250     _var.type = REDIS_STRING; \ 
 251     _var.encoding = REDIS_ENCODING_RAW; \ 
 253     _var.storage = REDIS_VM_MEMORY; \ 
 256 typedef struct redisDb 
{ 
 257     dict 
*dict
;                 /* The keyspace for this DB */ 
 258     dict 
*expires
;              /* Timeout of keys with a timeout set */ 
 259     dict 
*blocking_keys
;        /* Keys with clients waiting for data (BLPOP) */ 
 260     dict 
*io_keys
;              /* Keys with clients waiting for VM I/O */ 
 261     dict 
*watched_keys
;         /* WATCHED keys for MULTI/EXEC CAS */ 
 265 /* Client MULTI/EXEC state */ 
 266 typedef struct multiCmd 
{ 
 269     struct redisCommand 
*cmd
; 
 272 typedef struct multiState 
{ 
 273     multiCmd 
*commands
;     /* Array of MULTI commands */ 
 274     int count
;              /* Total number of MULTI commands */ 
 277 /* With multiplexing we need to take per-clinet state. 
 278  * Clients are taken in a liked list. */ 
 279 typedef struct redisClient 
{ 
 284     robj 
**argv
, **mbargv
; 
 286     int bulklen
;            /* bulk read len. -1 if not in bulk read mode */ 
 287     int multibulk
;          /* multi bulk command format active */ 
 290     time_t lastinteraction
; /* time of the last interaction, used for timeout */ 
 291     int flags
;              /* REDIS_SLAVE | REDIS_MONITOR | REDIS_MULTI ... */ 
 292     int slaveseldb
;         /* slave selected db, if this client is a slave */ 
 293     int authenticated
;      /* when requirepass is non-NULL */ 
 294     int replstate
;          /* replication state if this is a slave */ 
 295     int repldbfd
;           /* replication DB file descriptor */ 
 296     long repldboff
;         /* replication DB file offset */ 
 297     off_t repldbsize
;       /* replication DB file size */ 
 298     multiState mstate
;      /* MULTI/EXEC state */ 
 299     robj 
**blocking_keys
;   /* The key we are waiting to terminate a blocking 
 300                              * operation such as BLPOP. Otherwise NULL. */ 
 301     int blocking_keys_num
;  /* Number of blocking keys */ 
 302     time_t blockingto
;      /* Blocking operation timeout. If UNIX current time 
 303                              * is >= blockingto then the operation timed out. */ 
 304     list 
*io_keys
;          /* Keys this client is waiting to be loaded from the 
 305                              * swap file in order to continue. */ 
 306     list 
*watched_keys
;     /* Keys WATCHED for MULTI/EXEC CAS */ 
 307     dict 
*pubsub_channels
;  /* channels a client is interested in (SUBSCRIBE) */ 
 308     list 
*pubsub_patterns
;  /* patterns a client is interested in (SUBSCRIBE) */ 
 316 struct sharedObjectsStruct 
{ 
 317     robj 
*crlf
, *ok
, *err
, *emptybulk
, *czero
, *cone
, *cnegone
, *pong
, *space
, 
 318     *colon
, *nullbulk
, *nullmultibulk
, *queued
, 
 319     *emptymultibulk
, *wrongtypeerr
, *nokeyerr
, *syntaxerr
, *sameobjecterr
, 
 320     *outofrangeerr
, *plus
, 
 321     *select0
, *select1
, *select2
, *select3
, *select4
, 
 322     *select5
, *select6
, *select7
, *select8
, *select9
, 
 323     *messagebulk
, *pmessagebulk
, *subscribebulk
, *unsubscribebulk
, *mbulk3
, 
 324     *mbulk4
, *psubscribebulk
, *punsubscribebulk
, 
 325     *integers
[REDIS_SHARED_INTEGERS
]; 
 328 /* Global server state structure */ 
 330     pthread_t mainthread
; 
 334     long long dirty
;            /* changes to DB from the last save */ 
 336     list 
*slaves
, *monitors
; 
 337     char neterr
[ANET_ERR_LEN
]; 
 339     int cronloops
;              /* number of times the cron function run */ 
 340     list 
*objfreelist
;          /* A list of freed objects to avoid malloc() */ 
 341     time_t lastsave
;            /* Unix time of last save succeeede */ 
 342     /* Fields used only for stats */ 
 343     time_t stat_starttime
;         /* server start time */ 
 344     long long stat_numcommands
;    /* number of processed commands */ 
 345     long long stat_numconnections
; /* number of connections received */ 
 346     long long stat_expiredkeys
;   /* number of expired keys */ 
 355     int no_appendfsync_on_rewrite
; 
 361     pid_t bgsavechildpid
; 
 362     pid_t bgrewritechildpid
; 
 363     sds bgrewritebuf
; /* buffer taken by parent during oppend only rewrite */ 
 364     sds aofbuf
;       /* AOF buffer, written before entering the event loop */ 
 365     struct saveparam 
*saveparams
; 
 370     char *appendfilename
; 
 374     /* Replication related */ 
 379     redisClient 
*master
;    /* client that is master for this slave */ 
 381     unsigned int maxclients
; 
 382     unsigned long long maxmemory
; 
 383     unsigned int blpop_blocked_clients
; 
 384     unsigned int vm_blocked_clients
; 
 385     /* Sort parameters - qsort_r() is only available under BSD so we 
 386      * have to take this state global, in order to pass it to sortCompare() */ 
 390     /* Virtual memory configuration */ 
 395     unsigned long long vm_max_memory
; 
 396     /* Zip structure config */ 
 397     size_t hash_max_zipmap_entries
; 
 398     size_t hash_max_zipmap_value
; 
 399     size_t list_max_ziplist_entries
; 
 400     size_t list_max_ziplist_value
; 
 401     /* Virtual memory state */ 
 404     off_t vm_next_page
; /* Next probably empty page */ 
 405     off_t vm_near_pages
; /* Number of pages allocated sequentially */ 
 406     unsigned char *vm_bitmap
; /* Bitmap of free/used pages */ 
 407     time_t unixtime
;    /* Unix time sampled every second. */ 
 408     /* Virtual memory I/O threads stuff */ 
 409     /* An I/O thread process an element taken from the io_jobs queue and 
 410      * put the result of the operation in the io_done list. While the 
 411      * job is being processed, it's put on io_processing queue. */ 
 412     list 
*io_newjobs
; /* List of VM I/O jobs yet to be processed */ 
 413     list 
*io_processing
; /* List of VM I/O jobs being processed */ 
 414     list 
*io_processed
; /* List of VM I/O jobs already processed */ 
 415     list 
*io_ready_clients
; /* Clients ready to be unblocked. All keys loaded */ 
 416     pthread_mutex_t io_mutex
; /* lock to access io_jobs/io_done/io_thread_job */ 
 417     pthread_mutex_t obj_freelist_mutex
; /* safe redis objects creation/free */ 
 418     pthread_mutex_t io_swapfile_mutex
; /* So we can lseek + write */ 
 419     pthread_attr_t io_threads_attr
; /* attributes for threads creation */ 
 420     int io_active_threads
; /* Number of running I/O threads */ 
 421     int vm_max_threads
; /* Max number of I/O threads running at the same time */ 
 422     /* Our main thread is blocked on the event loop, locking for sockets ready 
 423      * to be read or written, so when a threaded I/O operation is ready to be 
 424      * processed by the main thread, the I/O thread will use a unix pipe to 
 425      * awake the main thread. The followings are the two pipe FDs. */ 
 426     int io_ready_pipe_read
; 
 427     int io_ready_pipe_write
; 
 428     /* Virtual memory stats */ 
 429     unsigned long long vm_stats_used_pages
; 
 430     unsigned long long vm_stats_swapped_objects
; 
 431     unsigned long long vm_stats_swapouts
; 
 432     unsigned long long vm_stats_swapins
; 
 434     dict 
*pubsub_channels
; /* Map channels to list of subscribed clients */ 
 435     list 
*pubsub_patterns
; /* A list of pubsub_patterns */ 
 438     unsigned lruclock
:22;        /* clock incrementing every minute, for LRU */ 
 439     unsigned lruclock_padding
:10; 
 442 typedef struct pubsubPattern 
{ 
 447 typedef void redisCommandProc(redisClient 
*c
); 
 448 typedef void redisVmPreloadProc(redisClient 
*c
, struct redisCommand 
*cmd
, int argc
, robj 
**argv
); 
 449 struct redisCommand 
{ 
 451     redisCommandProc 
*proc
; 
 454     /* Use a function to determine which keys need to be loaded 
 455      * in the background prior to executing this command. Takes precedence 
 456      * over vm_firstkey and others, ignored when NULL */ 
 457     redisVmPreloadProc 
*vm_preload_proc
; 
 458     /* What keys should be loaded in background when calling this command? */ 
 459     int vm_firstkey
; /* The first argument that's a key (0 = no keys) */ 
 460     int vm_lastkey
;  /* THe last argument that's a key */ 
 461     int vm_keystep
;  /* The step between first and last key */ 
 464 struct redisFunctionSym 
{ 
 466     unsigned long pointer
; 
 469 typedef struct _redisSortObject 
{ 
 477 typedef struct _redisSortOperation 
{ 
 480 } redisSortOperation
; 
 482 /* ZSETs use a specialized version of Skiplists */ 
 484 typedef struct zskiplistNode 
{ 
 485     struct zskiplistNode 
**forward
; 
 486     struct zskiplistNode 
*backward
; 
 492 typedef struct zskiplist 
{ 
 493     struct zskiplistNode 
*header
, *tail
; 
 494     unsigned long length
; 
 498 typedef struct zset 
{ 
 503 /* VM threaded I/O request message */ 
 504 #define REDIS_IOJOB_LOAD 0          /* Load from disk to memory */ 
 505 #define REDIS_IOJOB_PREPARE_SWAP 1  /* Compute needed pages */ 
 506 #define REDIS_IOJOB_DO_SWAP 2       /* Swap from memory to disk */ 
 507 typedef struct iojob 
{ 
 508     int type
;   /* Request type, REDIS_IOJOB_* */ 
 509     redisDb 
*db
;/* Redis database */ 
 510     robj 
*key
;  /* This I/O request is about swapping this key */ 
 511     robj 
*id
;   /* Unique identifier of this job: 
 512                    this is the object to swap for REDIS_IOREQ_*_SWAP, or the 
 513                    vmpointer objct for REDIS_IOREQ_LOAD. */ 
 514     robj 
*val
;  /* the value to swap for REDIS_IOREQ_*_SWAP, otherwise this 
 515                  * field is populated by the I/O thread for REDIS_IOREQ_LOAD. */ 
 516     off_t page
; /* Swap page where to read/write the object */ 
 517     off_t pages
; /* Swap pages needed to save object. PREPARE_SWAP return val */ 
 518     int canceled
; /* True if this command was canceled by blocking side of VM */ 
 519     pthread_t thread
; /* ID of the thread processing this entry */ 
 522 /* Structure to hold list iteration abstraction. */ 
 525     unsigned char encoding
; 
 526     unsigned char direction
; /* Iteration direction */ 
 531 /* Structure for an entry while iterating over a list. */ 
 533     listTypeIterator 
*li
; 
 534     unsigned char *zi
;  /* Entry in ziplist */ 
 535     listNode 
*ln
;       /* Entry in linked list */ 
 538 /* Structure to hold hash iteration abstration. Note that iteration over 
 539  * hashes involves both fields and values. Because it is possible that 
 540  * not both are required, store pointers in the iterator to avoid 
 541  * unnecessary memory allocation for fields/values. */ 
 545     unsigned char *zk
, *zv
; 
 546     unsigned int zklen
, zvlen
; 
 552 #define REDIS_HASH_KEY 1 
 553 #define REDIS_HASH_VALUE 2 
 555 /*----------------------------------------------------------------------------- 
 556  * Extern declarations 
 557  *----------------------------------------------------------------------------*/ 
 559 extern struct redisServer server
; 
 560 extern struct sharedObjectsStruct shared
; 
 561 extern dictType setDictType
; 
 562 extern dictType zsetDictType
; 
 563 extern double R_Zero
, R_PosInf
, R_NegInf
, R_Nan
; 
 564 dictType hashDictType
; 
 566 /*----------------------------------------------------------------------------- 
 567  * Functions prototypes 
 568  *----------------------------------------------------------------------------*/ 
 570 /* networking.c -- Networking and Client related operations */ 
 571 redisClient 
*createClient(int fd
); 
 572 void closeTimedoutClients(void); 
 573 void freeClient(redisClient 
*c
); 
 574 void resetClient(redisClient 
*c
); 
 575 void sendReplyToClient(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 576 void sendReplyToClientWritev(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 577 void addReply(redisClient 
*c
, robj 
*obj
); 
 578 void addReplySds(redisClient 
*c
, sds s
); 
 579 void processInputBuffer(redisClient 
*c
); 
 580 void acceptHandler(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 581 void readQueryFromClient(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 582 void addReplyBulk(redisClient 
*c
, robj 
*obj
); 
 583 void addReplyBulkCString(redisClient 
*c
, char *s
); 
 584 void acceptHandler(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 585 void addReply(redisClient 
*c
, robj 
*obj
); 
 586 void addReplySds(redisClient 
*c
, sds s
); 
 587 void addReplyDouble(redisClient 
*c
, double d
); 
 588 void addReplyLongLong(redisClient 
*c
, long long ll
); 
 589 void addReplyUlong(redisClient 
*c
, unsigned long ul
); 
 590 void *dupClientReplyValue(void *o
); 
 593 void listTypeTryConversion(robj 
*subject
, robj 
*value
); 
 594 void listTypePush(robj 
*subject
, robj 
*value
, int where
); 
 595 robj 
*listTypePop(robj 
*subject
, int where
); 
 596 unsigned long listTypeLength(robj 
*subject
); 
 597 listTypeIterator 
*listTypeInitIterator(robj 
*subject
, int index
, unsigned char direction
); 
 598 void listTypeReleaseIterator(listTypeIterator 
*li
); 
 599 int listTypeNext(listTypeIterator 
*li
, listTypeEntry 
*entry
); 
 600 robj 
*listTypeGet(listTypeEntry 
*entry
); 
 601 void listTypeInsert(listTypeEntry 
*entry
, robj 
*value
, int where
); 
 602 int listTypeEqual(listTypeEntry 
*entry
, robj 
*o
); 
 603 void listTypeDelete(listTypeEntry 
*entry
); 
 604 void listTypeConvert(robj 
*subject
, int enc
); 
 605 void unblockClientWaitingData(redisClient 
*c
); 
 606 int handleClientsWaitingListPush(redisClient 
*c
, robj 
*key
, robj 
*ele
); 
 607 void popGenericCommand(redisClient 
*c
, int where
); 
 609 /* MULTI/EXEC/WATCH... */ 
 610 void unwatchAllKeys(redisClient 
*c
); 
 611 void initClientMultiState(redisClient 
*c
); 
 612 void freeClientMultiState(redisClient 
*c
); 
 613 void queueMultiCommand(redisClient 
*c
, struct redisCommand 
*cmd
); 
 614 void touchWatchedKey(redisDb 
*db
, robj 
*key
); 
 615 void touchWatchedKeysOnFlush(int dbid
); 
 617 /* Redis object implementation */ 
 618 void decrRefCount(void *o
); 
 619 void incrRefCount(robj 
*o
); 
 620 void freeStringObject(robj 
*o
); 
 621 void freeListObject(robj 
*o
); 
 622 void freeSetObject(robj 
*o
); 
 623 void freeZsetObject(robj 
*o
); 
 624 void freeHashObject(robj 
*o
); 
 625 robj 
*createObject(int type
, void *ptr
); 
 626 robj 
*createStringObject(char *ptr
, size_t len
); 
 627 robj 
*dupStringObject(robj 
*o
); 
 628 robj 
*tryObjectEncoding(robj 
*o
); 
 629 robj 
*getDecodedObject(robj 
*o
); 
 630 size_t stringObjectLen(robj 
*o
); 
 631 int tryFreeOneObjectFromFreelist(void); 
 632 robj 
*createStringObjectFromLongLong(long long value
); 
 633 robj 
*createListObject(void); 
 634 robj 
*createZiplistObject(void); 
 635 robj 
*createSetObject(void); 
 636 robj 
*createHashObject(void); 
 637 robj 
*createZsetObject(void); 
 638 int getLongFromObjectOrReply(redisClient 
*c
, robj 
*o
, long *target
, const char *msg
); 
 639 int checkType(redisClient 
*c
, robj 
*o
, int type
); 
 640 int getLongLongFromObjectOrReply(redisClient 
*c
, robj 
*o
, long long *target
, const char *msg
); 
 641 int getDoubleFromObjectOrReply(redisClient 
*c
, robj 
*o
, double *target
, const char *msg
); 
 642 int getLongLongFromObject(robj 
*o
, long long *target
); 
 643 char *strEncoding(int encoding
); 
 644 int compareStringObjects(robj 
*a
, robj 
*b
); 
 645 int equalStringObjects(robj 
*a
, robj 
*b
); 
 648 void replicationFeedSlaves(list 
*slaves
, int dictid
, robj 
**argv
, int argc
); 
 649 void replicationFeedMonitors(list 
*monitors
, int dictid
, robj 
**argv
, int argc
); 
 650 int syncWithMaster(void); 
 651 void updateSlavesWaitingBgsave(int bgsaveerr
); 
 653 /* RDB persistence */ 
 654 int rdbLoad(char *filename
); 
 655 int rdbSaveBackground(char *filename
); 
 656 void rdbRemoveTempFile(pid_t childpid
); 
 657 int rdbSave(char *filename
); 
 658 int rdbSaveObject(FILE *fp
, robj 
*o
); 
 659 off_t 
rdbSavedObjectPages(robj 
*o
, FILE *fp
); 
 660 off_t 
rdbSavedObjectLen(robj 
*o
, FILE *fp
); 
 661 robj 
*rdbLoadObject(int type
, FILE *fp
); 
 662 void backgroundSaveDoneHandler(int statloc
); 
 664 /* AOF persistence */ 
 665 void flushAppendOnlyFile(void); 
 666 void feedAppendOnlyFile(struct redisCommand 
*cmd
, int dictid
, robj 
**argv
, int argc
); 
 667 void aofRemoveTempFile(pid_t childpid
); 
 668 int rewriteAppendOnlyFileBackground(void); 
 669 int loadAppendOnlyFile(char *filename
); 
 670 void stopAppendOnly(void); 
 671 int startAppendOnly(void); 
 672 void backgroundRewriteDoneHandler(int statloc
); 
 674 /* Sorted sets data type */ 
 675 zskiplist 
*zslCreate(void); 
 676 void zslFree(zskiplist 
*zsl
); 
 677 void zslInsert(zskiplist 
*zsl
, double score
, robj 
*obj
); 
 680 void freeMemoryIfNeeded(void); 
 681 int processCommand(redisClient 
*c
); 
 682 void setupSigSegvAction(void); 
 683 struct redisCommand 
*lookupCommand(char *name
); 
 684 void call(redisClient 
*c
, struct redisCommand 
*cmd
); 
 685 int prepareForShutdown(); 
 686 void redisLog(int level
, const char *fmt
, ...); 
 688 void updateDictResizePolicy(void); 
 689 int htNeedsResize(dict 
*dict
); 
 690 void oom(const char *msg
); 
 694 void vmMarkPagesFree(off_t page
, off_t count
); 
 695 robj 
*vmLoadObject(robj 
*o
); 
 696 robj 
*vmPreviewObject(robj 
*o
); 
 697 int vmSwapOneObjectBlocking(void); 
 698 int vmSwapOneObjectThreaded(void); 
 699 int vmCanSwapOut(void); 
 700 void vmThreadedIOCompletedJob(aeEventLoop 
*el
, int fd
, void *privdata
, int mask
); 
 701 void vmCancelThreadedIOJob(robj 
*o
); 
 702 void lockThreadedIO(void); 
 703 void unlockThreadedIO(void); 
 704 int vmSwapObjectThreaded(robj 
*key
, robj 
*val
, redisDb 
*db
); 
 705 void freeIOJob(iojob 
*j
); 
 706 void queueIOJob(iojob 
*j
); 
 707 int vmWriteObjectOnSwap(robj 
*o
, off_t page
); 
 708 robj 
*vmReadObjectFromSwap(off_t page
, int type
); 
 709 void waitEmptyIOJobsQueue(void); 
 710 void vmReopenSwapFile(void); 
 711 int vmFreePage(off_t page
); 
 712 void zunionInterBlockClientOnSwappedKeys(redisClient 
*c
, struct redisCommand 
*cmd
, int argc
, robj 
**argv
); 
 713 void execBlockClientOnSwappedKeys(redisClient 
*c
, struct redisCommand 
*cmd
, int argc
, robj 
**argv
); 
 714 int blockClientOnSwappedKeys(redisClient 
*c
, struct redisCommand 
*cmd
); 
 715 int dontWaitForSwappedKey(redisClient 
*c
, robj 
*key
); 
 716 void handleClientsBlockedOnSwappedKey(redisDb 
*db
, robj 
*key
); 
 717 vmpointer 
*vmSwapObjectBlocking(robj 
*val
); 
 720 void convertToRealHash(robj 
*o
); 
 721 void hashTypeTryConversion(robj 
*subject
, robj 
**argv
, int start
, int end
); 
 722 void hashTypeTryObjectEncoding(robj 
*subject
, robj 
**o1
, robj 
**o2
); 
 723 robj 
*hashTypeGet(robj 
*o
, robj 
*key
); 
 724 int hashTypeExists(robj 
*o
, robj 
*key
); 
 725 int hashTypeSet(robj 
*o
, robj 
*key
, robj 
*value
); 
 726 int hashTypeDelete(robj 
*o
, robj 
*key
); 
 727 unsigned long hashTypeLength(robj 
*o
); 
 728 hashTypeIterator 
*hashTypeInitIterator(robj 
*subject
); 
 729 void hashTypeReleaseIterator(hashTypeIterator 
*hi
); 
 730 int hashTypeNext(hashTypeIterator 
*hi
); 
 731 robj 
*hashTypeCurrent(hashTypeIterator 
*hi
, int what
); 
 732 robj 
*hashTypeLookupWriteOrCreate(redisClient 
*c
, robj 
*key
); 
 735 int pubsubUnsubscribeAllChannels(redisClient 
*c
, int notify
); 
 736 int pubsubUnsubscribeAllPatterns(redisClient 
*c
, int notify
); 
 737 void freePubsubPattern(void *p
); 
 738 int listMatchPubsubPattern(void *a
, void *b
); 
 740 /* Utility functions */ 
 741 int stringmatchlen(const char *pattern
, int patternLen
, 
 742         const char *string
, int stringLen
, int nocase
); 
 743 int stringmatch(const char *pattern
, const char *string
, int nocase
); 
 744 long long memtoll(const char *p
, int *err
); 
 745 int ll2string(char *s
, size_t len
, long long value
); 
 746 int isStringRepresentableAsLong(sds s
, long *longval
); 
 749 void loadServerConfig(char *filename
); 
 750 void appendServerSaveParams(time_t seconds
, int changes
); 
 751 void resetServerSaveParams(); 
 753 /* db.c -- Keyspace access API */ 
 754 int removeExpire(redisDb 
*db
, robj 
*key
); 
 755 int expireIfNeeded(redisDb 
*db
, robj 
*key
); 
 756 int deleteIfVolatile(redisDb 
*db
, robj 
*key
); 
 757 time_t getExpire(redisDb 
*db
, robj 
*key
); 
 758 int setExpire(redisDb 
*db
, robj 
*key
, time_t when
); 
 759 robj 
*lookupKey(redisDb 
*db
, robj 
*key
); 
 760 robj 
*lookupKeyRead(redisDb 
*db
, robj 
*key
); 
 761 robj 
*lookupKeyWrite(redisDb 
*db
, robj 
*key
); 
 762 robj 
*lookupKeyReadOrReply(redisClient 
*c
, robj 
*key
, robj 
*reply
); 
 763 robj 
*lookupKeyWriteOrReply(redisClient 
*c
, robj 
*key
, robj 
*reply
); 
 764 int dbAdd(redisDb 
*db
, robj 
*key
, robj 
*val
); 
 765 int dbReplace(redisDb 
*db
, robj 
*key
, robj 
*val
); 
 766 int dbExists(redisDb 
*db
, robj 
*key
); 
 767 robj 
*dbRandomKey(redisDb 
*db
); 
 768 int dbDelete(redisDb 
*db
, robj 
*key
); 
 770 int selectDb(redisClient 
*c
, int id
); 
 773 char *redisGitSHA1(void); 
 774 char *redisGitDirty(void); 
 776 /* Commands prototypes */ 
 777 void authCommand(redisClient 
*c
); 
 778 void pingCommand(redisClient 
*c
); 
 779 void echoCommand(redisClient 
*c
); 
 780 void setCommand(redisClient 
*c
); 
 781 void setnxCommand(redisClient 
*c
); 
 782 void setexCommand(redisClient 
*c
); 
 783 void getCommand(redisClient 
*c
); 
 784 void delCommand(redisClient 
*c
); 
 785 void existsCommand(redisClient 
*c
); 
 786 void incrCommand(redisClient 
*c
); 
 787 void decrCommand(redisClient 
*c
); 
 788 void incrbyCommand(redisClient 
*c
); 
 789 void decrbyCommand(redisClient 
*c
); 
 790 void selectCommand(redisClient 
*c
); 
 791 void randomkeyCommand(redisClient 
*c
); 
 792 void keysCommand(redisClient 
*c
); 
 793 void dbsizeCommand(redisClient 
*c
); 
 794 void lastsaveCommand(redisClient 
*c
); 
 795 void saveCommand(redisClient 
*c
); 
 796 void bgsaveCommand(redisClient 
*c
); 
 797 void bgrewriteaofCommand(redisClient 
*c
); 
 798 void shutdownCommand(redisClient 
*c
); 
 799 void moveCommand(redisClient 
*c
); 
 800 void renameCommand(redisClient 
*c
); 
 801 void renamenxCommand(redisClient 
*c
); 
 802 void lpushCommand(redisClient 
*c
); 
 803 void rpushCommand(redisClient 
*c
); 
 804 void lpushxCommand(redisClient 
*c
); 
 805 void rpushxCommand(redisClient 
*c
); 
 806 void linsertCommand(redisClient 
*c
); 
 807 void lpopCommand(redisClient 
*c
); 
 808 void rpopCommand(redisClient 
*c
); 
 809 void llenCommand(redisClient 
*c
); 
 810 void lindexCommand(redisClient 
*c
); 
 811 void lrangeCommand(redisClient 
*c
); 
 812 void ltrimCommand(redisClient 
*c
); 
 813 void typeCommand(redisClient 
*c
); 
 814 void lsetCommand(redisClient 
*c
); 
 815 void saddCommand(redisClient 
*c
); 
 816 void sremCommand(redisClient 
*c
); 
 817 void smoveCommand(redisClient 
*c
); 
 818 void sismemberCommand(redisClient 
*c
); 
 819 void scardCommand(redisClient 
*c
); 
 820 void spopCommand(redisClient 
*c
); 
 821 void srandmemberCommand(redisClient 
*c
); 
 822 void sinterCommand(redisClient 
*c
); 
 823 void sinterstoreCommand(redisClient 
*c
); 
 824 void sunionCommand(redisClient 
*c
); 
 825 void sunionstoreCommand(redisClient 
*c
); 
 826 void sdiffCommand(redisClient 
*c
); 
 827 void sdiffstoreCommand(redisClient 
*c
); 
 828 void syncCommand(redisClient 
*c
); 
 829 void flushdbCommand(redisClient 
*c
); 
 830 void flushallCommand(redisClient 
*c
); 
 831 void sortCommand(redisClient 
*c
); 
 832 void lremCommand(redisClient 
*c
); 
 833 void rpoplpushcommand(redisClient 
*c
); 
 834 void infoCommand(redisClient 
*c
); 
 835 void mgetCommand(redisClient 
*c
); 
 836 void monitorCommand(redisClient 
*c
); 
 837 void expireCommand(redisClient 
*c
); 
 838 void expireatCommand(redisClient 
*c
); 
 839 void getsetCommand(redisClient 
*c
); 
 840 void ttlCommand(redisClient 
*c
); 
 841 void slaveofCommand(redisClient 
*c
); 
 842 void debugCommand(redisClient 
*c
); 
 843 void msetCommand(redisClient 
*c
); 
 844 void msetnxCommand(redisClient 
*c
); 
 845 void zaddCommand(redisClient 
*c
); 
 846 void zincrbyCommand(redisClient 
*c
); 
 847 void zrangeCommand(redisClient 
*c
); 
 848 void zrangebyscoreCommand(redisClient 
*c
); 
 849 void zcountCommand(redisClient 
*c
); 
 850 void zrevrangeCommand(redisClient 
*c
); 
 851 void zcardCommand(redisClient 
*c
); 
 852 void zremCommand(redisClient 
*c
); 
 853 void zscoreCommand(redisClient 
*c
); 
 854 void zremrangebyscoreCommand(redisClient 
*c
); 
 855 void multiCommand(redisClient 
*c
); 
 856 void execCommand(redisClient 
*c
); 
 857 void discardCommand(redisClient 
*c
); 
 858 void blpopCommand(redisClient 
*c
); 
 859 void brpopCommand(redisClient 
*c
); 
 860 void appendCommand(redisClient 
*c
); 
 861 void substrCommand(redisClient 
*c
); 
 862 void strlenCommand(redisClient 
*c
); 
 863 void zrankCommand(redisClient 
*c
); 
 864 void zrevrankCommand(redisClient 
*c
); 
 865 void hsetCommand(redisClient 
*c
); 
 866 void hsetnxCommand(redisClient 
*c
); 
 867 void hgetCommand(redisClient 
*c
); 
 868 void hmsetCommand(redisClient 
*c
); 
 869 void hmgetCommand(redisClient 
*c
); 
 870 void hdelCommand(redisClient 
*c
); 
 871 void hlenCommand(redisClient 
*c
); 
 872 void zremrangebyrankCommand(redisClient 
*c
); 
 873 void zunionstoreCommand(redisClient 
*c
); 
 874 void zinterstoreCommand(redisClient 
*c
); 
 875 void hkeysCommand(redisClient 
*c
); 
 876 void hvalsCommand(redisClient 
*c
); 
 877 void hgetallCommand(redisClient 
*c
); 
 878 void hexistsCommand(redisClient 
*c
); 
 879 void configCommand(redisClient 
*c
); 
 880 void hincrbyCommand(redisClient 
*c
); 
 881 void subscribeCommand(redisClient 
*c
); 
 882 void unsubscribeCommand(redisClient 
*c
); 
 883 void psubscribeCommand(redisClient 
*c
); 
 884 void punsubscribeCommand(redisClient 
*c
); 
 885 void publishCommand(redisClient 
*c
); 
 886 void watchCommand(redisClient 
*c
); 
 887 void unwatchCommand(redisClient 
*c
); 
 889 #if defined(__GNUC__) 
 890 void *calloc(size_t count
, size_t size
) __attribute__ ((deprecated
)); 
 891 void free(void *ptr
) __attribute__ ((deprecated
)); 
 892 void *malloc(size_t size
) __attribute__ ((deprecated
)); 
 893 void *realloc(void *ptr
, size_t size
) __attribute__ ((deprecated
));