mirror of
https://github.com/fluencelabs/redis
synced 2025-04-03 16:21:03 +00:00
Threaded lazyfree WIP #1.
This commit is contained in:
parent
4d50d691e3
commit
9253d85073
@ -185,6 +185,8 @@ void *bioProcessBackgroundJobs(void *arg) {
|
|||||||
close((long)job->arg1);
|
close((long)job->arg1);
|
||||||
} else if (type == BIO_AOF_FSYNC) {
|
} else if (type == BIO_AOF_FSYNC) {
|
||||||
aof_fsync((long)job->arg1);
|
aof_fsync((long)job->arg1);
|
||||||
|
} else if (type == BIO_LAZY_FREE) {
|
||||||
|
decrRefCount((robj*)job->arg1);
|
||||||
} else {
|
} else {
|
||||||
serverPanic("Wrong job type in bioProcessBackgroundJobs().");
|
serverPanic("Wrong job type in bioProcessBackgroundJobs().");
|
||||||
}
|
}
|
||||||
|
@ -38,4 +38,5 @@ void bioKillThreads(void);
|
|||||||
/* Background job opcodes */
|
/* Background job opcodes */
|
||||||
#define BIO_CLOSE_FILE 0 /* Deferred close(2) syscall. */
|
#define BIO_CLOSE_FILE 0 /* Deferred close(2) syscall. */
|
||||||
#define BIO_AOF_FSYNC 1 /* Deferred AOF fsync. */
|
#define BIO_AOF_FSYNC 1 /* Deferred AOF fsync. */
|
||||||
#define BIO_NUM_OPS 2
|
#define BIO_LAZY_FREE 2 /* Deferred objects freeing. */
|
||||||
|
#define BIO_NUM_OPS 3
|
||||||
|
@ -1,4 +1,7 @@
|
|||||||
#include "server.h"
|
#include "server.h"
|
||||||
|
#include "bio.h"
|
||||||
|
|
||||||
|
static int lazyfree_threaded = 1; /* Use a thread to reclaim objects. */
|
||||||
|
|
||||||
/* Initialization of the lazy free engine. Must be called only once at server
|
/* Initialization of the lazy free engine. Must be called only once at server
|
||||||
* startup. */
|
* startup. */
|
||||||
@ -97,7 +100,17 @@ size_t lazyfreeFastStep(void) {
|
|||||||
|
|
||||||
/* Handles slow or fast collection steps. */
|
/* Handles slow or fast collection steps. */
|
||||||
size_t lazyfreeStep(int type) {
|
size_t lazyfreeStep(int type) {
|
||||||
if (type == LAZYFREE_STEP_FAST) return lazyfreeFastStep();
|
/* Threaded implementaiton: only block for STEP_OOM. */
|
||||||
|
if (lazyfree_threaded) {
|
||||||
|
if (type == LAZYFREE_STEP_OOM)
|
||||||
|
return bioWaitStepOfType(BIO_LAZY_FREE);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Non threaded implementation: free things incrementally avoiding
|
||||||
|
* to block. */
|
||||||
|
if (type == LAZYFREE_STEP_FAST ||
|
||||||
|
type == LAZYFREE_STEP_OOM) return lazyfreeFastStep();
|
||||||
|
|
||||||
size_t totalwork = 0;
|
size_t totalwork = 0;
|
||||||
mstime_t end = mstime()+2;
|
mstime_t end = mstime()+2;
|
||||||
@ -130,8 +143,12 @@ int dbAsyncDelete(redisDb *db, robj *key) {
|
|||||||
/* If releasing the object is too much work, let's put it into the
|
/* If releasing the object is too much work, let's put it into the
|
||||||
* lazy free list. */
|
* lazy free list. */
|
||||||
if (free_effort > LAZYFREE_THRESHOLD) {
|
if (free_effort > LAZYFREE_THRESHOLD) {
|
||||||
listAddNodeTail(server.lazyfree_obj,val);
|
if (lazyfree_threaded) {
|
||||||
server.lazyfree_elements += free_effort;
|
bioCreateBackgroundJob(BIO_LAZY_FREE,val,NULL,NULL);
|
||||||
|
} else {
|
||||||
|
listAddNodeTail(server.lazyfree_obj,val);
|
||||||
|
server.lazyfree_elements += free_effort;
|
||||||
|
}
|
||||||
dictSetVal(db->dict,de,NULL);
|
dictSetVal(db->dict,de,NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -165,6 +182,9 @@ int lazyfreeCron(struct aeEventLoop *eventLoop, long long id, void *clientData)
|
|||||||
UNUSED(id);
|
UNUSED(id);
|
||||||
UNUSED(clientData);
|
UNUSED(clientData);
|
||||||
|
|
||||||
|
/* Threaded lazy free does not need a timer, unregister the timer event. */
|
||||||
|
if (lazyfree_threaded) return AE_NOMORE;
|
||||||
|
|
||||||
static size_t prev_mem;
|
static size_t prev_mem;
|
||||||
static int timer_period = 1000; /* Defauls to 1HZ */
|
static int timer_period = 1000; /* Defauls to 1HZ */
|
||||||
static double mem_trend = 0;
|
static double mem_trend = 0;
|
||||||
|
@ -3330,7 +3330,7 @@ int freeMemoryIfNeeded(void) {
|
|||||||
latencyStartMonitor(eviction_latency);
|
latencyStartMonitor(eviction_latency);
|
||||||
while (mem_freed < mem_tofree) {
|
while (mem_freed < mem_tofree) {
|
||||||
delta = (long long) zmalloc_used_memory();
|
delta = (long long) zmalloc_used_memory();
|
||||||
size_t workdone = lazyfreeStep(LAZYFREE_STEP_FAST);
|
size_t workdone = lazyfreeStep(LAZYFREE_STEP_OOM);
|
||||||
delta -= (long long) zmalloc_used_memory();
|
delta -= (long long) zmalloc_used_memory();
|
||||||
mem_freed += delta;
|
mem_freed += delta;
|
||||||
if (!workdone) break; /* Lazy free list is empty. */
|
if (!workdone) break; /* Lazy free list is empty. */
|
||||||
|
11
src/server.h
11
src/server.h
@ -1402,9 +1402,14 @@ void slotToKeyAdd(robj *key);
|
|||||||
void slotToKeyDel(robj *key);
|
void slotToKeyDel(robj *key);
|
||||||
void slotToKeyFlush(void);
|
void slotToKeyFlush(void);
|
||||||
|
|
||||||
/* Lazy free */
|
/* Lazy free. Note that SLOW and FAST are only useful when incremental
|
||||||
#define LAZYFREE_STEP_SLOW 0
|
* lazy free is active. For threaded lazy free the actual freeing of objects
|
||||||
#define LAZYFREE_STEP_FAST 1
|
* happens in the background. Only STEP_OOM is used since it blocks waiting
|
||||||
|
* for the freeing thread to do some work before returning. */
|
||||||
|
#define LAZYFREE_STEP_SLOW 0 /* Take 1-2 milliseconds to reclaim memory. */
|
||||||
|
#define LAZYFREE_STEP_FAST 1 /* Free a few elements ASAP and return. */
|
||||||
|
#define LAZYFREE_STEP_OOM 2 /* Free a few elements at any cost if there
|
||||||
|
is something to free: we are out of memory */
|
||||||
int dbAsyncDelete(redisDb *db, robj *key);
|
int dbAsyncDelete(redisDb *db, robj *key);
|
||||||
void initLazyfreeEngine(void);
|
void initLazyfreeEngine(void);
|
||||||
size_t lazyfreeStep(int type);
|
size_t lazyfreeStep(int type);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user