6e9767bfbde03058200e47d6cd0a5bba1de6fee6
[kopensolaris-gnu/glibc.git] / hurd / alloc-fd.c
1 /* Copyright (C) 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 <hurd/fd.h>
20 #include <stdlib.h>
21
22 /* Allocate a new file descriptor and return it, locked.
23    The new descriptor number will be no less than FIRST_FD.
24    If the table is full, set errno and return NULL.  */
25 struct hurd_fd *
26 _hurd_alloc_fd (int *fd, const int first_fd)
27 {
28   int i;
29
30   __mutex_lock (&_hurd_dtable_lock);
31
32   for (i = first_fd; i < _hurd_dtable.size; ++i)
33     {
34       struct hurd_fd *d = &_hurd_dtable.d[i];
35       if (d == NULL)
36         {
37           /* Allocate a new descriptor structure for this slot.  */
38           d = malloc (sizeof (struct hurd_fd));
39           if (d == NULL)
40             {
41               __mutex_unlock (&hurd_dtable_lock);
42               return NULL;
43             }
44
45           /* Initialize the port cells.  The test below will catch and
46              return this descriptor cell after locking it.  */
47           _hurd_port_init (&d->port, MACH_PORT_NULL);
48           _hurd_port_init (&d->ctty, MACH_PORT_NULL);
49         }
50
51       __spin_lock (&d->port.lock);
52       if (d->port.port == MACH_PORT_NULL)
53         {
54           __mutex_unlock (&hurd_dtable_lock);
55           if (fd != NULL)
56             *fd = i;
57           return d;
58         }
59       else
60         __spin_unlock (&d->port.lock);
61     }
62
63   __mutex_unlock (&hurd_dtable_lock);
64
65   errno = EMFILE;
66   return NULL;
67 }