mirror of
http://git.haproxy.org/git/haproxy.git/
synced 2025-01-21 05:00:42 +00:00
MEDIUM: pipe/thread: maintain a per-thread local cache of recently used pipes
In order to completely remove the pipe locking cost and try to reuse hot pipes, each thread now maintains a local cache of recently used pipes that is no larger than its share (maxpipes/nbthreads). All extra pipes are instead refilled into the global pool. Allocations are made from the local pool first, and fall back to the global one before allocating one. This completely removes the observed pipe locking cost at high bit rates, which was still around 5-6%.
This commit is contained in:
parent
a945cfdfe0
commit
160287b676
21
src/pipe.c
21
src/pipe.c
@ -26,6 +26,9 @@ struct pipe *pipes_live = NULL; /* pipes which are still ready to use */
|
|||||||
|
|
||||||
__decl_spinlock(pipes_lock); /* lock used to protect pipes list */
|
__decl_spinlock(pipes_lock); /* lock used to protect pipes list */
|
||||||
|
|
||||||
|
static THREAD_LOCAL int local_pipes_free = 0; /* #cache objects */
|
||||||
|
static THREAD_LOCAL struct pipe *local_pipes = NULL;
|
||||||
|
|
||||||
int pipes_used = 0; /* # of pipes in use (2 fds each) */
|
int pipes_used = 0; /* # of pipes in use (2 fds each) */
|
||||||
int pipes_free = 0; /* # of pipes unused */
|
int pipes_free = 0; /* # of pipes unused */
|
||||||
|
|
||||||
@ -37,6 +40,15 @@ struct pipe *get_pipe()
|
|||||||
struct pipe *ret = NULL;
|
struct pipe *ret = NULL;
|
||||||
int pipefd[2];
|
int pipefd[2];
|
||||||
|
|
||||||
|
ret = local_pipes;
|
||||||
|
if (likely(ret)) {
|
||||||
|
local_pipes = ret->next;
|
||||||
|
local_pipes_free--;
|
||||||
|
HA_ATOMIC_SUB(&pipes_free, 1);
|
||||||
|
HA_ATOMIC_ADD(&pipes_used, 1);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
if (likely(pipes_live)) {
|
if (likely(pipes_live)) {
|
||||||
HA_SPIN_LOCK(PIPES_LOCK, &pipes_lock);
|
HA_SPIN_LOCK(PIPES_LOCK, &pipes_lock);
|
||||||
ret = pipes_live;
|
ret = pipes_live;
|
||||||
@ -96,6 +108,15 @@ void put_pipe(struct pipe *p)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (likely(local_pipes_free * global.nbthread < global.maxpipes)) {
|
||||||
|
p->next = local_pipes;
|
||||||
|
local_pipes = p;
|
||||||
|
local_pipes_free++;
|
||||||
|
HA_ATOMIC_ADD(&pipes_free, 1);
|
||||||
|
HA_ATOMIC_SUB(&pipes_used, 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
HA_SPIN_LOCK(PIPES_LOCK, &pipes_lock);
|
HA_SPIN_LOCK(PIPES_LOCK, &pipes_lock);
|
||||||
p->next = pipes_live;
|
p->next = pipes_live;
|
||||||
pipes_live = p;
|
pipes_live = p;
|
||||||
|
Loading…
Reference in New Issue
Block a user