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 */
29 #include "intset.h" /* Compact integer set structure */
36 /* Static server configuration */
37 #define REDIS_SERVERPORT 6379 /* TCP port */
38 #define REDIS_MAXIDLETIME (60*5) /* default client timeout */
39 #define REDIS_IOBUF_LEN 1024
40 #define REDIS_LOADBUF_LEN 1024
41 #define REDIS_STATIC_ARGS 8
42 #define REDIS_DEFAULT_DBNUM 16
43 #define REDIS_CONFIGLINE_MAX 1024
44 #define REDIS_OBJFREELIST_MAX 1000000 /* Max number of objects to cache */
45 #define REDIS_MAX_SYNC_TIME 60 /* Slave can't take more to sync */
46 #define REDIS_EXPIRELOOKUPS_PER_CRON 10 /* lookup 10 expires per loop */
47 #define REDIS_MAX_WRITE_PER_EVENT (1024*64)
48 #define REDIS_REQUEST_MAX_SIZE (1024*1024*256) /* max bytes in inline command */
49 #define REDIS_SHARED_INTEGERS 10000
50 #define REDIS_REPLY_CHUNK_BYTES (5*1500) /* 5 TCP packets with default MTU */
52 /* If more then REDIS_WRITEV_THRESHOLD write packets are pending use writev */
53 #define REDIS_WRITEV_THRESHOLD 3
54 /* Max number of iovecs used for each writev call */
55 #define REDIS_WRITEV_IOVEC_COUNT 256
57 /* Hash table parameters */
58 #define REDIS_HT_MINFILL 10 /* Minimal hash table fill 10% */
61 #define REDIS_CMD_BULK 1 /* Bulk write command */
62 #define REDIS_CMD_INLINE 2 /* Inline command */
63 /* REDIS_CMD_DENYOOM reserves a longer comment: all the commands marked with
64 this flags will return an error when the 'maxmemory' option is set in the
65 config file and the server is using more than maxmemory bytes of memory.
66 In short this commands are denied on low memory conditions. */
67 #define REDIS_CMD_DENYOOM 4
68 #define REDIS_CMD_FORCE_REPLICATION 8 /* Force replication even if dirty is 0 */
71 #define REDIS_STRING 0
76 #define REDIS_VMPOINTER 8
78 /* Objects encoding. Some kind of objects like Strings and Hashes can be
79 * internally represented in multiple ways. The 'encoding' field of the object
80 * is set to one of this fields for this object. */
81 #define REDIS_ENCODING_RAW 0 /* Raw representation */
82 #define REDIS_ENCODING_INT 1 /* Encoded as integer */
83 #define REDIS_ENCODING_HT 2 /* Encoded as hash table */
84 #define REDIS_ENCODING_ZIPMAP 3 /* Encoded as zipmap */
85 #define REDIS_ENCODING_LINKEDLIST 4 /* Encoded as regular linked list */
86 #define REDIS_ENCODING_ZIPLIST 5 /* Encoded as ziplist */
87 #define REDIS_ENCODING_INTSET 6 /* Encoded as intset */
89 /* Object types only used for dumping to disk */
90 #define REDIS_EXPIRETIME 253
91 #define REDIS_SELECTDB 254
94 /* Defines related to the dump file format. To store 32 bits lengths for short
95 * keys requires a lot of space, so we check the most significant 2 bits of
96 * the first byte to interpreter the length:
98 * 00|000000 => if the two MSB are 00 the len is the 6 bits of this byte
99 * 01|000000 00000000 => 01, the len is 14 byes, 6 bits + 8 bits of next byte
100 * 10|000000 [32 bit integer] => if it's 01, a full 32 bit len will follow
101 * 11|000000 this means: specially encoded object will follow. The six bits
102 * number specify the kind of object that follows.
103 * See the REDIS_RDB_ENC_* defines.
105 * Lenghts up to 63 are stored using a single byte, most DB keys, and may
106 * values, will fit inside. */
107 #define REDIS_RDB_6BITLEN 0
108 #define REDIS_RDB_14BITLEN 1
109 #define REDIS_RDB_32BITLEN 2
110 #define REDIS_RDB_ENCVAL 3
111 #define REDIS_RDB_LENERR UINT_MAX
113 /* When a length of a string object stored on disk has the first two bits
114 * set, the remaining two bits specify a special encoding for the object
115 * accordingly to the following defines: */
116 #define REDIS_RDB_ENC_INT8 0 /* 8 bit signed integer */
117 #define REDIS_RDB_ENC_INT16 1 /* 16 bit signed integer */
118 #define REDIS_RDB_ENC_INT32 2 /* 32 bit signed integer */
119 #define REDIS_RDB_ENC_LZF 3 /* string compressed with FASTLZ */
121 /* Virtual memory object->where field. */
122 #define REDIS_VM_MEMORY 0 /* The object is on memory */
123 #define REDIS_VM_SWAPPED 1 /* The object is on disk */
124 #define REDIS_VM_SWAPPING 2 /* Redis is swapping this object on disk */
125 #define REDIS_VM_LOADING 3 /* Redis is loading this object from disk */
127 /* Virtual memory static configuration stuff.
128 * Check vmFindContiguousPages() to know more about this magic numbers. */
129 #define REDIS_VM_MAX_NEAR_PAGES 65536
130 #define REDIS_VM_MAX_RANDOM_JUMP 4096
131 #define REDIS_VM_MAX_THREADS 32
132 #define REDIS_THREAD_STACK_SIZE (1024*1024*4)
133 /* The following is the *percentage* of completed I/O jobs to process when the
134 * handelr is called. While Virtual Memory I/O operations are performed by
135 * threads, this operations must be processed by the main thread when completed
136 * in order to take effect. */
137 #define REDIS_MAX_COMPLETED_JOBS_PROCESSED 1
140 #define REDIS_SLAVE 1 /* This client is a slave server */
141 #define REDIS_MASTER 2 /* This client is a master server */
142 #define REDIS_MONITOR 4 /* This client is a slave monitor, see MONITOR */
143 #define REDIS_MULTI 8 /* This client is in a MULTI context */
144 #define REDIS_BLOCKED 16 /* The client is waiting in a blocking operation */
145 #define REDIS_IO_WAIT 32 /* The client is waiting for Virtual Memory I/O */
146 #define REDIS_DIRTY_CAS 64 /* Watched keys modified. EXEC will fail. */
147 #define REDIS_QUIT 128 /* Client will be disconnected after reply is sent */
149 /* Slave replication state - slave side */
150 #define REDIS_REPL_NONE 0 /* No active replication */
151 #define REDIS_REPL_CONNECT 1 /* Must connect to master */
152 #define REDIS_REPL_CONNECTED 2 /* Connected to master */
154 /* Slave replication state - from the point of view of master
155 * Note that in SEND_BULK and ONLINE state the slave receives new updates
156 * in its output queue. In the WAIT_BGSAVE state instead the server is waiting
157 * to start the next background saving in order to send updates to it. */
158 #define REDIS_REPL_WAIT_BGSAVE_START 3 /* master waits bgsave to start feeding it */
159 #define REDIS_REPL_WAIT_BGSAVE_END 4 /* master waits bgsave to start bulk DB transmission */
160 #define REDIS_REPL_SEND_BULK 5 /* master is sending the bulk DB */
161 #define REDIS_REPL_ONLINE 6 /* bulk DB already transmitted, receive updates */
163 /* List related stuff */
167 /* Sort operations */
168 #define REDIS_SORT_GET 0
169 #define REDIS_SORT_ASC 1
170 #define REDIS_SORT_DESC 2
171 #define REDIS_SORTKEY_MAX 1024
174 #define REDIS_DEBUG 0
175 #define REDIS_VERBOSE 1
176 #define REDIS_NOTICE 2
177 #define REDIS_WARNING 3
179 /* Anti-warning macro... */
180 #define REDIS_NOTUSED(V) ((void) V)
182 #define ZSKIPLIST_MAXLEVEL 32 /* Should be enough for 2^32 elements */
183 #define ZSKIPLIST_P 0.25 /* Skiplist P = 1/4 */
185 /* Append only defines */
186 #define APPENDFSYNC_NO 0
187 #define APPENDFSYNC_ALWAYS 1
188 #define APPENDFSYNC_EVERYSEC 2
190 /* Zip structure related defaults */
191 #define REDIS_HASH_MAX_ZIPMAP_ENTRIES 64
192 #define REDIS_HASH_MAX_ZIPMAP_VALUE 512
193 #define REDIS_LIST_MAX_ZIPLIST_ENTRIES 1024
194 #define REDIS_LIST_MAX_ZIPLIST_VALUE 32
195 #define REDIS_SET_MAX_INTSET_ENTRIES 4096
197 /* Sets operations codes */
198 #define REDIS_OP_UNION 0
199 #define REDIS_OP_DIFF 1
200 #define REDIS_OP_INTER 2
202 /* We can print the stacktrace, so our assert is defined this way: */
203 #define redisAssert(_e) ((_e)?(void)0 : (_redisAssert(#_e,__FILE__,__LINE__),_exit(1)))
204 #define redisPanic(_e) _redisPanic(#_e,__FILE__,__LINE__),_exit(1)
205 void _redisAssert(char *estr
, char *file
, int line
);
206 void _redisPanic(char *msg
, char *file
, int line
);
208 /*-----------------------------------------------------------------------------
210 *----------------------------------------------------------------------------*/
212 /* A redis object, that is a type able to hold a string / list / set */
214 /* The actual Redis Object */
215 typedef struct redisObject
{
217 unsigned storage
:2; /* REDIS_VM_MEMORY or REDIS_VM_SWAPPING */
219 unsigned lru
:22; /* lru time (relative to server.lruclock) */
222 /* VM fields are only allocated if VM is active, otherwise the
223 * object allocation function will just allocate
224 * sizeof(redisObjct) minus sizeof(redisObjectVM), so using
225 * Redis without VM active will not have any overhead. */
228 /* The VM pointer structure - identifies an object in the swap file.
230 * This object is stored in place of the value
231 * object in the main key->value hash table representing a database.
232 * Note that the first fields (type, storage) are the same as the redisObject
233 * structure so that vmPointer strucuters can be accessed even when casted
234 * as redisObject structures.
236 * This is useful as we don't know if a value object is or not on disk, but we
237 * are always able to read obj->storage to check this. For vmPointer
238 * structures "type" is set to REDIS_VMPOINTER (even if without this field
239 * is still possible to check the kind of object from the value of 'storage').*/
240 typedef struct vmPointer
{
242 unsigned storage
:2; /* REDIS_VM_SWAPPED or REDIS_VM_LOADING */
244 unsigned int vtype
; /* type of the object stored in the swap file */
245 off_t page
; /* the page at witch the object is stored on disk */
246 off_t usedpages
; /* number of pages used on disk */
249 /* Macro used to initalize a Redis object allocated on the stack.
250 * Note that this macro is taken near the structure definition to make sure
251 * we'll update it when the structure is changed, to avoid bugs like
252 * bug #85 introduced exactly in this way. */
253 #define initStaticStringObject(_var,_ptr) do { \
255 _var.type = REDIS_STRING; \
256 _var.encoding = REDIS_ENCODING_RAW; \
258 _var.storage = REDIS_VM_MEMORY; \
261 typedef struct redisDb
{
262 dict
*dict
; /* The keyspace for this DB */
263 dict
*expires
; /* Timeout of keys with a timeout set */
264 dict
*blocking_keys
; /* Keys with clients waiting for data (BLPOP) */
265 dict
*io_keys
; /* Keys with clients waiting for VM I/O */
266 dict
*watched_keys
; /* WATCHED keys for MULTI/EXEC CAS */
270 /* Client MULTI/EXEC state */
271 typedef struct multiCmd
{
274 struct redisCommand
*cmd
;
277 typedef struct multiState
{
278 multiCmd
*commands
; /* Array of MULTI commands */
279 int count
; /* Total number of MULTI commands */
282 /* With multiplexing we need to take per-clinet state.
283 * Clients are taken in a liked list. */
284 typedef struct redisClient
{
289 robj
**argv
, **mbargv
;
290 char *newline
; /* pointing to the detected newline in querybuf */
292 long bulklen
; /* bulk read len. -1 if not in bulk read mode */
293 int multibulk
; /* multi bulk command format active */
296 time_t lastinteraction
; /* time of the last interaction, used for timeout */
297 int flags
; /* REDIS_SLAVE | REDIS_MONITOR | REDIS_MULTI ... */
298 int slaveseldb
; /* slave selected db, if this client is a slave */
299 int authenticated
; /* when requirepass is non-NULL */
300 int replstate
; /* replication state if this is a slave */
301 int repldbfd
; /* replication DB file descriptor */
302 long repldboff
; /* replication DB file offset */
303 off_t repldbsize
; /* replication DB file size */
304 multiState mstate
; /* MULTI/EXEC state */
305 robj
**blocking_keys
; /* The key we are waiting to terminate a blocking
306 * operation such as BLPOP. Otherwise NULL. */
307 int blocking_keys_num
; /* Number of blocking keys */
308 time_t blockingto
; /* Blocking operation timeout. If UNIX current time
309 * is >= blockingto then the operation timed out. */
310 list
*io_keys
; /* Keys this client is waiting to be loaded from the
311 * swap file in order to continue. */
312 list
*watched_keys
; /* Keys WATCHED for MULTI/EXEC CAS */
313 dict
*pubsub_channels
; /* channels a client is interested in (SUBSCRIBE) */
314 list
*pubsub_patterns
; /* patterns a client is interested in (SUBSCRIBE) */
316 /* Response buffer */
318 char buf
[REDIS_REPLY_CHUNK_BYTES
];
326 struct sharedObjectsStruct
{
327 robj
*crlf
, *ok
, *err
, *emptybulk
, *czero
, *cone
, *cnegone
, *pong
, *space
,
328 *colon
, *nullbulk
, *nullmultibulk
, *queued
,
329 *emptymultibulk
, *wrongtypeerr
, *nokeyerr
, *syntaxerr
, *sameobjecterr
,
330 *outofrangeerr
, *plus
,
331 *select0
, *select1
, *select2
, *select3
, *select4
,
332 *select5
, *select6
, *select7
, *select8
, *select9
,
333 *messagebulk
, *pmessagebulk
, *subscribebulk
, *unsubscribebulk
, *mbulk3
,
334 *mbulk4
, *psubscribebulk
, *punsubscribebulk
,
335 *integers
[REDIS_SHARED_INTEGERS
];
338 /* Global server state structure */
340 pthread_t mainthread
;
344 long long dirty
; /* changes to DB from the last save */
345 long long dirty_before_bgsave
; /* used to restore dirty on failed BGSAVE */
347 list
*slaves
, *monitors
;
348 char neterr
[ANET_ERR_LEN
];
350 int cronloops
; /* number of times the cron function run */
351 list
*objfreelist
; /* A list of freed objects to avoid malloc() */
352 time_t lastsave
; /* Unix time of last save succeeede */
353 /* Fields used only for stats */
354 time_t stat_starttime
; /* server start time */
355 long long stat_numcommands
; /* number of processed commands */
356 long long stat_numconnections
; /* number of connections received */
357 long long stat_expiredkeys
; /* number of expired keys */
366 int no_appendfsync_on_rewrite
;
372 pid_t bgsavechildpid
;
373 pid_t bgrewritechildpid
;
374 sds bgrewritebuf
; /* buffer taken by parent during oppend only rewrite */
375 sds aofbuf
; /* AOF buffer, written before entering the event loop */
376 struct saveparam
*saveparams
;
381 char *appendfilename
;
385 /* Replication related */
390 redisClient
*master
; /* client that is master for this slave */
392 unsigned int maxclients
;
393 unsigned long long maxmemory
;
394 unsigned int blpop_blocked_clients
;
395 unsigned int vm_blocked_clients
;
396 /* Sort parameters - qsort_r() is only available under BSD so we
397 * have to take this state global, in order to pass it to sortCompare() */
401 /* Virtual memory configuration */
406 unsigned long long vm_max_memory
;
407 /* Zip structure config */
408 size_t hash_max_zipmap_entries
;
409 size_t hash_max_zipmap_value
;
410 size_t list_max_ziplist_entries
;
411 size_t list_max_ziplist_value
;
412 size_t set_max_intset_entries
;
413 /* Virtual memory state */
416 off_t vm_next_page
; /* Next probably empty page */
417 off_t vm_near_pages
; /* Number of pages allocated sequentially */
418 unsigned char *vm_bitmap
; /* Bitmap of free/used pages */
419 time_t unixtime
; /* Unix time sampled every second. */
420 /* Virtual memory I/O threads stuff */
421 /* An I/O thread process an element taken from the io_jobs queue and
422 * put the result of the operation in the io_done list. While the
423 * job is being processed, it's put on io_processing queue. */
424 list
*io_newjobs
; /* List of VM I/O jobs yet to be processed */
425 list
*io_processing
; /* List of VM I/O jobs being processed */
426 list
*io_processed
; /* List of VM I/O jobs already processed */
427 list
*io_ready_clients
; /* Clients ready to be unblocked. All keys loaded */
428 pthread_mutex_t io_mutex
; /* lock to access io_jobs/io_done/io_thread_job */
429 pthread_mutex_t obj_freelist_mutex
; /* safe redis objects creation/free */
430 pthread_mutex_t io_swapfile_mutex
; /* So we can lseek + write */
431 pthread_attr_t io_threads_attr
; /* attributes for threads creation */
432 int io_active_threads
; /* Number of running I/O threads */
433 int vm_max_threads
; /* Max number of I/O threads running at the same time */
434 /* Our main thread is blocked on the event loop, locking for sockets ready
435 * to be read or written, so when a threaded I/O operation is ready to be
436 * processed by the main thread, the I/O thread will use a unix pipe to
437 * awake the main thread. The followings are the two pipe FDs. */
438 int io_ready_pipe_read
;
439 int io_ready_pipe_write
;
440 /* Virtual memory stats */
441 unsigned long long vm_stats_used_pages
;
442 unsigned long long vm_stats_swapped_objects
;
443 unsigned long long vm_stats_swapouts
;
444 unsigned long long vm_stats_swapins
;
446 dict
*pubsub_channels
; /* Map channels to list of subscribed clients */
447 list
*pubsub_patterns
; /* A list of pubsub_patterns */
450 unsigned lruclock
:22; /* clock incrementing every minute, for LRU */
451 unsigned lruclock_padding
:10;
454 typedef struct pubsubPattern
{
459 typedef void redisCommandProc(redisClient
*c
);
460 typedef void redisVmPreloadProc(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
461 struct redisCommand
{
463 redisCommandProc
*proc
;
466 /* Use a function to determine which keys need to be loaded
467 * in the background prior to executing this command. Takes precedence
468 * over vm_firstkey and others, ignored when NULL */
469 redisVmPreloadProc
*vm_preload_proc
;
470 /* What keys should be loaded in background when calling this command? */
471 int vm_firstkey
; /* The first argument that's a key (0 = no keys) */
472 int vm_lastkey
; /* THe last argument that's a key */
473 int vm_keystep
; /* The step between first and last key */
476 struct redisFunctionSym
{
478 unsigned long pointer
;
481 typedef struct _redisSortObject
{
489 typedef struct _redisSortOperation
{
492 } redisSortOperation
;
494 /* ZSETs use a specialized version of Skiplists */
495 typedef struct zskiplistNode
{
498 struct zskiplistNode
*backward
;
499 struct zskiplistLevel
{
500 struct zskiplistNode
*forward
;
505 typedef struct zskiplist
{
506 struct zskiplistNode
*header
, *tail
;
507 unsigned long length
;
511 typedef struct zset
{
516 /* VM threaded I/O request message */
517 #define REDIS_IOJOB_LOAD 0 /* Load from disk to memory */
518 #define REDIS_IOJOB_PREPARE_SWAP 1 /* Compute needed pages */
519 #define REDIS_IOJOB_DO_SWAP 2 /* Swap from memory to disk */
520 typedef struct iojob
{
521 int type
; /* Request type, REDIS_IOJOB_* */
522 redisDb
*db
;/* Redis database */
523 robj
*key
; /* This I/O request is about swapping this key */
524 robj
*id
; /* Unique identifier of this job:
525 this is the object to swap for REDIS_IOREQ_*_SWAP, or the
526 vmpointer objct for REDIS_IOREQ_LOAD. */
527 robj
*val
; /* the value to swap for REDIS_IOREQ_*_SWAP, otherwise this
528 * field is populated by the I/O thread for REDIS_IOREQ_LOAD. */
529 off_t page
; /* Swap page where to read/write the object */
530 off_t pages
; /* Swap pages needed to save object. PREPARE_SWAP return val */
531 int canceled
; /* True if this command was canceled by blocking side of VM */
532 pthread_t thread
; /* ID of the thread processing this entry */
535 /* Structure to hold list iteration abstraction. */
538 unsigned char encoding
;
539 unsigned char direction
; /* Iteration direction */
544 /* Structure for an entry while iterating over a list. */
546 listTypeIterator
*li
;
547 unsigned char *zi
; /* Entry in ziplist */
548 listNode
*ln
; /* Entry in linked list */
551 /* Structure to hold set iteration abstraction. */
555 int ii
; /* intset iterator */
559 /* Structure to hold hash iteration abstration. Note that iteration over
560 * hashes involves both fields and values. Because it is possible that
561 * not both are required, store pointers in the iterator to avoid
562 * unnecessary memory allocation for fields/values. */
566 unsigned char *zk
, *zv
;
567 unsigned int zklen
, zvlen
;
573 #define REDIS_HASH_KEY 1
574 #define REDIS_HASH_VALUE 2
576 /*-----------------------------------------------------------------------------
577 * Extern declarations
578 *----------------------------------------------------------------------------*/
580 extern struct redisServer server
;
581 extern struct sharedObjectsStruct shared
;
582 extern dictType setDictType
;
583 extern dictType zsetDictType
;
584 extern double R_Zero
, R_PosInf
, R_NegInf
, R_Nan
;
585 dictType hashDictType
;
587 /*-----------------------------------------------------------------------------
588 * Functions prototypes
589 *----------------------------------------------------------------------------*/
591 /* networking.c -- Networking and Client related operations */
592 redisClient
*createClient(int fd
);
593 void closeTimedoutClients(void);
594 void freeClient(redisClient
*c
);
595 void resetClient(redisClient
*c
);
596 void sendReplyToClient(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
597 void sendReplyToClientWritev(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
598 void addReply(redisClient
*c
, robj
*obj
);
599 void *addDeferredMultiBulkLength(redisClient
*c
);
600 void setDeferredMultiBulkLength(redisClient
*c
, void *node
, long length
);
601 void addReplySds(redisClient
*c
, sds s
);
602 void processInputBuffer(redisClient
*c
);
603 void acceptHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
604 void readQueryFromClient(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
605 void addReplyBulk(redisClient
*c
, robj
*obj
);
606 void addReplyBulkCString(redisClient
*c
, char *s
);
607 void acceptHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
608 void addReply(redisClient
*c
, robj
*obj
);
609 void addReplySds(redisClient
*c
, sds s
);
610 void addReplyError(redisClient
*c
, char *err
);
611 void addReplyStatus(redisClient
*c
, char *status
);
612 void addReplyDouble(redisClient
*c
, double d
);
613 void addReplyLongLong(redisClient
*c
, long long ll
);
614 void addReplyMultiBulkLen(redisClient
*c
, long length
);
615 void *dupClientReplyValue(void *o
);
618 void addReplyErrorFormat(redisClient
*c
, const char *fmt
, ...)
619 __attribute__((format(printf
, 2, 3)));
620 void addReplyStatusFormat(redisClient
*c
, const char *fmt
, ...)
621 __attribute__((format(printf
, 2, 3)));
623 void addReplyErrorFormat(redisClient
*c
, const char *fmt
, ...);
624 void addReplyStatusFormat(redisClient
*c
, const char *fmt
, ...);
628 void listTypeTryConversion(robj
*subject
, robj
*value
);
629 void listTypePush(robj
*subject
, robj
*value
, int where
);
630 robj
*listTypePop(robj
*subject
, int where
);
631 unsigned long listTypeLength(robj
*subject
);
632 listTypeIterator
*listTypeInitIterator(robj
*subject
, int index
, unsigned char direction
);
633 void listTypeReleaseIterator(listTypeIterator
*li
);
634 int listTypeNext(listTypeIterator
*li
, listTypeEntry
*entry
);
635 robj
*listTypeGet(listTypeEntry
*entry
);
636 void listTypeInsert(listTypeEntry
*entry
, robj
*value
, int where
);
637 int listTypeEqual(listTypeEntry
*entry
, robj
*o
);
638 void listTypeDelete(listTypeEntry
*entry
);
639 void listTypeConvert(robj
*subject
, int enc
);
640 void unblockClientWaitingData(redisClient
*c
);
641 int handleClientsWaitingListPush(redisClient
*c
, robj
*key
, robj
*ele
);
642 void popGenericCommand(redisClient
*c
, int where
);
644 /* MULTI/EXEC/WATCH... */
645 void unwatchAllKeys(redisClient
*c
);
646 void initClientMultiState(redisClient
*c
);
647 void freeClientMultiState(redisClient
*c
);
648 void queueMultiCommand(redisClient
*c
, struct redisCommand
*cmd
);
649 void touchWatchedKey(redisDb
*db
, robj
*key
);
650 void touchWatchedKeysOnFlush(int dbid
);
652 /* Redis object implementation */
653 void decrRefCount(void *o
);
654 void incrRefCount(robj
*o
);
655 void freeStringObject(robj
*o
);
656 void freeListObject(robj
*o
);
657 void freeSetObject(robj
*o
);
658 void freeZsetObject(robj
*o
);
659 void freeHashObject(robj
*o
);
660 robj
*createObject(int type
, void *ptr
);
661 robj
*createStringObject(char *ptr
, size_t len
);
662 robj
*dupStringObject(robj
*o
);
663 robj
*tryObjectEncoding(robj
*o
);
664 robj
*getDecodedObject(robj
*o
);
665 size_t stringObjectLen(robj
*o
);
666 int tryFreeOneObjectFromFreelist(void);
667 robj
*createStringObjectFromLongLong(long long value
);
668 robj
*createListObject(void);
669 robj
*createZiplistObject(void);
670 robj
*createSetObject(void);
671 robj
*createIntsetObject(void);
672 robj
*createHashObject(void);
673 robj
*createZsetObject(void);
674 int getLongFromObjectOrReply(redisClient
*c
, robj
*o
, long *target
, const char *msg
);
675 int checkType(redisClient
*c
, robj
*o
, int type
);
676 int getLongLongFromObjectOrReply(redisClient
*c
, robj
*o
, long long *target
, const char *msg
);
677 int getDoubleFromObjectOrReply(redisClient
*c
, robj
*o
, double *target
, const char *msg
);
678 int getLongLongFromObject(robj
*o
, long long *target
);
679 char *strEncoding(int encoding
);
680 int compareStringObjects(robj
*a
, robj
*b
);
681 int equalStringObjects(robj
*a
, robj
*b
);
684 void replicationFeedSlaves(list
*slaves
, int dictid
, robj
**argv
, int argc
);
685 void replicationFeedMonitors(list
*monitors
, int dictid
, robj
**argv
, int argc
);
686 int syncWithMaster(void);
687 void updateSlavesWaitingBgsave(int bgsaveerr
);
689 /* RDB persistence */
690 int rdbLoad(char *filename
);
691 int rdbSaveBackground(char *filename
);
692 void rdbRemoveTempFile(pid_t childpid
);
693 int rdbSave(char *filename
);
694 int rdbSaveObject(FILE *fp
, robj
*o
);
695 off_t
rdbSavedObjectPages(robj
*o
, FILE *fp
);
696 off_t
rdbSavedObjectLen(robj
*o
, FILE *fp
);
697 robj
*rdbLoadObject(int type
, FILE *fp
);
698 void backgroundSaveDoneHandler(int statloc
);
700 /* AOF persistence */
701 void flushAppendOnlyFile(void);
702 void feedAppendOnlyFile(struct redisCommand
*cmd
, int dictid
, robj
**argv
, int argc
);
703 void aofRemoveTempFile(pid_t childpid
);
704 int rewriteAppendOnlyFileBackground(void);
705 int loadAppendOnlyFile(char *filename
);
706 void stopAppendOnly(void);
707 int startAppendOnly(void);
708 void backgroundRewriteDoneHandler(int statloc
);
710 /* Sorted sets data type */
711 zskiplist
*zslCreate(void);
712 void zslFree(zskiplist
*zsl
);
713 zskiplistNode
*zslInsert(zskiplist
*zsl
, double score
, robj
*obj
);
716 void freeMemoryIfNeeded(void);
717 int processCommand(redisClient
*c
);
718 void setupSigSegvAction(void);
719 struct redisCommand
*lookupCommand(char *name
);
720 void call(redisClient
*c
, struct redisCommand
*cmd
);
721 int prepareForShutdown();
722 void redisLog(int level
, const char *fmt
, ...);
724 void updateDictResizePolicy(void);
725 int htNeedsResize(dict
*dict
);
726 void oom(const char *msg
);
730 void vmMarkPagesFree(off_t page
, off_t count
);
731 robj
*vmLoadObject(robj
*o
);
732 robj
*vmPreviewObject(robj
*o
);
733 int vmSwapOneObjectBlocking(void);
734 int vmSwapOneObjectThreaded(void);
735 int vmCanSwapOut(void);
736 void vmThreadedIOCompletedJob(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
737 void vmCancelThreadedIOJob(robj
*o
);
738 void lockThreadedIO(void);
739 void unlockThreadedIO(void);
740 int vmSwapObjectThreaded(robj
*key
, robj
*val
, redisDb
*db
);
741 void freeIOJob(iojob
*j
);
742 void queueIOJob(iojob
*j
);
743 int vmWriteObjectOnSwap(robj
*o
, off_t page
);
744 robj
*vmReadObjectFromSwap(off_t page
, int type
);
745 void waitEmptyIOJobsQueue(void);
746 void vmReopenSwapFile(void);
747 int vmFreePage(off_t page
);
748 void zunionInterBlockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
749 void execBlockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
750 int blockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
);
751 int dontWaitForSwappedKey(redisClient
*c
, robj
*key
);
752 void handleClientsBlockedOnSwappedKey(redisDb
*db
, robj
*key
);
753 vmpointer
*vmSwapObjectBlocking(robj
*val
);
756 robj
*setTypeCreate(robj
*value
);
757 int setTypeAdd(robj
*subject
, robj
*value
);
758 int setTypeRemove(robj
*subject
, robj
*value
);
759 int setTypeIsMember(robj
*subject
, robj
*value
);
760 setTypeIterator
*setTypeInitIterator(robj
*subject
);
761 void setTypeReleaseIterator(setTypeIterator
*si
);
762 robj
*setTypeNext(setTypeIterator
*si
);
763 robj
*setTypeRandomElement(robj
*subject
);
764 unsigned long setTypeSize(robj
*subject
);
765 void setTypeConvert(robj
*subject
, int enc
);
768 void convertToRealHash(robj
*o
);
769 void hashTypeTryConversion(robj
*subject
, robj
**argv
, int start
, int end
);
770 void hashTypeTryObjectEncoding(robj
*subject
, robj
**o1
, robj
**o2
);
771 robj
*hashTypeGet(robj
*o
, robj
*key
);
772 int hashTypeExists(robj
*o
, robj
*key
);
773 int hashTypeSet(robj
*o
, robj
*key
, robj
*value
);
774 int hashTypeDelete(robj
*o
, robj
*key
);
775 unsigned long hashTypeLength(robj
*o
);
776 hashTypeIterator
*hashTypeInitIterator(robj
*subject
);
777 void hashTypeReleaseIterator(hashTypeIterator
*hi
);
778 int hashTypeNext(hashTypeIterator
*hi
);
779 robj
*hashTypeCurrent(hashTypeIterator
*hi
, int what
);
780 robj
*hashTypeLookupWriteOrCreate(redisClient
*c
, robj
*key
);
783 int pubsubUnsubscribeAllChannels(redisClient
*c
, int notify
);
784 int pubsubUnsubscribeAllPatterns(redisClient
*c
, int notify
);
785 void freePubsubPattern(void *p
);
786 int listMatchPubsubPattern(void *a
, void *b
);
788 /* Utility functions */
789 int stringmatchlen(const char *pattern
, int patternLen
,
790 const char *string
, int stringLen
, int nocase
);
791 int stringmatch(const char *pattern
, const char *string
, int nocase
);
792 long long memtoll(const char *p
, int *err
);
793 int ll2string(char *s
, size_t len
, long long value
);
794 int isStringRepresentableAsLong(sds s
, long *longval
);
795 int isStringRepresentableAsLongLong(sds s
, long long *longval
);
796 int isObjectRepresentableAsLongLong(robj
*o
, long long *llongval
);
799 void loadServerConfig(char *filename
);
800 void appendServerSaveParams(time_t seconds
, int changes
);
801 void resetServerSaveParams();
803 /* db.c -- Keyspace access API */
804 int removeExpire(redisDb
*db
, robj
*key
);
805 void propagateExpire(redisDb
*db
, robj
*key
);
806 int expireIfNeeded(redisDb
*db
, robj
*key
);
807 time_t getExpire(redisDb
*db
, robj
*key
);
808 void setExpire(redisDb
*db
, robj
*key
, time_t when
);
809 robj
*lookupKey(redisDb
*db
, robj
*key
);
810 robj
*lookupKeyRead(redisDb
*db
, robj
*key
);
811 robj
*lookupKeyWrite(redisDb
*db
, robj
*key
);
812 robj
*lookupKeyReadOrReply(redisClient
*c
, robj
*key
, robj
*reply
);
813 robj
*lookupKeyWriteOrReply(redisClient
*c
, robj
*key
, robj
*reply
);
814 int dbAdd(redisDb
*db
, robj
*key
, robj
*val
);
815 int dbReplace(redisDb
*db
, robj
*key
, robj
*val
);
816 int dbExists(redisDb
*db
, robj
*key
);
817 robj
*dbRandomKey(redisDb
*db
);
818 int dbDelete(redisDb
*db
, robj
*key
);
820 int selectDb(redisClient
*c
, int id
);
823 char *redisGitSHA1(void);
824 char *redisGitDirty(void);
826 /* Commands prototypes */
827 void authCommand(redisClient
*c
);
828 void pingCommand(redisClient
*c
);
829 void echoCommand(redisClient
*c
);
830 void setCommand(redisClient
*c
);
831 void setnxCommand(redisClient
*c
);
832 void setexCommand(redisClient
*c
);
833 void getCommand(redisClient
*c
);
834 void delCommand(redisClient
*c
);
835 void existsCommand(redisClient
*c
);
836 void incrCommand(redisClient
*c
);
837 void decrCommand(redisClient
*c
);
838 void incrbyCommand(redisClient
*c
);
839 void decrbyCommand(redisClient
*c
);
840 void selectCommand(redisClient
*c
);
841 void randomkeyCommand(redisClient
*c
);
842 void keysCommand(redisClient
*c
);
843 void dbsizeCommand(redisClient
*c
);
844 void lastsaveCommand(redisClient
*c
);
845 void saveCommand(redisClient
*c
);
846 void bgsaveCommand(redisClient
*c
);
847 void bgrewriteaofCommand(redisClient
*c
);
848 void shutdownCommand(redisClient
*c
);
849 void moveCommand(redisClient
*c
);
850 void renameCommand(redisClient
*c
);
851 void renamenxCommand(redisClient
*c
);
852 void lpushCommand(redisClient
*c
);
853 void rpushCommand(redisClient
*c
);
854 void lpushxCommand(redisClient
*c
);
855 void rpushxCommand(redisClient
*c
);
856 void linsertCommand(redisClient
*c
);
857 void lpopCommand(redisClient
*c
);
858 void rpopCommand(redisClient
*c
);
859 void llenCommand(redisClient
*c
);
860 void lindexCommand(redisClient
*c
);
861 void lrangeCommand(redisClient
*c
);
862 void ltrimCommand(redisClient
*c
);
863 void typeCommand(redisClient
*c
);
864 void lsetCommand(redisClient
*c
);
865 void saddCommand(redisClient
*c
);
866 void sremCommand(redisClient
*c
);
867 void smoveCommand(redisClient
*c
);
868 void sismemberCommand(redisClient
*c
);
869 void scardCommand(redisClient
*c
);
870 void spopCommand(redisClient
*c
);
871 void srandmemberCommand(redisClient
*c
);
872 void sinterCommand(redisClient
*c
);
873 void sinterstoreCommand(redisClient
*c
);
874 void sunionCommand(redisClient
*c
);
875 void sunionstoreCommand(redisClient
*c
);
876 void sdiffCommand(redisClient
*c
);
877 void sdiffstoreCommand(redisClient
*c
);
878 void syncCommand(redisClient
*c
);
879 void flushdbCommand(redisClient
*c
);
880 void flushallCommand(redisClient
*c
);
881 void sortCommand(redisClient
*c
);
882 void lremCommand(redisClient
*c
);
883 void rpoplpushcommand(redisClient
*c
);
884 void infoCommand(redisClient
*c
);
885 void mgetCommand(redisClient
*c
);
886 void monitorCommand(redisClient
*c
);
887 void expireCommand(redisClient
*c
);
888 void expireatCommand(redisClient
*c
);
889 void getsetCommand(redisClient
*c
);
890 void ttlCommand(redisClient
*c
);
891 void persistCommand(redisClient
*c
);
892 void slaveofCommand(redisClient
*c
);
893 void debugCommand(redisClient
*c
);
894 void msetCommand(redisClient
*c
);
895 void msetnxCommand(redisClient
*c
);
896 void zaddCommand(redisClient
*c
);
897 void zincrbyCommand(redisClient
*c
);
898 void zrangeCommand(redisClient
*c
);
899 void zrangebyscoreCommand(redisClient
*c
);
900 void zcountCommand(redisClient
*c
);
901 void zrevrangeCommand(redisClient
*c
);
902 void zcardCommand(redisClient
*c
);
903 void zremCommand(redisClient
*c
);
904 void zscoreCommand(redisClient
*c
);
905 void zremrangebyscoreCommand(redisClient
*c
);
906 void multiCommand(redisClient
*c
);
907 void execCommand(redisClient
*c
);
908 void discardCommand(redisClient
*c
);
909 void blpopCommand(redisClient
*c
);
910 void brpopCommand(redisClient
*c
);
911 void appendCommand(redisClient
*c
);
912 void substrCommand(redisClient
*c
);
913 void strlenCommand(redisClient
*c
);
914 void zrankCommand(redisClient
*c
);
915 void zrevrankCommand(redisClient
*c
);
916 void hsetCommand(redisClient
*c
);
917 void hsetnxCommand(redisClient
*c
);
918 void hgetCommand(redisClient
*c
);
919 void hmsetCommand(redisClient
*c
);
920 void hmgetCommand(redisClient
*c
);
921 void hdelCommand(redisClient
*c
);
922 void hlenCommand(redisClient
*c
);
923 void zremrangebyrankCommand(redisClient
*c
);
924 void zunionstoreCommand(redisClient
*c
);
925 void zinterstoreCommand(redisClient
*c
);
926 void hkeysCommand(redisClient
*c
);
927 void hvalsCommand(redisClient
*c
);
928 void hgetallCommand(redisClient
*c
);
929 void hexistsCommand(redisClient
*c
);
930 void configCommand(redisClient
*c
);
931 void hincrbyCommand(redisClient
*c
);
932 void subscribeCommand(redisClient
*c
);
933 void unsubscribeCommand(redisClient
*c
);
934 void psubscribeCommand(redisClient
*c
);
935 void punsubscribeCommand(redisClient
*c
);
936 void publishCommand(redisClient
*c
);
937 void watchCommand(redisClient
*c
);
938 void unwatchCommand(redisClient
*c
);
940 #if defined(__GNUC__)
941 void *calloc(size_t count
, size_t size
) __attribute__ ((deprecated
));
942 void free(void *ptr
) __attribute__ ((deprecated
));
943 void *malloc(size_t size
) __attribute__ ((deprecated
));
944 void *realloc(void *ptr
, size_t size
) __attribute__ ((deprecated
));