]>
Commit | Line | Data |
---|---|---|
63ae2a94 | 1 | /* |
fee64d3c | 2 | * Copyright (C) 2000 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com) |
63ae2a94 JD |
3 | * Licensed under the GPL |
4 | */ | |
5 | ||
6 | #include <stdlib.h> | |
63ae2a94 | 7 | #include <errno.h> |
fee64d3c | 8 | #include <poll.h> |
63ae2a94 JD |
9 | #include <signal.h> |
10 | #include <string.h> | |
37185b33 AV |
11 | #include <irq_user.h> |
12 | #include <os.h> | |
13 | #include <um_malloc.h> | |
63ae2a94 | 14 | |
f2e62992 JD |
15 | /* |
16 | * Locked by irq_lock in arch/um/kernel/irq.c. Changed by os_create_pollfd | |
17 | * and os_free_irq_by_cb, which are called under irq_lock. | |
18 | */ | |
63ae2a94 JD |
19 | static struct pollfd *pollfds = NULL; |
20 | static int pollfds_num = 0; | |
21 | static int pollfds_size = 0; | |
22 | ||
23 | int os_waiting_for_events(struct irq_fd *active_fds) | |
24 | { | |
25 | struct irq_fd *irq_fd; | |
26 | int i, n, err; | |
27 | ||
28 | n = poll(pollfds, pollfds_num, 0); | |
191ef966 | 29 | if (n < 0) { |
63ae2a94 | 30 | err = -errno; |
191ef966 | 31 | if (errno != EINTR) |
fee64d3c | 32 | printk(UM_KERN_ERR "os_waiting_for_events:" |
63ae2a94 JD |
33 | " poll returned %d, errno = %d\n", n, errno); |
34 | return err; | |
35 | } | |
36 | ||
191ef966 | 37 | if (n == 0) |
63ae2a94 JD |
38 | return 0; |
39 | ||
40 | irq_fd = active_fds; | |
41 | ||
191ef966 JJ |
42 | for (i = 0; i < pollfds_num; i++) { |
43 | if (pollfds[i].revents != 0) { | |
63ae2a94 JD |
44 | irq_fd->current_events = pollfds[i].revents; |
45 | pollfds[i].fd = -1; | |
46 | } | |
47 | irq_fd = irq_fd->next; | |
48 | } | |
49 | return n; | |
50 | } | |
51 | ||
63ae2a94 JD |
52 | int os_create_pollfd(int fd, int events, void *tmp_pfd, int size_tmpfds) |
53 | { | |
54 | if (pollfds_num == pollfds_size) { | |
55 | if (size_tmpfds <= pollfds_size * sizeof(pollfds[0])) { | |
56 | /* return min size needed for new pollfds area */ | |
f2e62992 | 57 | return (pollfds_size + 1) * sizeof(pollfds[0]); |
63ae2a94 JD |
58 | } |
59 | ||
191ef966 | 60 | if (pollfds != NULL) { |
63ae2a94 JD |
61 | memcpy(tmp_pfd, pollfds, |
62 | sizeof(pollfds[0]) * pollfds_size); | |
63 | /* remove old pollfds */ | |
64 | kfree(pollfds); | |
65 | } | |
66 | pollfds = tmp_pfd; | |
67 | pollfds_size++; | |
191ef966 JJ |
68 | } else |
69 | kfree(tmp_pfd); /* remove not used tmp_pfd */ | |
63ae2a94 | 70 | |
191ef966 JJ |
71 | pollfds[pollfds_num] = ((struct pollfd) { .fd = fd, |
72 | .events = events, | |
73 | .revents = 0 }); | |
63ae2a94 JD |
74 | pollfds_num++; |
75 | ||
191ef966 | 76 | return 0; |
63ae2a94 JD |
77 | } |
78 | ||
79 | void os_free_irq_by_cb(int (*test)(struct irq_fd *, void *), void *arg, | |
80 | struct irq_fd *active_fds, struct irq_fd ***last_irq_ptr2) | |
81 | { | |
82 | struct irq_fd **prev; | |
83 | int i = 0; | |
84 | ||
85 | prev = &active_fds; | |
191ef966 JJ |
86 | while (*prev != NULL) { |
87 | if ((*test)(*prev, arg)) { | |
63ae2a94 | 88 | struct irq_fd *old_fd = *prev; |
191ef966 JJ |
89 | if ((pollfds[i].fd != -1) && |
90 | (pollfds[i].fd != (*prev)->fd)) { | |
fee64d3c JD |
91 | printk(UM_KERN_ERR "os_free_irq_by_cb - " |
92 | "mismatch between active_fds and " | |
93 | "pollfds, fd %d vs %d\n", | |
63ae2a94 JD |
94 | (*prev)->fd, pollfds[i].fd); |
95 | goto out; | |
96 | } | |
97 | ||
98 | pollfds_num--; | |
99 | ||
fee64d3c JD |
100 | /* |
101 | * This moves the *whole* array after pollfds[i] | |
63ae2a94 JD |
102 | * (though it doesn't spot as such)! |
103 | */ | |
63ae2a94 JD |
104 | memmove(&pollfds[i], &pollfds[i + 1], |
105 | (pollfds_num - i) * sizeof(pollfds[0])); | |
fee64d3c | 106 | if (*last_irq_ptr2 == &old_fd->next) |
63ae2a94 JD |
107 | *last_irq_ptr2 = prev; |
108 | ||
109 | *prev = (*prev)->next; | |
fee64d3c | 110 | if (old_fd->type == IRQ_WRITE) |
63ae2a94 JD |
111 | ignore_sigio_fd(old_fd->fd); |
112 | kfree(old_fd); | |
113 | continue; | |
114 | } | |
115 | prev = &(*prev)->next; | |
116 | i++; | |
117 | } | |
118 | out: | |
119 | return; | |
120 | } | |
121 | ||
63ae2a94 JD |
122 | int os_get_pollfd(int i) |
123 | { | |
191ef966 | 124 | return pollfds[i].fd; |
63ae2a94 JD |
125 | } |
126 | ||
127 | void os_set_pollfd(int i, int fd) | |
128 | { | |
129 | pollfds[i].fd = fd; | |
130 | } | |
131 | ||
132 | void os_set_ioignore(void) | |
133 | { | |
4b84c69b | 134 | signal(SIGIO, SIG_IGN); |
63ae2a94 | 135 | } |