chiark / gitweb /
Apply https://sourceware.org/git/?p=glibc.git;a=commit;h=d5dd6189d506068ed11c8bfa1e1e...
[eglibc.git] / io / tst-renameat.c
1 #include <dirent.h>
2 #include <fcntl.h>
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <string.h>
6 #include <unistd.h>
7
8
9 static void prepare (void);
10 #define PREPARE(argc, argv) prepare ()
11
12 static int do_test (void);
13 #define TEST_FUNCTION do_test ()
14
15 #include "../test-skeleton.c"
16
17 static int dir_fd;
18
19 static void
20 prepare (void)
21 {
22   size_t test_dir_len = strlen (test_dir);
23   static const char dir_name[] = "/tst-renameat.XXXXXX";
24
25   size_t dirbuflen = test_dir_len + sizeof (dir_name);
26   char *dirbuf = malloc (dirbuflen);
27   if (dirbuf == NULL)
28     {
29       puts ("out of memory");
30       exit (1);
31     }
32
33   snprintf (dirbuf, dirbuflen, "%s%s", test_dir, dir_name);
34   if (mkdtemp (dirbuf) == NULL)
35     {
36       puts ("cannot create temporary directory");
37       exit (1);
38     }
39
40   add_temp_file (dirbuf);
41
42   dir_fd = open (dirbuf, O_RDONLY | O_DIRECTORY);
43   if (dir_fd == -1)
44     {
45       puts ("cannot open directory");
46       exit (1);
47     }
48 }
49
50
51 static int
52 do_test (void)
53 {
54   /* fdopendir takes over the descriptor, make a copy.  */
55   int dupfd = dup (dir_fd);
56   if (dupfd == -1)
57     {
58       puts ("dup failed");
59       return 1;
60     }
61   if (lseek (dupfd, 0, SEEK_SET) != 0)
62     {
63       puts ("1st lseek failed");
64       return 1;
65     }
66
67   /* The directory should be empty safe the . and .. files.  */
68   DIR *dir = fdopendir (dupfd);
69   if (dir == NULL)
70     {
71       puts ("fdopendir failed");
72       return 1;
73     }
74   struct dirent64 *d;
75   while ((d = readdir64 (dir)) != NULL)
76     if (strcmp (d->d_name, ".") != 0 && strcmp (d->d_name, "..") != 0)
77       {
78         printf ("temp directory contains file \"%s\"\n", d->d_name);
79         return 1;
80       }
81   closedir (dir);
82
83   /* Try to create a file.  */
84   int fd = openat (dir_fd, "some-file", O_CREAT|O_RDWR|O_EXCL, 0666);
85   if (fd == -1)
86     {
87       if (errno == ENOSYS)
88         {
89           puts ("*at functions not supported");
90           return 0;
91         }
92
93       puts ("file creation failed");
94       return 1;
95     }
96   write (fd, "hello", 5);
97   puts ("file created");
98
99   struct stat64 st1;
100   if (fstat64 (fd, &st1) != 0)
101     {
102       puts ("fstat64 failed");
103       return 1;
104     }
105
106   /* Using a descriptor for a normal file must fail.  */
107   if (renameat (fd, "some-file", dir_fd, "another-file") == 0)
108     {
109       puts ("renameat with normal file descriptor succeeded");
110       return 1;
111     }
112   if (errno != ENOTDIR)
113     {
114       puts ("error for renameat with normal file descriptor not ENOTDIR");
115       return 1;
116     }
117
118   if (renameat (dir_fd, "some-file", fd, "another-file") == 0)
119     {
120       puts ("2nd renameat with normal file descriptor succeeded");
121       return 1;
122     }
123   if (errno != ENOTDIR)
124     {
125       puts ("error for 2nd renameat with normal file descriptor not ENOTDIR");
126       return 1;
127     }
128
129   close (fd);
130
131   if (renameat (dir_fd, "some-file", dir_fd, "another-file") != 0)
132     {
133       puts ("renameat failed");
134       return 1;
135     }
136
137   struct stat64 st2;
138   if (fstatat64 (dir_fd, "some-file", &st2, 0) == 0)
139     {
140       puts ("fstatat64 succeeded");
141       return 1;
142     }
143   if (errno != ENOENT)
144     {
145       puts ("fstatat64 did not fail with ENOENT");
146       return 1;
147     }
148
149   if (fstatat64 (dir_fd, "another-file", &st2, 0) != 0)
150     {
151       puts ("2nd fstatat64 failed");
152       return 1;
153     }
154
155   if (st1.st_dev != st2.st_dev
156       || st1.st_ino != st2.st_ino
157       || st1.st_size != st2.st_size)
158     {
159       puts ("stat results do not match");
160       return 1;
161     }
162
163   /* Create a file descriptor which is closed again right away.  */
164   int dir_fd2 = dup (dir_fd);
165   if (dir_fd2 == -1)
166     {
167       puts ("dup failed");
168       return 1;
169     }
170   close (dir_fd2);
171
172   if (renameat (dir_fd2, "another-file", dir_fd, "some-file") == 0)
173     {
174       puts ("renameat with closed file descriptor succeeded");
175       return 1;
176     }
177   if (errno != EBADF)
178     {
179       puts ("error for renameat with closed file descriptor not EBADF");
180       return 1;
181     }
182
183   if (renameat (dir_fd, "another-file", dir_fd2, "some-file") == 0)
184     {
185       puts ("2nd renameat with closed file descriptor succeeded");
186       return 1;
187     }
188   if (errno != EBADF)
189     {
190       puts ("error for 2nd renameat with closed file descriptor not EBADF");
191       return 1;
192     }
193
194   if (unlinkat (dir_fd, "another-file", 0) != 0)
195     {
196       puts ("unlinkat failed");
197       return 1;
198     }
199
200   if (renameat (-1, "another-file", dir_fd, "some-file") == 0)
201     {
202       puts ("renameat with invalid file descriptor succeeded");
203       return 1;
204     }
205   if (errno != EBADF)
206     {
207       puts ("error for renameat with invalid file descriptor not EBADF");
208       return 1;
209     }
210
211   if (renameat (dir_fd, "another-file", -1, "some-file") == 0)
212     {
213       puts ("2nd renameat with invalid file descriptor succeeded");
214       return 1;
215     }
216   if (errno != EBADF)
217     {
218       puts ("error for 2nd renameat with invalid file descriptor not EBADF");
219       return 1;
220     }
221
222   close (dir_fd);
223
224   return 0;
225 }