2.9
[glibc/nacl-glibc.git] / sysdeps / unix / sysv / linux / xmknodat.c
blobef27b686cc29bc3b6ce6afc0f49ec186d7740235
1 /* Copyright (C) 2005, 2006 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
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 Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, write to the Free
16 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
17 02111-1307 USA. */
19 #include <errno.h>
20 #include <fcntl.h>
21 #include <stdio.h>
22 #include <string.h>
23 #include <sys/types.h>
24 #include <sys/stat.h>
25 #include <sys/sysmacros.h>
27 #include <sysdep.h>
28 #include <kernel-features.h>
29 #include <sys/syscall.h>
30 #include <bp-checks.h>
33 /* Create a device file named PATH relative to FD, with permission and
34 special bits MODE and device number DEV (which can be constructed
35 from major and minor device numbers with the `makedev' macro above). */
36 int
37 __xmknodat (int vers, int fd, const char *file, mode_t mode, dev_t *dev)
39 if (vers != _MKNOD_VER)
41 __set_errno (EINVAL);
42 return -1;
45 /* We must convert the value to dev_t type used by the kernel. */
46 unsigned long long int k_dev = (*dev) & ((1ULL << 32) - 1);
47 if (k_dev != *dev)
49 __set_errno (EINVAL);
50 return -1;
53 #ifdef __NR_mknodat
54 # ifndef __ASSUME_ATFCTS
55 if (__have_atfcts >= 0)
56 # endif
58 int res = INLINE_SYSCALL (mknodat, 4, fd, file, mode,
59 (unsigned int) k_dev);
60 # ifndef __ASSUME_ATFCTS
61 if (res == -1 && errno == ENOSYS)
62 __have_atfcts = -1;
63 else
64 # endif
65 return res;
67 #endif
69 #ifndef __ASSUME_ATFCTS
70 char *buf = NULL;
72 if (fd != AT_FDCWD && file[0] != '/')
74 size_t filelen = strlen (file);
75 static const char procfd[] = "/proc/self/fd/%d/%s";
76 /* Buffer for the path name we are going to use. It consists of
77 - the string /proc/self/fd/
78 - the file descriptor number
79 - the file name provided.
80 The final NUL is included in the sizeof. A bit of overhead
81 due to the format elements compensates for possible negative
82 numbers. */
83 size_t buflen = sizeof (procfd) + sizeof (int) * 3 + filelen;
84 buf = alloca (buflen);
86 __snprintf (buf, buflen, procfd, fd, file);
87 file = buf;
90 return INLINE_SYSCALL (mknod, 3, CHECK_STRING (file), mode,
91 (unsigned int) k_dev);
92 #endif
95 libc_hidden_def (__xmknodat)