panic() cleanup.
[minix.git] / servers / vfs / select.c
blob550acc40dc70d207f4235326b3454ae407b48a59
1 /* Implement entry point to select system call.
3 * The entry points into this file are
4 * do_select: perform the SELECT system call
5 * select_callback: notify select system of possible fd operation
6 * select_notified: low-level entry for device notifying select
7 * select_unsuspend_by_endpt: cancel a blocking select on exiting driver
8 */
10 #define DEBUG_SELECT 0
12 #include "fs.h"
13 #include "select.h"
14 #include "file.h"
15 #include "vnode.h"
17 #include <sys/time.h>
18 #include <sys/select.h>
19 #include <minix/com.h>
20 #include <minix/u64.h>
21 #include <string.h>
23 /* max. number of simultaneously pending select() calls */
24 #define MAXSELECTS 25
25 #define FROM_PROC 0
26 #define TO_PROC 1
28 PRIVATE struct selectentry {
29 struct fproc *requestor; /* slot is free iff this is NULL */
30 int req_endpt;
31 fd_set readfds, writefds, errorfds;
32 fd_set ready_readfds, ready_writefds, ready_errorfds;
33 fd_set *vir_readfds, *vir_writefds, *vir_errorfds;
34 struct filp *filps[OPEN_MAX];
35 int type[OPEN_MAX];
36 int deferred; /* awaiting initial reply from driver */
37 int deferred_fd; /* fd awaiting initial reply from driver */
38 int nfds, nreadyfds;
39 char block;
40 clock_t expiry;
41 timer_t timer; /* if expiry > 0 */
42 } selecttab[MAXSELECTS];
44 FORWARD _PROTOTYPE(int copy_fdsets, (struct selectentry *se, int nfds,
45 int direction) );
46 FORWARD _PROTOTYPE(void filp_status, (struct filp *fp, int status) );
47 FORWARD _PROTOTYPE(void restart_proc, (int slot) );
48 FORWARD _PROTOTYPE(void ops2tab, (int ops, int fd, struct selectentry *e));
49 FORWARD _PROTOTYPE(int select_reevaluate, (struct filp *fp) );
50 FORWARD _PROTOTYPE(int select_request_file, (struct filp *f, int *ops,
51 int block) );
52 FORWARD _PROTOTYPE(int select_match_file, (struct filp *f) );
53 FORWARD _PROTOTYPE(int select_request_general, (struct filp *f, int *ops,
54 int block) );
55 FORWARD _PROTOTYPE(int select_request_asynch, (struct filp *f, int *ops,
56 int block) );
57 FORWARD _PROTOTYPE(int select_major_match, (int match_major,
58 struct filp *file) );
59 FORWARD _PROTOTYPE(void select_cancel_all, (struct selectentry *e) );
60 FORWARD _PROTOTYPE(void select_wakeup, (struct selectentry *e, int r) );
61 FORWARD _PROTOTYPE(void select_return, (struct selectentry *, int) );
62 FORWARD _PROTOTYPE(void sel_restart_dev, (void) );
63 FORWARD _PROTOTYPE(int tab2ops, (int fd, struct selectentry *e) );
64 FORWARD _PROTOTYPE(void wipe_select, (struct selectentry *s) );
66 PRIVATE struct fdtype {
67 int (*select_request)(struct filp *, int *ops, int block);
68 int (*select_match)(struct filp *);
69 int select_major;
70 } fdtypes[] = {
71 { select_request_file, select_match_file, 0 },
72 { select_request_general, NULL, TTY_MAJOR },
73 { select_request_general, NULL, INET_MAJOR },
74 { select_request_pipe, select_match_pipe, 0 },
75 { select_request_asynch, NULL, LOG_MAJOR },
77 #define SEL_FDS (sizeof(fdtypes) / sizeof(fdtypes[0]))
79 /*===========================================================================*
80 * do_select *
81 *===========================================================================*/
82 PUBLIC int do_select(void)
84 /* Implement the select(nfds, readfds, writefds, errorfds, timeout) system
85 * call. First we copy the arguments and verify their sanity. Then we check
86 * whether there are file descriptors that satisfy the select call right of the
87 * bat. If so, or if there are no ready file descriptors but the process
88 * requested to return immediately, we return the result. Otherwise we set a
89 * timeout and wait for either the file descriptors to become ready or the
90 * timer to go off. If no timeout value was provided, we wait indefinitely. */
92 int r, nfds, do_timeout = 0, nonzero_timeout = 0, fd, s;
93 struct timeval timeout;
94 struct selectentry *se;
96 nfds = m_in.SEL_NFDS;
98 /* Sane amount of file descriptors? */
99 if (nfds < 0 || nfds > OPEN_MAX) return(EINVAL);
101 /* Find a slot to store this select request */
102 for (s = 0; s < MAXSELECTS; s++)
103 if (selecttab[s].requestor == NULL) /* Unused slot */
104 break;
105 if (s >= MAXSELECTS) return(ENOSPC);
107 se = &selecttab[s];
108 wipe_select(se); /* Clear results of previous usage.*/
109 se->req_endpt = who_e;
110 se->vir_readfds = (fd_set *) m_in.SEL_READFDS;
111 se->vir_writefds = (fd_set *) m_in.SEL_WRITEFDS;
112 se->vir_errorfds = (fd_set *) m_in.SEL_ERRORFDS;
114 /* Copy fdsets from the process */
115 if ((r = copy_fdsets(se, nfds, FROM_PROC)) != OK) return(r);
117 /* Did the process set a timeout value? If so, retrieve it. */
118 if (m_in.SEL_TIMEOUT != NULL) {
119 do_timeout = 1;
120 r = sys_vircopy(who_e, D, (vir_bytes) m_in.SEL_TIMEOUT, SELF, D,
121 (vir_bytes) &timeout, sizeof(timeout));
122 if (r != OK) return(r);
125 /* No nonsense in the timeval. */
126 if(do_timeout && (timeout.tv_sec < 0 || timeout.tv_usec < 0)) return(EINVAL);
128 /* If there is no timeout, we block forever. Otherwise, we block up to the
129 * specified time interval.
131 if(!do_timeout) /* No timeout value set */
132 se->block = 1;
133 else if (do_timeout && (timeout.tv_sec > 0 || timeout.tv_usec > 0))
134 se->block = 1;
135 else /* timeout set as (0,0) - this effects a poll */
136 se->block = 0;
137 se->expiry = 0; /* no timer set (yet) */
139 /* Check all file descriptors in the set whether one is 'ready' now. */
140 for (fd = 0; fd < nfds; fd++) {
141 int ops, t, type = -1, r;
142 struct filp *filp;
144 if (!(ops = tab2ops(fd, se)))
145 continue; /* No operations set; nothing to do for this fd */
147 /* Get filp belonging to this fd */
148 filp = se->filps[fd] = get_filp(fd);
149 if (filp == NIL_FILP) {
150 if (err_code == EBADF) {
151 select_cancel_all(se);
152 return(EBADF);
155 /* File descriptor is 'ready' to return EIO */
156 printf("VFS do_select: EIO after driver failure\n");
157 ops2tab(SEL_RD|SEL_WR|SEL_ERR, fd, se);
158 continue;
161 /* Figure out what type of file we're dealing with */
162 for(t = 0; t < SEL_FDS; t++) {
163 if (fdtypes[t].select_match) {
164 if (fdtypes[t].select_match(filp)) {
165 type = t;
167 } else if (select_major_match(fdtypes[t].select_major, filp)) {
168 type = t;
172 if (type == -1) return(EBADF);
173 se->type[fd] = type;
175 /* Test filp for select operations if not already done so. e.g., files
176 * sharing a filp and both doing a select on that filp. */
177 if ((se->filps[fd]->filp_select_ops & ops) != ops) {
178 int wantops;
180 wantops = (se->filps[fd]->filp_select_ops |= ops);
181 r = fdtypes[type].select_request(filp, &wantops, se->block);
182 if (r != SEL_OK) {
183 if (r == SEL_DEFERRED) {
184 se->deferred = TRUE;
185 se->deferred_fd = 0;
186 continue;
189 /* Error or bogus return code; cancel select. */
190 select_cancel_all(se);
191 return(EINVAL);
194 /* The select request above might have turned on/off some
195 * operations because they were 'ready' or not meaningful.
196 * Either way, we might have a result and we need to store them
197 * in the select table entry. */
198 if (wantops & ops) ops2tab(wantops, fd, se);
201 se->nfds = fd+1;
202 se->filps[fd]->filp_selectors++;
205 if (se->nreadyfds > 0 || (!se->block && !se->deferred)) {
206 /* fd's were found that were ready to go right away, and/or
207 * we were instructed not to block at all. Must return
208 * immediately.
210 r = copy_fdsets(se, se->nfds, TO_PROC);
211 select_cancel_all(se);
212 se->requestor = NULL;
214 if (r != OK) return(r);
215 else return(se->nreadyfds);
218 /* Convert timeval to ticks and set the timer. If it fails, undo
219 * all, return error.
221 if (do_timeout) {
222 int ticks;
223 /* Open Group:
224 * "If the requested timeout interval requires a finer
225 * granularity than the implementation supports, the
226 * actual timeout interval shall be rounded up to the next
227 * supported value."
229 #define USECPERSEC 1000000
230 while(timeout.tv_usec >= USECPERSEC) {
231 /* this is to avoid overflow with *system_hz below */
232 timeout.tv_usec -= USECPERSEC;
233 timeout.tv_sec++;
235 ticks = timeout.tv_sec * system_hz +
236 (timeout.tv_usec * system_hz + USECPERSEC-1) / USECPERSEC;
237 se->expiry = ticks;
238 fs_set_timer(&se->timer, ticks, select_timeout_check, s);
241 /* if we're blocking, the table entry is now valid. */
242 se->requestor = fp;
244 /* process now blocked */
245 suspend(FP_BLOCKED_ON_SELECT);
246 return(SUSPEND);
250 /*===========================================================================*
251 * select_request_file *
252 *===========================================================================*/
253 PRIVATE int select_request_file(struct filp *f, int *ops, int block)
255 /* output *ops is input *ops */
256 return(SEL_OK);
260 /*===========================================================================*
261 * select_match_file *
262 *===========================================================================*/
263 PRIVATE int select_match_file(struct filp *file)
265 return(file && file->filp_vno && (file->filp_vno->v_mode & I_REGULAR));
269 /*===========================================================================*
270 * select_request_general *
271 *===========================================================================*/
272 PRIVATE int select_request_general(struct filp *f, int *ops, int block)
274 int rops = *ops;
275 if (block) rops |= SEL_NOTIFY;
276 *ops = dev_io(VFS_DEV_SELECT, f->filp_vno->v_sdev, rops, NULL,
277 cvu64(0), 0, 0, FALSE);
278 if (*ops < 0)
279 return(SEL_ERR);
281 return(SEL_OK);
285 /*===========================================================================*
286 * select_request_asynch *
287 *===========================================================================*/
288 PRIVATE int select_request_asynch(struct filp *f, int *ops, int block)
290 int r, rops;
291 struct dmap *dp;
293 rops = *ops;
294 f->filp_select_flags |= FSF_UPDATE;
295 if (block) {
296 rops |= SEL_NOTIFY;
297 f->filp_select_flags |= FSF_BLOCK;
300 if (f->filp_select_flags & FSF_BUSY)
301 return(SEL_DEFERRED);
303 dp = &dmap[((f->filp_vno->v_sdev) >> MAJOR) & BYTE];
304 if (dp->dmap_sel_filp)
305 return(SEL_DEFERRED);
307 f->filp_select_flags &= ~FSF_UPDATE;
308 r = dev_io(VFS_DEV_SELECT, f->filp_vno->v_sdev, rops, NULL,
309 cvu64(0), 0, 0, FALSE);
310 if (r < 0 && r != SUSPEND)
311 return(SEL_ERR);
313 if (r != SUSPEND)
314 panic("select_request_asynch: expected SUSPEND got: %d", r);
316 f->filp_count++;
317 dp->dmap_sel_filp = f;
318 f->filp_select_flags |= FSF_BUSY;
320 return(SEL_DEFERRED);
324 /*===========================================================================*
325 * select_major_match *
326 *===========================================================================*/
327 PRIVATE int select_major_match(int match_major, struct filp *file)
329 int major;
330 if (!(file && file->filp_vno &&
331 (file->filp_vno->v_mode & I_TYPE) == I_CHAR_SPECIAL))
332 return(0);
333 major = (file->filp_vno->v_sdev >> MAJOR) & BYTE;
334 if (major == match_major) return 1;
335 return 0;
339 /*===========================================================================*
340 * tab2ops *
341 *===========================================================================*/
342 PRIVATE int tab2ops(int fd, struct selectentry *e)
344 int ops = 0;
345 if (FD_ISSET(fd, &e->readfds)) ops |= SEL_RD;
346 if (FD_ISSET(fd, &e->writefds)) ops |= SEL_WR;
347 if (FD_ISSET(fd, &e->errorfds)) ops |= SEL_ERR;
349 return(ops);
353 /*===========================================================================*
354 * ops2tab *
355 *===========================================================================*/
356 PRIVATE void ops2tab(int ops, int fd, struct selectentry *e)
358 if ((ops & SEL_RD) && e->vir_readfds && FD_ISSET(fd, &e->readfds) &&
359 !FD_ISSET(fd, &e->ready_readfds)) {
360 FD_SET(fd, &e->ready_readfds);
361 e->nreadyfds++;
364 if ((ops & SEL_WR) && e->vir_writefds && FD_ISSET(fd, &e->writefds) &&
365 !FD_ISSET(fd, &e->ready_writefds)) {
366 FD_SET(fd, &e->ready_writefds);
367 e->nreadyfds++;
370 if ((ops & SEL_ERR) && e->vir_errorfds && FD_ISSET(fd, &e->errorfds) &&
371 !FD_ISSET(fd, &e->ready_errorfds)) {
372 FD_SET(fd, &e->ready_errorfds);
373 e->nreadyfds++;
378 /*===========================================================================*
379 * copy_fdsets *
380 *===========================================================================*/
381 PRIVATE int copy_fdsets(struct selectentry *se, int nfds, int direction)
383 int fd_setsize, r;
384 endpoint_t src_e, dst_e;
385 fd_set *src_fds, *dst_fds;
387 if(nfds < 0 || nfds > OPEN_MAX)
388 panic("select copy_fdsets: nfds wrong: %d", nfds);
390 /* Only copy back as many bits as the user expects. */
391 fd_setsize = _FDSETWORDS(nfds) * _FDSETBITSPERWORD/8;
393 /* Set source and destination endpoints */
394 src_e = (direction == FROM_PROC) ? se->req_endpt : SELF;
395 dst_e = (direction == FROM_PROC) ? SELF : se->req_endpt;
397 /* read set */
398 src_fds = (direction == FROM_PROC) ? se->vir_readfds : &se->ready_readfds;
399 dst_fds = (direction == FROM_PROC) ? &se->readfds : se->vir_readfds;
400 if (se->vir_readfds) {
401 r = sys_vircopy(src_e, D, (vir_bytes) src_fds, dst_e, D,
402 (vir_bytes) dst_fds, fd_setsize);
403 if (r != OK) return(r);
406 /* write set */
407 src_fds = (direction == FROM_PROC) ? se->vir_writefds : &se->ready_writefds;
408 dst_fds = (direction == FROM_PROC) ? &se->writefds : se->vir_writefds;
409 if (se->vir_writefds) {
410 r = sys_vircopy(src_e, D, (vir_bytes) src_fds, dst_e, D,
411 (vir_bytes) dst_fds, fd_setsize);
412 if (r != OK) return(r);
415 /* error set */
416 src_fds = (direction == FROM_PROC) ? se->vir_errorfds : &se->ready_errorfds;
417 dst_fds = (direction == FROM_PROC) ? &se->errorfds : se->vir_errorfds;
418 if (se->vir_errorfds) {
419 r = sys_vircopy(src_e, D, (vir_bytes) src_fds, dst_e, D,
420 (vir_bytes) dst_fds, fd_setsize);
421 if (r != OK) return(r);
424 return(OK);
428 /*===========================================================================*
429 * select_cancel_all *
430 *===========================================================================*/
431 PRIVATE void select_cancel_all(struct selectentry *e)
433 int fd;
435 for(fd = 0; fd < e->nfds; fd++) {
436 struct filp *fp;
437 fp = e->filps[fd];
438 if (!fp) {
439 #if DEBUG_SELECT
440 printf("[ fd %d/%d NULL ] ", fd, e->nfds);
441 #endif
442 continue;
444 if (fp->filp_selectors < 1) {
445 #if DEBUG_SELECT
446 printf("select: %d selectors?!\n", fp->filp_selectors);
447 #endif
448 continue;
450 fp->filp_selectors--;
451 e->filps[fd] = NULL;
452 select_reevaluate(fp);
455 if (e->expiry > 0) {
456 #if DEBUG_SELECT
457 printf("cancelling timer %d\n", e - selecttab);
458 #endif
459 fs_cancel_timer(&e->timer);
460 e->expiry = 0;
463 return;
467 /*===========================================================================*
468 * select_wakeup *
469 *===========================================================================*/
470 PRIVATE void select_wakeup(struct selectentry *e, int r)
472 revive(e->req_endpt, r);
476 /*===========================================================================*
477 * select_reevaluate *
478 *===========================================================================*/
479 PRIVATE int select_reevaluate(struct filp *fp)
481 int s, remain_ops = 0, fd;
483 if (!fp) {
484 printf("fs: select: reevalute NULL fp\n");
485 return 0;
488 for(s = 0; s < MAXSELECTS; s++) {
489 if (selecttab[s].requestor != NULL) continue;
491 for(fd = 0; fd < selecttab[s].nfds; fd++)
492 if (fp == selecttab[s].filps[fd]) {
493 remain_ops |= tab2ops(fd, &selecttab[s]);
497 /* If there are any select()s open that want any operations on
498 * this fd that haven't been satisfied by this callback, then we're
499 * still in the market for it.
501 fp->filp_select_ops = remain_ops;
502 #if DEBUG_SELECT
503 printf("remaining operations on fp are %d\n", fp->filp_select_ops);
504 #endif
506 return remain_ops;
510 /*===========================================================================*
511 * select_return *
512 *===========================================================================*/
513 PRIVATE void select_return(struct selectentry *se, int r)
515 select_cancel_all(se);
516 copy_fdsets(se, se->nfds, TO_PROC); /* FIXME, return error status */
517 select_wakeup(se, r ? r : se->nreadyfds);
518 se->requestor = NULL;
522 /*===========================================================================*
523 * select_callback *
524 *===========================================================================*/
525 PUBLIC int select_callback(struct filp *fp, int ops)
527 int s, fd;
529 /* We are being notified that file pointer fp is available for
530 * operations 'ops'. We must re-register the select for
531 * operations that we are still interested in, if any.
534 for(s = 0; s < MAXSELECTS; s++) {
535 int wakehim = 0;
536 if (selecttab[s].requestor == NULL) continue;
538 for(fd = 0; fd < selecttab[s].nfds; fd++) {
539 if (!selecttab[s].filps[fd])
540 continue;
541 if (selecttab[s].filps[fd] == fp) {
542 int this_want_ops;
543 this_want_ops = tab2ops(fd, &selecttab[s]);
544 if (this_want_ops & ops) {
545 /* this select() has been satisfied. */
546 ops2tab(ops, fd, &selecttab[s]);
547 wakehim = 1;
551 if (wakehim)
552 select_return(&selecttab[s], 0);
555 return 0;
559 /*===========================================================================*
560 * select_notified *
561 *===========================================================================*/
562 PUBLIC int select_notified(int major, int minor, int selected_ops)
564 int s, f, t;
566 #if DEBUG_SELECT
567 printf("select callback: %d, %d: %d\n", major, minor, selected_ops);
568 #endif
570 for(t = 0; t < SEL_FDS; t++)
571 if (!fdtypes[t].select_match && fdtypes[t].select_major == major)
572 break;
574 if (t >= SEL_FDS) {
575 #if DEBUG_SELECT
576 printf("select callback: no fdtype found for device %d\n", major);
577 #endif
578 return OK;
581 /* We have a select callback from major device no.
582 * d, which corresponds to our select type t.
585 for(s = 0; s < MAXSELECTS; s++) {
586 int s_minor, ops;
587 if (selecttab[s].requestor == NULL) continue;
588 for(f = 0; f < selecttab[s].nfds; f++) {
589 if (!selecttab[s].filps[f] ||
590 !select_major_match(major, selecttab[s].filps[f]))
591 continue;
592 ops = tab2ops(f, &selecttab[s]);
593 s_minor =
594 (selecttab[s].filps[f]->filp_vno->v_sdev >> MINOR)
595 & BYTE;
596 if ((s_minor == minor) &&
597 (selected_ops & ops)) {
598 select_callback(selecttab[s].filps[f], (selected_ops & ops));
603 return OK;
607 /*===========================================================================*
608 * init_select *
609 *===========================================================================*/
610 PUBLIC void init_select(void)
612 int s;
614 for(s = 0; s < MAXSELECTS; s++)
615 fs_init_timer(&selecttab[s].timer);
619 /*===========================================================================*
620 * select_forget *
621 *===========================================================================*/
622 PUBLIC void select_forget(int proc_e)
624 /* something has happened (e.g. signal delivered that interrupts
625 * select()). totally forget about the select().
627 int s;
629 for(s = 0; s < MAXSELECTS; s++) {
630 if (selecttab[s].requestor != NULL &&
631 selecttab[s].req_endpt == proc_e) {
632 break;
636 if (s >= MAXSELECTS) {
637 #if DEBUG_SELECT
638 printf("select: cancelled select() not found");
639 #endif
640 return;
643 select_cancel_all(&selecttab[s]);
644 selecttab[s].requestor = NULL;
646 return;
650 /*===========================================================================*
651 * select_timeout_check *
652 *===========================================================================*/
653 PUBLIC void select_timeout_check(timer_t *timer)
655 int s;
656 struct selectentry *se;
658 s = tmr_arg(timer)->ta_int;
659 if (s < 0 || s >= MAXSELECTS) {
660 #if DEBUG_SELECT
661 printf("select: bogus slot arg to watchdog %d\n", s);
662 #endif
663 return;
665 se = &selecttab[s]; /* Point to select table entry */
667 if (se->requestor == NULL) {
668 #if DEBUG_SELECT
669 printf("select: no requestor in watchdog\n");
670 #endif
671 return;
674 if (se->expiry <= 0) {
675 #if DEBUG_SELECT
676 printf("select: strange expiry value in watchdog\n", s);
677 #endif
678 return;
681 se->expiry = 0;
682 select_return(se, 0);
687 /*===========================================================================*
688 * select_unsuspend_by_endpt *
689 *===========================================================================*/
690 PUBLIC void select_unsuspend_by_endpt(endpoint_t proc_e)
692 int fd, s, maj;
694 for(s = 0; s < MAXSELECTS; s++) {
695 if (selecttab[s].requestor == NULL) continue;
697 for(fd = 0; fd < selecttab[s].nfds; fd++) {
698 if (selecttab[s].filps[fd] == NIL_FILP ||
699 selecttab[s].filps[fd]->filp_vno == NIL_VNODE) {
700 continue;
703 maj = (selecttab[s].filps[fd]->filp_vno->v_sdev >> MAJOR)&BYTE;
704 if (dmap_driver_match(proc_e, maj))
705 select_return(&selecttab[s], EAGAIN);
712 /*===========================================================================*
713 * select_reply1 *
714 *===========================================================================*/
715 PUBLIC void select_reply1()
717 int i, s, minor, status;
718 endpoint_t driver_e;
719 dev_t dev;
720 struct filp *fp;
721 struct dmap *dp;
722 struct vnode *vp;
724 driver_e= m_in.m_source;
725 minor= m_in.DEV_MINOR;
726 status= m_in.DEV_SEL_OPS;
728 /* Locate dmap entry */
729 for (i= 0, dp= dmap; i<NR_DEVICES; i++, dp++)
731 if (dp->dmap_driver == driver_e)
732 break;
734 if (i >= NR_DEVICES)
736 printf("select_reply1: proc %d is not a recoqnized driver\n",
737 driver_e);
738 return;
740 dev= (i << MAJOR) | (minor & BYTE);
742 fp= dp->dmap_sel_filp;
743 if (!fp)
745 printf("select_reply1: strange, no dmap_sel_filp\n");
746 return;
749 if (!(fp->filp_select_flags & FSF_BUSY))
750 panic("select_reply1: strange; not FSF_BUSY");
752 vp= fp->filp_vno;
753 if (!vp)
754 panic("select_reply1: FSF_BUSY but no vp");
756 if ((vp->v_mode & I_TYPE) != I_CHAR_SPECIAL) {
757 panic("select_reply1: FSF_BUSY but not char special");
760 if (vp->v_sdev != dev)
762 printf("select_reply1: strange, reply from wrong dev\n");
763 return;
766 dp->dmap_sel_filp= NULL;
767 fp->filp_select_flags &= ~FSF_BUSY;
768 if (!(fp->filp_select_flags & (FSF_UPDATE|FSF_BLOCK)))
769 fp->filp_select_ops= 0;
770 if (status != 0)
772 if (status > 0)
774 /* Clear the replied bits from the request mask unless
775 * FSF_UPDATE is set.
777 if (!(fp->filp_select_flags & FSF_UPDATE))
778 fp->filp_select_ops &= ~status;
780 filp_status(fp, status);
782 if (fp->filp_count > 1)
783 fp->filp_count--;
784 else
786 if (fp->filp_count != 1) {
787 panic("select_reply1: bad filp_count: %d", fp->filp_count);
789 close_filp(fp);
791 sel_restart_dev();
795 /*===========================================================================*
796 * select_reply2 *
797 *===========================================================================*/
798 PUBLIC void select_reply2()
800 int i, s, minor, status;
801 endpoint_t driver_e;
802 dev_t dev;
803 struct filp *fp;
804 struct dmap *dp;
805 struct vnode *vp;
807 driver_e= m_in.m_source;
808 minor= m_in.DEV_MINOR;
809 status= m_in.DEV_SEL_OPS;
811 /* Locate dmap entry */
812 for (i= 0, dp= dmap; i<NR_DEVICES; i++, dp++)
814 if (dp->dmap_driver == driver_e)
815 break;
817 if (i >= NR_DEVICES)
819 printf("select_reply2: proc %d is not a recognized driver\n",
820 driver_e);
821 return;
823 dev= (i << MAJOR) | (minor & BYTE);
825 /* Find filedescriptors for this device */
826 for (s= 0; s<MAXSELECTS; s++)
828 if (selecttab[s].requestor == NULL)
829 continue; /* empty slot */
831 for (i= 0; i<OPEN_MAX; i++)
833 fp= selecttab[s].filps[i];
834 if (!fp)
835 continue;
836 vp= fp->filp_vno;
837 if (!vp)
838 continue;
839 if ((vp->v_mode & I_TYPE) != I_CHAR_SPECIAL)
840 continue;
842 if (vp->v_sdev != dev)
843 continue;
845 if (status < 0)
847 printf("select_reply2: should handle error\n");
849 else
851 /* Clear the replied bits from the request
852 * mask unless FSF_UPDATE is set.
854 if (!(fp->filp_select_flags & FSF_UPDATE))
855 fp->filp_select_ops &= ~status;
856 ops2tab(status, i, &selecttab[s]);
859 if (selecttab[s].nreadyfds > 0)
860 restart_proc(s);
865 PRIVATE void sel_restart_dev()
867 int i, s;
868 struct filp *fp;
869 struct vnode *vp;
870 struct dmap *dp;
872 /* Locate filps that can be restarted */
873 for (s= 0; s<MAXSELECTS; s++)
875 if (selecttab[s].requestor == NULL)
876 continue; /* empty slot */
877 if (!selecttab[s].deferred)
878 continue; /* process is not waiting for an
879 * initial reply.
881 for (i= 0; i<OPEN_MAX; i++)
883 fp= selecttab[s].filps[i];
884 if (!fp)
885 continue;
886 if (fp->filp_select_flags & FSF_BUSY)
887 continue;
888 if (!(fp->filp_select_flags & FSF_UPDATE))
889 continue;
891 vp= fp->filp_vno;
892 if (!vp) {
893 panic("sel_restart_dev: FSF_UPDATE but no vp");
895 if ((vp->v_mode & I_TYPE) != I_CHAR_SPECIAL) {
896 panic("sel_restart_dev: FSF_UPDATE but not char special");
899 dp = &dmap[((vp->v_sdev) >> MAJOR) & BYTE];
900 if (dp->dmap_sel_filp)
901 continue;
903 printf(
904 "sel_restart_dev: should consider fd %d in slot %d\n",
905 i, s);
911 PRIVATE void filp_status(fp, status)
912 struct filp *fp;
913 int status;
915 int i, s;
917 /* Locate processes that need to know about this result */
918 for (s= 0; s<MAXSELECTS; s++)
920 if (selecttab[s].requestor == NULL) continue; /* empty slot */
922 for (i= 0; i<OPEN_MAX; i++)
924 if (selecttab[s].filps[i] != fp)
925 continue;
927 if (status < 0)
929 printf("filp_status: should handle error\n");
931 else
932 ops2tab(status, i, &selecttab[s]);
934 restart_proc(s);
940 PRIVATE void restart_proc(slot)
941 int slot;
943 int fd;
944 struct selectentry *se;
945 struct filp *fp;
947 se= &selecttab[slot];
948 if (se->deferred)
950 for (fd= se->deferred_fd; fd < OPEN_MAX; fd++)
952 fp= se->filps[fd];
953 if (!fp)
954 continue;
955 if (fp->filp_select_flags & (FSF_UPDATE|FSF_BUSY))
956 break;
958 if (fd < OPEN_MAX)
960 se->deferred_fd= fd;
961 return;
963 se->deferred= FALSE;
965 if (se->nreadyfds > 0 || !se->block) {
966 copy_fdsets(se, se->nfds, TO_PROC); /* FIXME, return error */
967 select_wakeup(se, se->nreadyfds);
968 se->requestor = NULL;
972 /*===========================================================================*
973 * wipe_select *
974 *===========================================================================*/
975 PRIVATE void wipe_select(struct selectentry *se)
977 se->deferred = FALSE;
978 se->nfds = 0;
979 se->nreadyfds = 0;
980 /* memset(se->filps, 0, OPEN_MAX * sizeof(struct filp *)); */
981 memset(se->filps, 0, sizeof(se->filps));
983 FD_ZERO(&se->readfds);
984 FD_ZERO(&se->writefds);
985 FD_ZERO(&se->errorfds);
986 FD_ZERO(&se->ready_readfds);
987 FD_ZERO(&se->ready_writefds);
988 FD_ZERO(&se->ready_errorfds);