Commit 9dd405aa authored by Manfred Spraul's avatar Manfred Spraul Committed by Linus Torvalds

[PATCH] new attempt at sys_poll allocation (was: Re: Poll patches..)

This replaces the dynamically allocated two-level array in sys_poll with
a dynamically allocated linked list.  The current implementation causes
at least two alloc/free calls, even if only one or two descriptors are
polled.  This reduces that to one alloc/free, and the .text segment is
around 220 bytes shorter.  The microbenchmark that polls one pipe fd is
around 30% faster.  [1140 cycles instead of 1604 cycles, Celeron mobile
1.13 GHz]
parent 564dede9
...@@ -360,7 +360,13 @@ sys_select(int n, fd_set *inp, fd_set *outp, fd_set *exp, struct timeval *tvp) ...@@ -360,7 +360,13 @@ sys_select(int n, fd_set *inp, fd_set *outp, fd_set *exp, struct timeval *tvp)
return ret; return ret;
} }
#define POLLFD_PER_PAGE ((PAGE_SIZE) / sizeof(struct pollfd)) struct poll_list {
struct poll_list *next;
int len;
struct pollfd entries[0];
};
#define POLLFD_PER_PAGE ((PAGE_SIZE-sizeof(struct poll_list)) / sizeof(struct pollfd))
static void do_pollfd(unsigned int num, struct pollfd * fdpage, static void do_pollfd(unsigned int num, struct pollfd * fdpage,
poll_table ** pwait, int *count) poll_table ** pwait, int *count)
...@@ -394,21 +400,23 @@ static void do_pollfd(unsigned int num, struct pollfd * fdpage, ...@@ -394,21 +400,23 @@ static void do_pollfd(unsigned int num, struct pollfd * fdpage,
} }
} }
static int do_poll(unsigned int nfds, unsigned int nchunks, unsigned int nleft, static int do_poll(unsigned int nfds, struct poll_list *list,
struct pollfd *fds[], struct poll_wqueues *wait, long timeout) struct poll_wqueues *wait, long timeout)
{ {
int count; int count = 0;
poll_table* pt = &wait->pt; poll_table* pt = &wait->pt;
for (;;) { if (!timeout)
unsigned int i; pt = NULL;
for (;;) {
struct poll_list *walk;
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
count = 0; walk = list;
for (i=0; i < nchunks; i++) while(walk != NULL) {
do_pollfd(POLLFD_PER_PAGE, fds[i], &pt, &count); do_pollfd( walk->len, walk->entries, &pt, &count);
if (nleft) walk = walk->next;
do_pollfd(nleft, fds[nchunks], &pt, &count); }
pt = NULL; pt = NULL;
if (count || !timeout || signal_pending(current)) if (count || !timeout || signal_pending(current))
break; break;
...@@ -423,13 +431,14 @@ static int do_poll(unsigned int nfds, unsigned int nchunks, unsigned int nleft, ...@@ -423,13 +431,14 @@ static int do_poll(unsigned int nfds, unsigned int nchunks, unsigned int nleft,
asmlinkage long sys_poll(struct pollfd * ufds, unsigned int nfds, long timeout) asmlinkage long sys_poll(struct pollfd * ufds, unsigned int nfds, long timeout)
{ {
int i, j, fdcount, err; struct poll_wqueues table;
struct pollfd **fds; int fdcount, err;
struct poll_wqueues table, *wait; unsigned int i;
int nchunks, nleft; struct poll_list *head;
struct poll_list *walk;
/* Do a sanity check on nfds ... */ /* Do a sanity check on nfds ... */
if (nfds > NR_OPEN) if (nfds > current->files->max_fdset && nfds > OPEN_MAX)
return -EINVAL; return -EINVAL;
if (timeout) { if (timeout) {
...@@ -441,68 +450,59 @@ asmlinkage long sys_poll(struct pollfd * ufds, unsigned int nfds, long timeout) ...@@ -441,68 +450,59 @@ asmlinkage long sys_poll(struct pollfd * ufds, unsigned int nfds, long timeout)
} }
poll_initwait(&table); poll_initwait(&table);
wait = &table;
if (!timeout)
wait = NULL;
head = NULL;
walk = NULL;
i = nfds;
err = -ENOMEM; err = -ENOMEM;
fds = NULL; while(i!=0) {
if (nfds != 0) { struct poll_list *pp;
fds = (struct pollfd **)kmalloc( pp = kmalloc(sizeof(struct poll_list)+
(1 + (nfds - 1) / POLLFD_PER_PAGE) * sizeof(struct pollfd *), sizeof(struct pollfd)*
(i>POLLFD_PER_PAGE?POLLFD_PER_PAGE:i),
GFP_KERNEL); GFP_KERNEL);
if (fds == NULL) if(pp==NULL)
goto out;
}
nchunks = 0;
nleft = nfds;
while (nleft > POLLFD_PER_PAGE) { /* allocate complete PAGE_SIZE chunks */
fds[nchunks] = (struct pollfd *)__get_free_page(GFP_KERNEL);
if (fds[nchunks] == NULL)
goto out_fds; goto out_fds;
nchunks++; pp->next=NULL;
nleft -= POLLFD_PER_PAGE; pp->len = (i>POLLFD_PER_PAGE?POLLFD_PER_PAGE:i);
} if (head == NULL)
if (nleft) { /* allocate last PAGE_SIZE chunk, only nleft elements used */ head = pp;
fds[nchunks] = (struct pollfd *)__get_free_page(GFP_KERNEL); else
if (fds[nchunks] == NULL) walk->next = pp;
walk = pp;
if (copy_from_user(pp+1, ufds + nfds-i,
sizeof(struct pollfd)*pp->len)) {
err = -EFAULT;
goto out_fds; goto out_fds;
} }
i -= pp->len;
err = -EFAULT;
for (i=0; i < nchunks; i++)
if (copy_from_user(fds[i], ufds + i*POLLFD_PER_PAGE, PAGE_SIZE))
goto out_fds1;
if (nleft) {
if (copy_from_user(fds[nchunks], ufds + nchunks*POLLFD_PER_PAGE,
nleft * sizeof(struct pollfd)))
goto out_fds1;
} }
fdcount = do_poll(nfds, head, &table, timeout);
fdcount = do_poll(nfds, nchunks, nleft, fds, wait, timeout);
/* OK, now copy the revents fields back to user space. */ /* OK, now copy the revents fields back to user space. */
for(i=0; i < nchunks; i++) walk = head;
for (j=0; j < POLLFD_PER_PAGE; j++, ufds++) err = -EFAULT;
__put_user((fds[i] + j)->revents, &ufds->revents); while(walk != NULL) {
if (nleft) struct pollfd *fds = walk->entries;
for (j=0; j < nleft; j++, ufds++) int j;
__put_user((fds[nchunks] + j)->revents, &ufds->revents);
for (j=0; j < walk->len; j++, ufds++) {
if(__put_user(fds[j].revents, &ufds->revents))
goto out_fds;
}
walk = walk->next;
}
err = fdcount; err = fdcount;
if (!fdcount && signal_pending(current)) if (!fdcount && signal_pending(current))
err = -EINTR; err = -EINTR;
out_fds1:
if (nleft)
free_page((unsigned long)(fds[nchunks]));
out_fds: out_fds:
for (i=0; i < nchunks; i++) walk = head;
free_page((unsigned long)(fds[i])); while(walk!=NULL) {
if (nfds != 0) struct poll_list *pp = walk->next;
kfree(fds); kfree(walk);
out: walk = pp;
}
poll_freewait(&table); poll_freewait(&table);
return err; return err;
} }
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