6f8214d9fa5ce34467ecb9e8486d6a93e0d0ef42
[kopensolaris-gnu/glibc.git] / hurd / fd-read.c
1 /* Copyright (C) 1993, 1994 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Library General Public License as
6 published by the Free Software Foundation; either version 2 of the
7 License, or (at your option) any later version.
8
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12 Library General Public License for more details.
13
14 You should have received a copy of the GNU Library General Public
15 License along with the GNU C Library; see the file COPYING.LIB.  If
16 not, write to the Free Software Foundation, Inc., 675 Mass Ave,
17 Cambridge, MA 02139, USA.  */
18
19 #include <errno.h>
20 #include <unistd.h>
21 #include <hurd.h>
22 #include <hurd/fd.h>
23
24 error_t
25 _hurd_fd_read (struct hurd_fd *fd, void *buf, size_t *nbytes)
26 {
27   error_t err;
28   char *data;
29   mach_msg_type_size_t nread;
30
31   data = buf;
32   err = HURD_FD_PORT_USE
33     (fd,
34      ({
35        do
36          {
37            err = __io_read (port, &data, &nread, -1, *nbytes);
38            if (ctty != MACH_PORT_NULL && err == EBACKGROUND)
39              {
40                /* We are a background job and tried to read from the tty.
41                   We should probably get a SIGTTIN signal.  */
42                struct _hurd_sigstate *ss;
43                if (_hurd_orphaned)
44                  /* Our process group is orphaned.  Don't stop; just fail.  */
45                  err = EIO;
46                else
47                  {
48                    ss = _hurd_self_sigstate ();
49                    if (__sigismember (SIGTTIN, &ss->blocked) ||
50                        ss->actions[SIGTTIN].sa_handler == SIG_IGN)
51                      /* We are blocking or ignoring SIGTTIN.  Just fail.  */
52                      err = EIO;
53                    __mutex_unlock (&ss->lock);
54                  }
55                if (err == EBACKGROUND)
56                  {
57                    /* Send a SIGTTIN signal to our process group.  */
58                    int restart;
59                    err = __USEPORT (CTTYID, _hurd_sig_post (0, SIGTTIN, port));
60                    /* XXX what to do if error here? */
61                    /* At this point we should have just run the handler for
62                       SIGTTIN or resumed after being stopped.  Now this is
63                       still a "system call", so check to see if we should
64                       restart it.  */
65                    __mutex_lock (&ss->lock);
66                    if (!(ss->actions[SIGTTIN].sa_flags & SA_RESTART))
67                      err = EINTR;
68                    __mutex_unlock (&ss->lock);
69                  }
70              }
71          } while (err != EBACKGROUND);
72        err;
73      }));
74
75   if (err)
76     return err;
77
78   if (data != buf)
79     {
80       memcpy (buf, data, nread);
81       __vm_deallocate (__mach_task_self (), (vm_address_t) data, nread);
82     }
83
84   *nbytes = nread;
85   return 0;
86 }