725fcc25ffb51bb458779b5b8e6e1f41d798b155
[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.h>
20 #include <hurd/fd.h>
21 #include <stdlib.h>
22
23 /* Allocate a new file descriptor and return it, locked.  The new
24    descriptor number will be no less than FIRST_FD.  If the table is full,
25    set errno to EMFILE and return NULL.  If FIRST_FD is negative or bigger
26    than the size of the table, set errno to EINVAL and return NULL.  */
27
28 struct hurd_fd *
29 _hurd_alloc_fd (int *fd, const int first_fd)
30 {
31   int i;
32
33   if (first_fd < 0)
34     {
35       errno = EINVAL;
36       return NULL;
37     }
38
39   __mutex_lock (&_hurd_dtable_lock);
40
41   for (i = first_fd; i < _hurd_dtable.size; ++i)
42     {
43       struct hurd_fd *d = &_hurd_dtable.d[i];
44       if (d == NULL)
45         {
46           /* Allocate a new descriptor structure for this slot,
47              initializing its port cells to nil.  The test below will catch
48              and return this descriptor cell after locking it.  */
49           d = _hurd_new_fd (MACH_PORT_NULL, MACH_PORT_NULL);
50           if (d == NULL)
51             {
52               __mutex_unlock (&hurd_dtable_lock);
53               return NULL;
54             }
55         }
56
57       __spin_lock (&d->port.lock);
58       if (d->port.port == MACH_PORT_NULL)
59         {
60           __mutex_unlock (&hurd_dtable_lock);
61           if (fd != NULL)
62             *fd = i;
63           return d;
64         }
65       else
66         __spin_unlock (&d->port.lock);
67     }
68
69   errno = first_fd < _hurd_dtable.size ? EMFILE : EINVAL;
70
71   __mutex_unlock (&hurd_dtable_lock);
72
73   return NULL;
74 }