iobuf: option to provide larger size of buffers
provide an option to failover to standard allocation if iobuf of required size doesn't exists. this can be achieved by keeping an arena dedicated for all the out of boundary allocations. Change-Id: I41a2bd7d353dc7bcb2e1a6e4b41735afe9865975 Signed-off-by: Amar Tumballi <amarts@redhat.com> BUG: 812784 Reviewed-on: http://review.gluster.com/3136 Tested-by: Gluster Build System <jenkins@build.gluster.com> Reviewed-by: Anand Avati <avati@redhat.com>
This commit is contained in:
parent
d879951ca9
commit
e4b3c0d90c
@ -317,6 +317,31 @@ out:
|
||||
return;
|
||||
}
|
||||
|
||||
static void
|
||||
iobuf_create_stdalloc_arena (struct iobuf_pool *iobuf_pool)
|
||||
{
|
||||
struct iobuf_arena *iobuf_arena = NULL;
|
||||
|
||||
/* No locking required here as its called only once during init */
|
||||
iobuf_arena = GF_CALLOC (sizeof (*iobuf_arena), 1,
|
||||
gf_common_mt_iobuf_arena);
|
||||
if (!iobuf_arena)
|
||||
goto err;
|
||||
|
||||
INIT_LIST_HEAD (&iobuf_arena->list);
|
||||
INIT_LIST_HEAD (&iobuf_arena->active.list);
|
||||
INIT_LIST_HEAD (&iobuf_arena->passive.list);
|
||||
|
||||
iobuf_arena->iobuf_pool = iobuf_pool;
|
||||
|
||||
iobuf_arena->page_size = 0x7fffffff;
|
||||
|
||||
list_add_tail (&iobuf_arena->list,
|
||||
&iobuf_pool->arenas[IOBUF_ARENA_MAX_INDEX]);
|
||||
|
||||
err:
|
||||
return;
|
||||
}
|
||||
|
||||
struct iobuf_pool *
|
||||
iobuf_pool_new (void)
|
||||
@ -333,7 +358,7 @@ iobuf_pool_new (void)
|
||||
goto out;
|
||||
|
||||
pthread_mutex_init (&iobuf_pool->mutex, NULL);
|
||||
for (i = 0; i < IOBUF_ARENA_MAX_INDEX; i++) {
|
||||
for (i = 0; i <= IOBUF_ARENA_MAX_INDEX; i++) {
|
||||
INIT_LIST_HEAD (&iobuf_pool->arenas[i]);
|
||||
INIT_LIST_HEAD (&iobuf_pool->filled[i]);
|
||||
INIT_LIST_HEAD (&iobuf_pool->purge[i]);
|
||||
@ -351,6 +376,9 @@ iobuf_pool_new (void)
|
||||
arena_size += page_size * num_pages;
|
||||
}
|
||||
|
||||
/* Need an arena to handle all the bigger iobuf requests */
|
||||
iobuf_create_stdalloc_arena (iobuf_pool);
|
||||
|
||||
iobuf_pool->arena_size = arena_size;
|
||||
out:
|
||||
|
||||
@ -508,6 +536,51 @@ out:
|
||||
return iobuf;
|
||||
}
|
||||
|
||||
struct iobuf *
|
||||
iobuf_get_from_stdalloc (struct iobuf_pool *iobuf_pool, size_t page_size)
|
||||
{
|
||||
struct iobuf *iobuf = NULL;
|
||||
struct iobuf_arena *iobuf_arena = NULL;
|
||||
struct iobuf_arena *trav = NULL;
|
||||
int ret = -1;
|
||||
|
||||
/* The first arena in the 'MAX-INDEX' will always be used for misc */
|
||||
list_for_each_entry (trav, &iobuf_pool->arenas[IOBUF_ARENA_MAX_INDEX],
|
||||
list) {
|
||||
iobuf_arena = trav;
|
||||
break;
|
||||
}
|
||||
|
||||
iobuf = GF_CALLOC (1, sizeof (*iobuf), gf_common_mt_iobuf);
|
||||
if (!iobuf)
|
||||
goto out;
|
||||
|
||||
/* 4096 is the alignment */
|
||||
iobuf->free_ptr = GF_CALLOC (1, ((page_size + GF_IOBUF_ALIGN_SIZE) - 1),
|
||||
gf_common_mt_char);
|
||||
if (!iobuf->free_ptr)
|
||||
goto out;
|
||||
|
||||
iobuf->ptr = GF_ALIGN_BUF (iobuf->free_ptr, GF_IOBUF_ALIGN_SIZE);
|
||||
iobuf->iobuf_arena = iobuf_arena;
|
||||
LOCK_INIT (&iobuf->lock);
|
||||
|
||||
/* Hold a ref because you are allocating and using it */
|
||||
iobuf->ref = 1;
|
||||
|
||||
ret = 0;
|
||||
out:
|
||||
if (ret && iobuf) {
|
||||
if (iobuf->free_ptr)
|
||||
GF_FREE (iobuf->free_ptr);
|
||||
GF_FREE (iobuf);
|
||||
iobuf = NULL;
|
||||
}
|
||||
|
||||
return iobuf;
|
||||
}
|
||||
|
||||
|
||||
struct iobuf *
|
||||
iobuf_get2 (struct iobuf_pool *iobuf_pool, size_t page_size)
|
||||
{
|
||||
@ -521,11 +594,17 @@ iobuf_get2 (struct iobuf_pool *iobuf_pool, size_t page_size)
|
||||
|
||||
rounded_size = gf_iobuf_get_pagesize (page_size);
|
||||
if (rounded_size == -1) {
|
||||
gf_log ("iobuf", GF_LOG_ERROR, "page_size (%zu) of "
|
||||
"iobufs in arena being requested is greater than max "
|
||||
"available", page_size);
|
||||
/* make sure to provide the requested buffer with standard
|
||||
memory allocations */
|
||||
iobuf = iobuf_get_from_stdalloc (iobuf_pool, page_size);
|
||||
|
||||
gf_log ("iobuf", GF_LOG_DEBUG, "request for iobuf of size %zu "
|
||||
"is serviced using standard calloc() (%p) as it "
|
||||
"exceeds the maximum available buffer size",
|
||||
page_size, iobuf);
|
||||
|
||||
iobuf_pool->request_misses++;
|
||||
return NULL;
|
||||
return iobuf;
|
||||
}
|
||||
|
||||
pthread_mutex_lock (&iobuf_pool->mutex);
|
||||
@ -594,9 +673,13 @@ __iobuf_put (struct iobuf *iobuf, struct iobuf_arena *iobuf_arena)
|
||||
|
||||
index = gf_iobuf_get_arena_index (iobuf_arena->page_size);
|
||||
if (index == -1) {
|
||||
gf_log ("iobuf", GF_LOG_ERROR, "page_size (%zu) of "
|
||||
"iobufs in arena being added is greater than max "
|
||||
"available", iobuf_arena->page_size);
|
||||
gf_log ("iobuf", GF_LOG_DEBUG, "freeing the iobuf (%p) "
|
||||
"allocated with standard calloc()", iobuf);
|
||||
|
||||
/* free up properly without bothering about lists and all */
|
||||
LOCK_DESTROY (&iobuf->lock);
|
||||
GF_FREE (iobuf->free_ptr);
|
||||
GF_FREE (iobuf);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -40,6 +40,11 @@
|
||||
#define MAP_ANONYMOUS MAP_ANON
|
||||
#endif
|
||||
|
||||
#define GF_ALIGN_BUF(ptr,bound) ((void *)((unsigned long)(ptr + bound - 1) & \
|
||||
(unsigned long)(~(bound - 1))))
|
||||
|
||||
#define GF_IOBUF_ALIGN_SIZE 512
|
||||
|
||||
/* one allocatable unit for the consumers of the IOBUF API */
|
||||
/* each unit hosts @page_size bytes of memory */
|
||||
struct iobuf;
|
||||
@ -71,6 +76,9 @@ struct iobuf {
|
||||
int ref; /* 0 == passive, >0 == active */
|
||||
|
||||
void *ptr; /* usable memory region by the consumer */
|
||||
|
||||
void *free_ptr; /* in case of stdalloc, this is the
|
||||
one to be freed */
|
||||
};
|
||||
|
||||
|
||||
|
@ -1890,9 +1890,6 @@ out:
|
||||
return 0;
|
||||
}
|
||||
|
||||
#define ALIGN_BUF(ptr,bound) ((void *)((unsigned long)(ptr + bound - 1) & \
|
||||
(unsigned long)(~(bound - 1))))
|
||||
|
||||
int
|
||||
posix_readv (call_frame_t *frame, xlator_t *this,
|
||||
fd_t *fd, size_t size, off_t offset, uint32_t flags, dict_t *xdata)
|
||||
@ -2055,7 +2052,7 @@ __posix_writev (int fd, struct iovec *vector, int count, off_t startoff,
|
||||
internal_off = startoff;
|
||||
for (idx = 0; idx < count; idx++) {
|
||||
/* page aligned buffer */
|
||||
buf = ALIGN_BUF (alloc_buf, align);
|
||||
buf = GF_ALIGN_BUF (alloc_buf, align);
|
||||
|
||||
memcpy (buf, vector[idx].iov_base, vector[idx].iov_len);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user