From fa416594f9ed7b81432a2eb7d6defd9a3d79b362 Mon Sep 17 00:00:00 2001 From: Nikita Popov Date: Mon, 26 Nov 2018 19:56:04 +0100 Subject: [PATCH] Try to reduce SMA fragmentation Instead of picking the first fitting block, check if there is a smaller block that still fits. We only look at a limited, small number of additional blocks. This is an approximation of best-fit allocation. This is intended to reduce fragmentation, in particular for the case where a large block was recently freed (and is on top of the free list) and a small allocation is performed. --- apc_sma.c | 73 +++++++++++++++++++++++++++++------------------ tests/sma001.phpt | 31 ++++++++++++++++++++ 2 files changed, 76 insertions(+), 28 deletions(-) create mode 100644 tests/sma001.phpt diff --git a/apc_sma.c b/apc_sma.c index 8833c72b..fbb662c3 100644 --- a/apc_sma.c +++ b/apc_sma.c @@ -108,9 +108,48 @@ struct block_t { #define RESET_CANARY(v) #endif -/* {{{ MINBLOCKSIZE */ #define MINBLOCKSIZE (ALIGNWORD(1) + ALIGNWORD(sizeof(block_t))) -/* }}} */ + +/* How many extra blocks to check for a better fit */ +#define BEST_FIT_LIMIT 3 + +static inline block_t *find_block(sma_header_t *header, size_t realsize) { + void *shmaddr = header; + block_t *cur, *prv = BLOCKAT(ALIGNWORD(sizeof(sma_header_t))); + block_t *found = NULL; + uint32_t i; + CHECK_CANARY(prv); + + while (prv->fnext) { + cur = BLOCKAT(prv->fnext); + CHECK_CANARY(cur); + + /* Found a suitable block */ + if (cur->size >= realsize) { + found = cur; + break; + } + + prv = cur; + } + + if (found) { + /* Try to find a smaller block that also fits */ + prv = cur; + for (i = 0; i < BEST_FIT_LIMIT && prv->fnext; i++) { + cur = BLOCKAT(prv->fnext); + CHECK_CANARY(cur); + + if (cur->size >= realsize && cur->size < found->size) { + found = cur; + } + + prv = cur; + } + } + + return found; +} /* {{{ sma_allocate: tries to allocate at least size bytes in a segment */ static APC_HOTSPOT size_t sma_allocate(sma_header_t *header, size_t size, size_t fragment, size_t *allocated) @@ -118,7 +157,6 @@ static APC_HOTSPOT size_t sma_allocate(sma_header_t *header, size_t size, size_t void* shmaddr; /* header of shared memory segment */ block_t* prv; /* block prior to working block */ block_t* cur; /* working block in list */ - block_t* prvnextfit; /* block before next fit */ size_t realsize; /* actual size of block needed, including header */ size_t block_size = ALIGNWORD(sizeof(struct block_t)); @@ -134,37 +172,16 @@ static APC_HOTSPOT size_t sma_allocate(sma_header_t *header, size_t size, size_t return -1; } - prvnextfit = 0; /* initially null (no fit) */ - prv = BLOCKAT(ALIGNWORD(sizeof(sma_header_t))); - - CHECK_CANARY(prv); - - while (prv->fnext != 0) { - cur = BLOCKAT(prv->fnext); - - CHECK_CANARY(cur); - - /* If it can fit realsize bytes in cur block, stop searching */ - if (cur->size >= realsize) { - prvnextfit = prv; - break; - } - prv = cur; - } - - if (prvnextfit == 0) { + cur = find_block(header, realsize); + if (!cur) { + /* No suitable block found */ return -1; } - prv = prvnextfit; - cur = BLOCKAT(prv->fnext); - - CHECK_CANARY(prv); - CHECK_CANARY(cur); - if (cur->size == realsize || (cur->size > realsize && cur->size < (realsize + (MINBLOCKSIZE + fragment)))) { /* cur is big enough for realsize, but too small to split - unlink it */ *(allocated) = cur->size - block_size; + prv = BLOCKAT(cur->fprev); prv->fnext = cur->fnext; BLOCKAT(cur->fnext)->fprev = OFFSET(prv); NEXT_SBLOCK(cur)->prev_size = 0; /* block is alloc'd */ diff --git a/tests/sma001.phpt b/tests/sma001.phpt new file mode 100644 index 00000000..dc3ddc34 --- /dev/null +++ b/tests/sma001.phpt @@ -0,0 +1,31 @@ +--TEST-- +Test SMA behavior #1 +--INI-- +apc.enabled=1 +apc.enable_cli=1 +apc.shm_size=16M +--FILE-- + +===DONE=== +--EXPECT-- +===DONE===