Update copyright notices with scripts/update-copyrights
[glibc.git] / sysdeps / unix / sysv / linux / xmknodat.c
blob62e47e1eed43f94b857458ecd412d81972926bbb
1 /* Copyright (C) 2005-2014 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, see
16 <http://www.gnu.org/licenses/>. */
18 #include <errno.h>
19 #include <fcntl.h>
20 #include <stdio.h>
21 #include <string.h>
22 #include <sys/types.h>
23 #include <sys/stat.h>
24 #include <sys/sysmacros.h>
26 #include <sysdep.h>
27 #include <kernel-features.h>
28 #include <sys/syscall.h>
31 /* Create a device file named PATH relative to FD, with permission and
32 special bits MODE and device number DEV (which can be constructed
33 from major and minor device numbers with the `makedev' macro above). */
34 int
35 __xmknodat (int vers, int fd, const char *file, mode_t mode, dev_t *dev)
37 if (vers != _MKNOD_VER)
39 __set_errno (EINVAL);
40 return -1;
43 /* We must convert the value to dev_t type used by the kernel. */
44 unsigned long long int k_dev = (*dev) & ((1ULL << 32) - 1);
45 if (k_dev != *dev)
47 __set_errno (EINVAL);
48 return -1;
51 #ifdef __NR_mknodat
52 # ifndef __ASSUME_ATFCTS
53 if (__have_atfcts >= 0)
54 # endif
56 int res = INLINE_SYSCALL (mknodat, 4, fd, file, mode,
57 (unsigned int) k_dev);
58 # ifndef __ASSUME_ATFCTS
59 if (res == -1 && errno == ENOSYS)
60 __have_atfcts = -1;
61 else
62 # endif
63 return res;
65 #endif
67 #ifndef __ASSUME_ATFCTS
68 char *buf = NULL;
70 if (fd != AT_FDCWD && file[0] != '/')
72 size_t filelen = strlen (file);
73 if (__builtin_expect (filelen == 0, 0))
75 __set_errno (ENOENT);
76 return -1;
79 static const char procfd[] = "/proc/self/fd/%d/%s";
80 /* Buffer for the path name we are going to use. It consists of
81 - the string /proc/self/fd/
82 - the file descriptor number
83 - the file name provided.
84 The final NUL is included in the sizeof. A bit of overhead
85 due to the format elements compensates for possible negative
86 numbers. */
87 size_t buflen = sizeof (procfd) + sizeof (int) * 3 + filelen;
88 buf = alloca (buflen);
90 __snprintf (buf, buflen, procfd, fd, file);
91 file = buf;
94 return INLINE_SYSCALL (mknod, 3, file, mode, (unsigned int) k_dev);
95 #endif
98 libc_hidden_def (__xmknodat)