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
;
337 long long dirty
; /* changes to DB from the last save */
339 list
*slaves
, *monitors
;
340 char neterr
[ANET_ERR_LEN
];
342 int cronloops
; /* number of times the cron function run */
343 list
*objfreelist
; /* A list of freed objects to avoid malloc() */
344 time_t lastsave
; /* Unix time of last save succeeede */
345 /* Fields used only for stats */
346 time_t stat_starttime
; /* server start time */
347 long long stat_numcommands
; /* number of processed commands */
348 long long stat_numconnections
; /* number of connections received */
349 long long stat_expiredkeys
; /* number of expired keys */
358 int no_appendfsync_on_rewrite
;
364 pid_t bgsavechildpid
;
365 pid_t bgrewritechildpid
;
366 sds bgrewritebuf
; /* buffer taken by parent during oppend only rewrite */
367 sds aofbuf
; /* AOF buffer, written before entering the event loop */
368 struct saveparam
*saveparams
;
372 char *appendfilename
;
376 /* Replication related */
381 redisClient
*master
; /* client that is master for this slave */
383 unsigned int maxclients
;
384 unsigned long long maxmemory
;
385 unsigned int blpop_blocked_clients
;
386 unsigned int vm_blocked_clients
;
387 /* Sort parameters - qsort_r() is only available under BSD so we
388 * have to take this state global, in order to pass it to sortCompare() */
392 /* Virtual memory configuration */
397 unsigned long long vm_max_memory
;
398 /* Zip structure config */
399 size_t hash_max_zipmap_entries
;
400 size_t hash_max_zipmap_value
;
401 size_t list_max_ziplist_entries
;
402 size_t list_max_ziplist_value
;
403 /* Virtual memory state */
406 off_t vm_next_page
; /* Next probably empty page */
407 off_t vm_near_pages
; /* Number of pages allocated sequentially */
408 unsigned char *vm_bitmap
; /* Bitmap of free/used pages */
409 time_t unixtime
; /* Unix time sampled every second. */
410 /* Virtual memory I/O threads stuff */
411 /* An I/O thread process an element taken from the io_jobs queue and
412 * put the result of the operation in the io_done list. While the
413 * job is being processed, it's put on io_processing queue. */
414 list
*io_newjobs
; /* List of VM I/O jobs yet to be processed */
415 list
*io_processing
; /* List of VM I/O jobs being processed */
416 list
*io_processed
; /* List of VM I/O jobs already processed */
417 list
*io_ready_clients
; /* Clients ready to be unblocked. All keys loaded */
418 pthread_mutex_t io_mutex
; /* lock to access io_jobs/io_done/io_thread_job */
419 pthread_mutex_t obj_freelist_mutex
; /* safe redis objects creation/free */
420 pthread_mutex_t io_swapfile_mutex
; /* So we can lseek + write */
421 pthread_attr_t io_threads_attr
; /* attributes for threads creation */
422 int io_active_threads
; /* Number of running I/O threads */
423 int vm_max_threads
; /* Max number of I/O threads running at the same time */
424 /* Our main thread is blocked on the event loop, locking for sockets ready
425 * to be read or written, so when a threaded I/O operation is ready to be
426 * processed by the main thread, the I/O thread will use a unix pipe to
427 * awake the main thread. The followings are the two pipe FDs. */
428 int io_ready_pipe_read
;
429 int io_ready_pipe_write
;
430 /* Virtual memory stats */
431 unsigned long long vm_stats_used_pages
;
432 unsigned long long vm_stats_swapped_objects
;
433 unsigned long long vm_stats_swapouts
;
434 unsigned long long vm_stats_swapins
;
436 dict
*pubsub_channels
; /* Map channels to list of subscribed clients */
437 list
*pubsub_patterns
; /* A list of pubsub_patterns */
440 unsigned lruclock
:22; /* clock incrementing every minute, for LRU */
441 unsigned lruclock_padding
:10;
444 typedef struct pubsubPattern
{
449 typedef void redisCommandProc(redisClient
*c
);
450 typedef void redisVmPreloadProc(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
451 struct redisCommand
{
453 redisCommandProc
*proc
;
456 /* Use a function to determine which keys need to be loaded
457 * in the background prior to executing this command. Takes precedence
458 * over vm_firstkey and others, ignored when NULL */
459 redisVmPreloadProc
*vm_preload_proc
;
460 /* What keys should be loaded in background when calling this command? */
461 int vm_firstkey
; /* The first argument that's a key (0 = no keys) */
462 int vm_lastkey
; /* THe last argument that's a key */
463 int vm_keystep
; /* The step between first and last key */
466 struct redisFunctionSym
{
468 unsigned long pointer
;
471 typedef struct _redisSortObject
{
479 typedef struct _redisSortOperation
{
482 } redisSortOperation
;
484 /* ZSETs use a specialized version of Skiplists */
486 typedef struct zskiplistNode
{
487 struct zskiplistNode
**forward
;
488 struct zskiplistNode
*backward
;
494 typedef struct zskiplist
{
495 struct zskiplistNode
*header
, *tail
;
496 unsigned long length
;
500 typedef struct zset
{
505 /* VM threaded I/O request message */
506 #define REDIS_IOJOB_LOAD 0 /* Load from disk to memory */
507 #define REDIS_IOJOB_PREPARE_SWAP 1 /* Compute needed pages */
508 #define REDIS_IOJOB_DO_SWAP 2 /* Swap from memory to disk */
509 typedef struct iojob
{
510 int type
; /* Request type, REDIS_IOJOB_* */
511 redisDb
*db
;/* Redis database */
512 robj
*key
; /* This I/O request is about swapping this key */
513 robj
*id
; /* Unique identifier of this job:
514 this is the object to swap for REDIS_IOREQ_*_SWAP, or the
515 vmpointer objct for REDIS_IOREQ_LOAD. */
516 robj
*val
; /* the value to swap for REDIS_IOREQ_*_SWAP, otherwise this
517 * field is populated by the I/O thread for REDIS_IOREQ_LOAD. */
518 off_t page
; /* Swap page where to read/write the object */
519 off_t pages
; /* Swap pages needed to save object. PREPARE_SWAP return val */
520 int canceled
; /* True if this command was canceled by blocking side of VM */
521 pthread_t thread
; /* ID of the thread processing this entry */
524 /* Structure to hold list iteration abstraction. */
527 unsigned char encoding
;
528 unsigned char direction
; /* Iteration direction */
533 /* Structure for an entry while iterating over a list. */
535 listTypeIterator
*li
;
536 unsigned char *zi
; /* Entry in ziplist */
537 listNode
*ln
; /* Entry in linked list */
540 /* Structure to hold hash iteration abstration. Note that iteration over
541 * hashes involves both fields and values. Because it is possible that
542 * not both are required, store pointers in the iterator to avoid
543 * unnecessary memory allocation for fields/values. */
547 unsigned char *zk
, *zv
;
548 unsigned int zklen
, zvlen
;
554 #define REDIS_HASH_KEY 1
555 #define REDIS_HASH_VALUE 2
557 /*-----------------------------------------------------------------------------
558 * Extern declarations
559 *----------------------------------------------------------------------------*/
561 extern struct redisServer server
;
562 extern struct sharedObjectsStruct shared
;
563 extern dictType setDictType
;
564 extern dictType zsetDictType
;
565 extern double R_Zero
, R_PosInf
, R_NegInf
, R_Nan
;
566 dictType hashDictType
;
568 /*-----------------------------------------------------------------------------
569 * Functions prototypes
570 *----------------------------------------------------------------------------*/
572 /* networking.c -- Networking and Client related operations */
573 redisClient
*createClient(int fd
);
574 void closeTimedoutClients(void);
575 void freeClient(redisClient
*c
);
576 void resetClient(redisClient
*c
);
577 void sendReplyToClient(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
578 void sendReplyToClientWritev(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
579 void addReply(redisClient
*c
, robj
*obj
);
580 void addReplySds(redisClient
*c
, sds s
);
581 void processInputBuffer(redisClient
*c
);
582 void acceptHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
583 void readQueryFromClient(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
584 void addReplyBulk(redisClient
*c
, robj
*obj
);
585 void addReplyBulkCString(redisClient
*c
, char *s
);
586 void acceptHandler(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
587 void addReply(redisClient
*c
, robj
*obj
);
588 void addReplySds(redisClient
*c
, sds s
);
589 void addReplyDouble(redisClient
*c
, double d
);
590 void addReplyLongLong(redisClient
*c
, long long ll
);
591 void addReplyUlong(redisClient
*c
, unsigned long ul
);
592 void *dupClientReplyValue(void *o
);
595 void listTypeTryConversion(robj
*subject
, robj
*value
);
596 void listTypePush(robj
*subject
, robj
*value
, int where
);
597 robj
*listTypePop(robj
*subject
, int where
);
598 unsigned long listTypeLength(robj
*subject
);
599 listTypeIterator
*listTypeInitIterator(robj
*subject
, int index
, unsigned char direction
);
600 void listTypeReleaseIterator(listTypeIterator
*li
);
601 int listTypeNext(listTypeIterator
*li
, listTypeEntry
*entry
);
602 robj
*listTypeGet(listTypeEntry
*entry
);
603 void listTypeInsert(listTypeEntry
*entry
, robj
*value
, int where
);
604 int listTypeEqual(listTypeEntry
*entry
, robj
*o
);
605 void listTypeDelete(listTypeEntry
*entry
);
606 void listTypeConvert(robj
*subject
, int enc
);
607 void unblockClientWaitingData(redisClient
*c
);
608 int handleClientsWaitingListPush(redisClient
*c
, robj
*key
, robj
*ele
);
609 void popGenericCommand(redisClient
*c
, int where
);
611 /* MULTI/EXEC/WATCH... */
612 void unwatchAllKeys(redisClient
*c
);
613 void initClientMultiState(redisClient
*c
);
614 void freeClientMultiState(redisClient
*c
);
615 void queueMultiCommand(redisClient
*c
, struct redisCommand
*cmd
);
616 void touchWatchedKey(redisDb
*db
, robj
*key
);
617 void touchWatchedKeysOnFlush(int dbid
);
619 /* Redis object implementation */
620 void decrRefCount(void *o
);
621 void incrRefCount(robj
*o
);
622 void freeStringObject(robj
*o
);
623 void freeListObject(robj
*o
);
624 void freeSetObject(robj
*o
);
625 void freeZsetObject(robj
*o
);
626 void freeHashObject(robj
*o
);
627 robj
*createObject(int type
, void *ptr
);
628 robj
*createStringObject(char *ptr
, size_t len
);
629 robj
*dupStringObject(robj
*o
);
630 robj
*tryObjectEncoding(robj
*o
);
631 robj
*getDecodedObject(robj
*o
);
632 size_t stringObjectLen(robj
*o
);
633 int tryFreeOneObjectFromFreelist(void);
634 robj
*createStringObjectFromLongLong(long long value
);
635 robj
*createListObject(void);
636 robj
*createZiplistObject(void);
637 robj
*createSetObject(void);
638 robj
*createHashObject(void);
639 robj
*createZsetObject(void);
640 int getLongFromObjectOrReply(redisClient
*c
, robj
*o
, long *target
, const char *msg
);
641 int checkType(redisClient
*c
, robj
*o
, int type
);
642 int getLongLongFromObjectOrReply(redisClient
*c
, robj
*o
, long long *target
, const char *msg
);
643 int getDoubleFromObjectOrReply(redisClient
*c
, robj
*o
, double *target
, const char *msg
);
644 int getLongLongFromObject(robj
*o
, long long *target
);
645 char *strEncoding(int encoding
);
646 int compareStringObjects(robj
*a
, robj
*b
);
647 int equalStringObjects(robj
*a
, robj
*b
);
650 void replicationFeedSlaves(list
*slaves
, int dictid
, robj
**argv
, int argc
);
651 void replicationFeedMonitors(list
*monitors
, int dictid
, robj
**argv
, int argc
);
652 int syncWithMaster(void);
653 void updateSlavesWaitingBgsave(int bgsaveerr
);
655 /* RDB persistence */
656 int rdbLoad(char *filename
);
657 int rdbSaveBackground(char *filename
);
658 void rdbRemoveTempFile(pid_t childpid
);
659 int rdbSave(char *filename
);
660 int rdbSaveObject(FILE *fp
, robj
*o
);
661 off_t
rdbSavedObjectPages(robj
*o
, FILE *fp
);
662 off_t
rdbSavedObjectLen(robj
*o
, FILE *fp
);
663 robj
*rdbLoadObject(int type
, FILE *fp
);
664 void backgroundSaveDoneHandler(int statloc
);
666 /* AOF persistence */
667 void flushAppendOnlyFile(void);
668 void feedAppendOnlyFile(struct redisCommand
*cmd
, int dictid
, robj
**argv
, int argc
);
669 void aofRemoveTempFile(pid_t childpid
);
670 int rewriteAppendOnlyFileBackground(void);
671 int loadAppendOnlyFile(char *filename
);
672 void stopAppendOnly(void);
673 int startAppendOnly(void);
674 void backgroundRewriteDoneHandler(int statloc
);
676 /* Sorted sets data type */
677 zskiplist
*zslCreate(void);
678 void zslFree(zskiplist
*zsl
);
679 void zslInsert(zskiplist
*zsl
, double score
, robj
*obj
);
682 void freeMemoryIfNeeded(void);
683 int processCommand(redisClient
*c
);
684 void setupSigSegvAction(void);
685 struct redisCommand
*lookupCommand(char *name
);
686 void call(redisClient
*c
, struct redisCommand
*cmd
);
687 int prepareForShutdown();
688 void redisLog(int level
, const char *fmt
, ...);
690 void updateDictResizePolicy(void);
691 int htNeedsResize(dict
*dict
);
692 void oom(const char *msg
);
696 void vmMarkPagesFree(off_t page
, off_t count
);
697 robj
*vmLoadObject(robj
*o
);
698 robj
*vmPreviewObject(robj
*o
);
699 int vmSwapOneObjectBlocking(void);
700 int vmSwapOneObjectThreaded(void);
701 int vmCanSwapOut(void);
702 void vmThreadedIOCompletedJob(aeEventLoop
*el
, int fd
, void *privdata
, int mask
);
703 void vmCancelThreadedIOJob(robj
*o
);
704 void lockThreadedIO(void);
705 void unlockThreadedIO(void);
706 int vmSwapObjectThreaded(robj
*key
, robj
*val
, redisDb
*db
);
707 void freeIOJob(iojob
*j
);
708 void queueIOJob(iojob
*j
);
709 int vmWriteObjectOnSwap(robj
*o
, off_t page
);
710 robj
*vmReadObjectFromSwap(off_t page
, int type
);
711 void waitEmptyIOJobsQueue(void);
712 void vmReopenSwapFile(void);
713 int vmFreePage(off_t page
);
714 void zunionInterBlockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
715 void execBlockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
, int argc
, robj
**argv
);
716 int blockClientOnSwappedKeys(redisClient
*c
, struct redisCommand
*cmd
);
717 int dontWaitForSwappedKey(redisClient
*c
, robj
*key
);
718 void handleClientsBlockedOnSwappedKey(redisDb
*db
, robj
*key
);
719 vmpointer
*vmSwapObjectBlocking(robj
*val
);
722 void convertToRealHash(robj
*o
);
723 void hashTypeTryConversion(robj
*subject
, robj
**argv
, int start
, int end
);
724 void hashTypeTryObjectEncoding(robj
*subject
, robj
**o1
, robj
**o2
);
725 robj
*hashTypeGet(robj
*o
, robj
*key
);
726 int hashTypeExists(robj
*o
, robj
*key
);
727 int hashTypeSet(robj
*o
, robj
*key
, robj
*value
);
728 int hashTypeDelete(robj
*o
, robj
*key
);
729 unsigned long hashTypeLength(robj
*o
);
730 hashTypeIterator
*hashTypeInitIterator(robj
*subject
);
731 void hashTypeReleaseIterator(hashTypeIterator
*hi
);
732 int hashTypeNext(hashTypeIterator
*hi
);
733 robj
*hashTypeCurrent(hashTypeIterator
*hi
, int what
);
734 robj
*hashTypeLookupWriteOrCreate(redisClient
*c
, robj
*key
);
737 int pubsubUnsubscribeAllChannels(redisClient
*c
, int notify
);
738 int pubsubUnsubscribeAllPatterns(redisClient
*c
, int notify
);
739 void freePubsubPattern(void *p
);
740 int listMatchPubsubPattern(void *a
, void *b
);
742 /* Utility functions */
743 int stringmatchlen(const char *pattern
, int patternLen
,
744 const char *string
, int stringLen
, int nocase
);
745 int stringmatch(const char *pattern
, const char *string
, int nocase
);
746 long long memtoll(const char *p
, int *err
);
747 int ll2string(char *s
, size_t len
, long long value
);
748 int isStringRepresentableAsLong(sds s
, long *longval
);
751 void loadServerConfig(char *filename
);
752 void appendServerSaveParams(time_t seconds
, int changes
);
753 void resetServerSaveParams();
755 /* db.c -- Keyspace access API */
756 int removeExpire(redisDb
*db
, robj
*key
);
757 int expireIfNeeded(redisDb
*db
, robj
*key
);
758 int deleteIfVolatile(redisDb
*db
, robj
*key
);
759 time_t getExpire(redisDb
*db
, robj
*key
);
760 int setExpire(redisDb
*db
, robj
*key
, time_t when
);
761 robj
*lookupKey(redisDb
*db
, robj
*key
);
762 robj
*lookupKeyRead(redisDb
*db
, robj
*key
);
763 robj
*lookupKeyWrite(redisDb
*db
, robj
*key
);
764 robj
*lookupKeyReadOrReply(redisClient
*c
, robj
*key
, robj
*reply
);
765 robj
*lookupKeyWriteOrReply(redisClient
*c
, robj
*key
, robj
*reply
);
766 int dbAdd(redisDb
*db
, robj
*key
, robj
*val
);
767 int dbReplace(redisDb
*db
, robj
*key
, robj
*val
);
768 int dbExists(redisDb
*db
, robj
*key
);
769 robj
*dbRandomKey(redisDb
*db
);
770 int dbDelete(redisDb
*db
, robj
*key
);
772 int selectDb(redisClient
*c
, int id
);
775 char *redisGitSHA1(void);
776 char *redisGitDirty(void);
778 /* Commands prototypes */
779 void authCommand(redisClient
*c
);
780 void pingCommand(redisClient
*c
);
781 void echoCommand(redisClient
*c
);
782 void setCommand(redisClient
*c
);
783 void setnxCommand(redisClient
*c
);
784 void setexCommand(redisClient
*c
);
785 void getCommand(redisClient
*c
);
786 void delCommand(redisClient
*c
);
787 void existsCommand(redisClient
*c
);
788 void incrCommand(redisClient
*c
);
789 void decrCommand(redisClient
*c
);
790 void incrbyCommand(redisClient
*c
);
791 void decrbyCommand(redisClient
*c
);
792 void selectCommand(redisClient
*c
);
793 void randomkeyCommand(redisClient
*c
);
794 void keysCommand(redisClient
*c
);
795 void dbsizeCommand(redisClient
*c
);
796 void lastsaveCommand(redisClient
*c
);
797 void saveCommand(redisClient
*c
);
798 void bgsaveCommand(redisClient
*c
);
799 void bgrewriteaofCommand(redisClient
*c
);
800 void shutdownCommand(redisClient
*c
);
801 void moveCommand(redisClient
*c
);
802 void renameCommand(redisClient
*c
);
803 void renamenxCommand(redisClient
*c
);
804 void lpushCommand(redisClient
*c
);
805 void rpushCommand(redisClient
*c
);
806 void lpushxCommand(redisClient
*c
);
807 void rpushxCommand(redisClient
*c
);
808 void linsertCommand(redisClient
*c
);
809 void lpopCommand(redisClient
*c
);
810 void rpopCommand(redisClient
*c
);
811 void llenCommand(redisClient
*c
);
812 void lindexCommand(redisClient
*c
);
813 void lrangeCommand(redisClient
*c
);
814 void ltrimCommand(redisClient
*c
);
815 void typeCommand(redisClient
*c
);
816 void lsetCommand(redisClient
*c
);
817 void saddCommand(redisClient
*c
);
818 void sremCommand(redisClient
*c
);
819 void smoveCommand(redisClient
*c
);
820 void sismemberCommand(redisClient
*c
);
821 void scardCommand(redisClient
*c
);
822 void spopCommand(redisClient
*c
);
823 void srandmemberCommand(redisClient
*c
);
824 void sinterCommand(redisClient
*c
);
825 void sinterstoreCommand(redisClient
*c
);
826 void sunionCommand(redisClient
*c
);
827 void sunionstoreCommand(redisClient
*c
);
828 void sdiffCommand(redisClient
*c
);
829 void sdiffstoreCommand(redisClient
*c
);
830 void syncCommand(redisClient
*c
);
831 void flushdbCommand(redisClient
*c
);
832 void flushallCommand(redisClient
*c
);
833 void sortCommand(redisClient
*c
);
834 void lremCommand(redisClient
*c
);
835 void rpoplpushcommand(redisClient
*c
);
836 void infoCommand(redisClient
*c
);
837 void mgetCommand(redisClient
*c
);
838 void monitorCommand(redisClient
*c
);
839 void expireCommand(redisClient
*c
);
840 void expireatCommand(redisClient
*c
);
841 void getsetCommand(redisClient
*c
);
842 void ttlCommand(redisClient
*c
);
843 void slaveofCommand(redisClient
*c
);
844 void debugCommand(redisClient
*c
);
845 void msetCommand(redisClient
*c
);
846 void msetnxCommand(redisClient
*c
);
847 void zaddCommand(redisClient
*c
);
848 void zincrbyCommand(redisClient
*c
);
849 void zrangeCommand(redisClient
*c
);
850 void zrangebyscoreCommand(redisClient
*c
);
851 void zcountCommand(redisClient
*c
);
852 void zrevrangeCommand(redisClient
*c
);
853 void zcardCommand(redisClient
*c
);
854 void zremCommand(redisClient
*c
);
855 void zscoreCommand(redisClient
*c
);
856 void zremrangebyscoreCommand(redisClient
*c
);
857 void multiCommand(redisClient
*c
);
858 void execCommand(redisClient
*c
);
859 void discardCommand(redisClient
*c
);
860 void blpopCommand(redisClient
*c
);
861 void brpopCommand(redisClient
*c
);
862 void appendCommand(redisClient
*c
);
863 void substrCommand(redisClient
*c
);
864 void strlenCommand(redisClient
*c
);
865 void zrankCommand(redisClient
*c
);
866 void zrevrankCommand(redisClient
*c
);
867 void hsetCommand(redisClient
*c
);
868 void hsetnxCommand(redisClient
*c
);
869 void hgetCommand(redisClient
*c
);
870 void hmsetCommand(redisClient
*c
);
871 void hmgetCommand(redisClient
*c
);
872 void hdelCommand(redisClient
*c
);
873 void hlenCommand(redisClient
*c
);
874 void zremrangebyrankCommand(redisClient
*c
);
875 void zunionstoreCommand(redisClient
*c
);
876 void zinterstoreCommand(redisClient
*c
);
877 void hkeysCommand(redisClient
*c
);
878 void hvalsCommand(redisClient
*c
);
879 void hgetallCommand(redisClient
*c
);
880 void hexistsCommand(redisClient
*c
);
881 void configCommand(redisClient
*c
);
882 void hincrbyCommand(redisClient
*c
);
883 void subscribeCommand(redisClient
*c
);
884 void unsubscribeCommand(redisClient
*c
);
885 void psubscribeCommand(redisClient
*c
);
886 void punsubscribeCommand(redisClient
*c
);
887 void publishCommand(redisClient
*c
);
888 void watchCommand(redisClient
*c
);
889 void unwatchCommand(redisClient
*c
);
891 #if defined(__GNUC__)
892 void *calloc(size_t count
, size_t size
) __attribute__ ((deprecated
));
893 void free(void *ptr
) __attribute__ ((deprecated
));
894 void *malloc(size_t size
) __attribute__ ((deprecated
));
895 void *realloc(void *ptr
, size_t size
) __attribute__ ((deprecated
));