Commit 5466b456 authored by Vadim Lobanov's avatar Vadim Lobanov Committed by Linus Torvalds

[PATCH] fdtable: Implement new pagesize-based fdtable allocator

This patch provides an improved fdtable allocation scheme, useful for
expanding fdtable file descriptor entries.  The main focus is on the fdarray,
as its memory usage grows 128 times faster than that of an fdset.

The allocation algorithm sizes the fdarray in such a way that its memory usage
increases in easy page-sized chunks. The overall algorithm expands the allowed
size in powers of two, in order to amortize the cost of invoking vmalloc() for
larger allocation sizes. Namely, the following sizes for the fdarray are
considered, and the smallest that accommodates the requested fd count is
chosen:

    pagesize / 4
    pagesize / 2
    pagesize      <- memory allocator switch point
    pagesize * 2
    pagesize * 4
    ...etc...

Unlike the current implementation, this allocation scheme does not require a
loop to compute the optimal fdarray size, and can be done in efficient
straightline code.

Furthermore, since the fdarray overflows the pagesize boundary long before any
of the fdsets do, it makes sense to optimize run-time by allocating both
fdsets in a single swoop.  Even together, they will still be, by far, smaller
than the fdarray.  The fdtable->open_fds is now used as the anchor for the
fdset memory allocation.
Signed-off-by: default avatarVadim Lobanov <vlobanov@speakeasy.net>
Cc: Christoph Hellwig <hch@lst.de>
Cc: Al Viro <viro@zeniv.linux.org.uk>
Cc: Dipankar Sarma <dipankar@in.ibm.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 4fd45812
...@@ -32,46 +32,28 @@ struct fdtable_defer { ...@@ -32,46 +32,28 @@ struct fdtable_defer {
*/ */
static DEFINE_PER_CPU(struct fdtable_defer, fdtable_defer_list); static DEFINE_PER_CPU(struct fdtable_defer, fdtable_defer_list);
static inline void * alloc_fdmem(unsigned int size)
/*
* Allocate an fd array, using kmalloc or vmalloc.
* Note: the array isn't cleared at allocation time.
*/
struct file ** alloc_fd_array(int num)
{ {
struct file **new_fds;
int size = num * sizeof(struct file *);
if (size <= PAGE_SIZE) if (size <= PAGE_SIZE)
new_fds = (struct file **) kmalloc(size, GFP_KERNEL); return kmalloc(size, GFP_KERNEL);
else else
new_fds = (struct file **) vmalloc(size); return vmalloc(size);
return new_fds;
} }
void free_fd_array(struct file **array, int num) static inline void free_fdarr(struct fdtable *fdt)
{ {
int size = num * sizeof(struct file *); if (fdt->max_fds <= (PAGE_SIZE / sizeof(struct file *)))
kfree(fdt->fd);
if (!array) {
printk (KERN_ERR "free_fd_array: array = 0 (num = %d)\n", num);
return;
}
if (num <= NR_OPEN_DEFAULT) /* Don't free the embedded fd array! */
return;
else if (size <= PAGE_SIZE)
kfree(array);
else else
vfree(array); vfree(fdt->fd);
} }
static void __free_fdtable(struct fdtable *fdt) static inline void free_fdset(struct fdtable *fdt)
{ {
free_fdset(fdt->open_fds, fdt->max_fds); if (fdt->max_fds <= (PAGE_SIZE * BITS_PER_BYTE / 2))
free_fdset(fdt->close_on_exec, fdt->max_fds); kfree(fdt->open_fds);
free_fd_array(fdt->fd, fdt->max_fds); else
kfree(fdt); vfree(fdt->open_fds);
} }
static void free_fdtable_work(struct work_struct *work) static void free_fdtable_work(struct work_struct *work)
...@@ -86,7 +68,9 @@ static void free_fdtable_work(struct work_struct *work) ...@@ -86,7 +68,9 @@ static void free_fdtable_work(struct work_struct *work)
spin_unlock_bh(&f->lock); spin_unlock_bh(&f->lock);
while(fdt) { while(fdt) {
struct fdtable *next = fdt->next; struct fdtable *next = fdt->next;
__free_fdtable(fdt); vfree(fdt->fd);
free_fdset(fdt);
kfree(fdt);
fdt = next; fdt = next;
} }
} }
...@@ -94,12 +78,9 @@ static void free_fdtable_work(struct work_struct *work) ...@@ -94,12 +78,9 @@ static void free_fdtable_work(struct work_struct *work)
void free_fdtable_rcu(struct rcu_head *rcu) void free_fdtable_rcu(struct rcu_head *rcu)
{ {
struct fdtable *fdt = container_of(rcu, struct fdtable, rcu); struct fdtable *fdt = container_of(rcu, struct fdtable, rcu);
int fdset_size, fdarray_size;
struct fdtable_defer *fddef; struct fdtable_defer *fddef;
BUG_ON(!fdt); BUG_ON(!fdt);
fdset_size = fdt->max_fds / 8;
fdarray_size = fdt->max_fds * sizeof(struct file *);
if (fdt->max_fds <= NR_OPEN_DEFAULT) { if (fdt->max_fds <= NR_OPEN_DEFAULT) {
/* /*
...@@ -110,10 +91,9 @@ void free_fdtable_rcu(struct rcu_head *rcu) ...@@ -110,10 +91,9 @@ void free_fdtable_rcu(struct rcu_head *rcu)
container_of(fdt, struct files_struct, fdtab)); container_of(fdt, struct files_struct, fdtab));
return; return;
} }
if (fdset_size <= PAGE_SIZE && fdarray_size <= PAGE_SIZE) { if (fdt->max_fds <= (PAGE_SIZE / sizeof(struct file *))) {
kfree(fdt->open_fds);
kfree(fdt->close_on_exec);
kfree(fdt->fd); kfree(fdt->fd);
kfree(fdt->open_fds);
kfree(fdt); kfree(fdt);
} else { } else {
fddef = &get_cpu_var(fdtable_defer_list); fddef = &get_cpu_var(fdtable_defer_list);
...@@ -131,116 +111,70 @@ void free_fdtable_rcu(struct rcu_head *rcu) ...@@ -131,116 +111,70 @@ void free_fdtable_rcu(struct rcu_head *rcu)
* Expand the fdset in the files_struct. Called with the files spinlock * Expand the fdset in the files_struct. Called with the files spinlock
* held for write. * held for write.
*/ */
static void copy_fdtable(struct fdtable *nfdt, struct fdtable *fdt) static void copy_fdtable(struct fdtable *nfdt, struct fdtable *ofdt)
{
int i;
int count;
BUG_ON(nfdt->max_fds < fdt->max_fds);
/* Copy the existing tables and install the new pointers */
i = fdt->max_fds / (sizeof(unsigned long) * 8);
count = (nfdt->max_fds - fdt->max_fds) / 8;
/*
* Don't copy the entire array if the current fdset is
* not yet initialised.
*/
if (i) {
memcpy (nfdt->open_fds, fdt->open_fds,
fdt->max_fds/8);
memcpy (nfdt->close_on_exec, fdt->close_on_exec,
fdt->max_fds/8);
memset (&nfdt->open_fds->fds_bits[i], 0, count);
memset (&nfdt->close_on_exec->fds_bits[i], 0, count);
}
/* Don't copy/clear the array if we are creating a new
fd array for fork() */
if (fdt->max_fds) {
memcpy(nfdt->fd, fdt->fd,
fdt->max_fds * sizeof(struct file *));
/* clear the remainder of the array */
memset(&nfdt->fd[fdt->max_fds], 0,
(nfdt->max_fds - fdt->max_fds) *
sizeof(struct file *));
}
}
/*
* Allocate an fdset array, using kmalloc or vmalloc.
* Note: the array isn't cleared at allocation time.
*/
fd_set * alloc_fdset(int num)
{ {
fd_set *new_fdset; unsigned int cpy, set;
int size = num / 8;
if (size <= PAGE_SIZE) BUG_ON(nfdt->max_fds < ofdt->max_fds);
new_fdset = (fd_set *) kmalloc(size, GFP_KERNEL); if (ofdt->max_fds == 0)
else
new_fdset = (fd_set *) vmalloc(size);
return new_fdset;
}
void free_fdset(fd_set *array, int num)
{
if (num <= NR_OPEN_DEFAULT) /* Don't free an embedded fdset */
return; return;
else if (num <= 8 * PAGE_SIZE)
kfree(array); cpy = ofdt->max_fds * sizeof(struct file *);
else set = (nfdt->max_fds - ofdt->max_fds) * sizeof(struct file *);
vfree(array); memcpy(nfdt->fd, ofdt->fd, cpy);
memset((char *)(nfdt->fd) + cpy, 0, set);
cpy = ofdt->max_fds / BITS_PER_BYTE;
set = (nfdt->max_fds - ofdt->max_fds) / BITS_PER_BYTE;
memcpy(nfdt->open_fds, ofdt->open_fds, cpy);
memset((char *)(nfdt->open_fds) + cpy, 0, set);
memcpy(nfdt->close_on_exec, ofdt->close_on_exec, cpy);
memset((char *)(nfdt->close_on_exec) + cpy, 0, set);
} }
static struct fdtable *alloc_fdtable(int nr) static struct fdtable * alloc_fdtable(unsigned int nr)
{ {
struct fdtable *fdt = NULL; struct fdtable *fdt;
int nfds = 0; char *data;
fd_set *new_openset = NULL, *new_execset = NULL;
struct file **new_fds;
fdt = kzalloc(sizeof(*fdt), GFP_KERNEL);
if (!fdt)
goto out;
nfds = NR_OPEN_DEFAULT;
/* /*
* Expand to the max in easy steps, and keep expanding it until * Figure out how many fds we actually want to support in this fdtable.
* we have enough for the requested fd array size. * Allocation steps are keyed to the size of the fdarray, since it
* grows far faster than any of the other dynamic data. We try to fit
* the fdarray into comfortable page-tuned chunks: starting at 1024B
* and growing in powers of two from there on.
*/ */
do { nr /= (1024 / sizeof(struct file *));
#if NR_OPEN_DEFAULT < 256 nr = roundup_pow_of_two(nr + 1);
if (nfds < 256) nr *= (1024 / sizeof(struct file *));
nfds = 256; if (nr > NR_OPEN)
else nr = NR_OPEN;
#endif
if (nfds < (PAGE_SIZE / sizeof(struct file *)))
nfds = PAGE_SIZE / sizeof(struct file *);
else {
nfds = nfds * 2;
if (nfds > NR_OPEN)
nfds = NR_OPEN;
}
} while (nfds <= nr);
new_openset = alloc_fdset(nfds); fdt = kmalloc(sizeof(struct fdtable), GFP_KERNEL);
new_execset = alloc_fdset(nfds); if (!fdt)
if (!new_openset || !new_execset)
goto out; goto out;
fdt->open_fds = new_openset; fdt->max_fds = nr;
fdt->close_on_exec = new_execset; data = alloc_fdmem(nr * sizeof(struct file *));
if (!data)
goto out_fdt;
fdt->fd = (struct file **)data;
data = alloc_fdmem(max_t(unsigned int,
2 * nr / BITS_PER_BYTE, L1_CACHE_BYTES));
if (!data)
goto out_arr;
fdt->open_fds = (fd_set *)data;
data += nr / BITS_PER_BYTE;
fdt->close_on_exec = (fd_set *)data;
INIT_RCU_HEAD(&fdt->rcu);
fdt->next = NULL;
new_fds = alloc_fd_array(nfds);
if (!new_fds)
goto out;
fdt->fd = new_fds;
fdt->max_fds = nfds;
return fdt; return fdt;
out:
free_fdset(new_openset, nfds); out_arr:
free_fdset(new_execset, nfds); free_fdarr(fdt);
out_fdt:
kfree(fdt); kfree(fdt);
out:
return NULL; return NULL;
} }
...@@ -275,7 +209,9 @@ static int expand_fdtable(struct files_struct *files, int nr) ...@@ -275,7 +209,9 @@ static int expand_fdtable(struct files_struct *files, int nr)
call_rcu(&cur_fdt->rcu, free_fdtable_rcu); call_rcu(&cur_fdt->rcu, free_fdtable_rcu);
} else { } else {
/* Somebody else expanded, so undo our attempt */ /* Somebody else expanded, so undo our attempt */
__free_fdtable(new_fdt); free_fdarr(new_fdt);
free_fdset(new_fdt);
kfree(new_fdt);
} }
return 1; return 1;
} }
......
...@@ -76,12 +76,6 @@ extern int get_unused_fd(void); ...@@ -76,12 +76,6 @@ extern int get_unused_fd(void);
extern void FASTCALL(put_unused_fd(unsigned int fd)); extern void FASTCALL(put_unused_fd(unsigned int fd));
struct kmem_cache; struct kmem_cache;
extern struct file ** alloc_fd_array(int);
extern void free_fd_array(struct file **, int);
extern fd_set *alloc_fdset(int);
extern void free_fdset(fd_set *, int);
extern int expand_files(struct files_struct *, int nr); extern int expand_files(struct files_struct *, int nr);
extern void free_fdtable_rcu(struct rcu_head *rcu); extern void free_fdtable_rcu(struct rcu_head *rcu);
extern void __init files_defer_init(void); extern void __init files_defer_init(void);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment