Fix large-file macro usage in bits/statvfs.h
[kopensolaris-gnu/glibc.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 == ENOSYS)
113     {
114       puts ("renameat function not supported");
115       return 0;
116     }
117   else if (errno != ENOTDIR)
118     {
119       puts ("error for renameat with normal file descriptor not ENOTDIR");
120       return 1;
121     }
122
123   if (renameat (dir_fd, "some-file", fd, "another-file") == 0)
124     {
125       puts ("2nd renameat with normal file descriptor succeeded");
126       return 1;
127     }
128   if (errno != ENOTDIR)
129     {
130       puts ("error for 2nd renameat with normal file descriptor not ENOTDIR");
131       return 1;
132     }
133
134   close (fd);
135
136   if (renameat (dir_fd, "some-file", dir_fd, "another-file") != 0)
137     {
138       puts ("renameat failed");
139       return 1;
140     }
141
142   struct stat64 st2;
143   if (fstatat64 (dir_fd, "some-file", &st2, 0) == 0)
144     {
145       puts ("fstatat64 succeeded");
146       return 1;
147     }
148   if (errno != ENOENT)
149     {
150       puts ("fstatat64 did not fail with ENOENT");
151       return 1;
152     }
153
154   if (fstatat64 (dir_fd, "another-file", &st2, 0) != 0)
155     {
156       puts ("2nd fstatat64 failed");
157       return 1;
158     }
159
160   if (st1.st_dev != st2.st_dev
161       || st1.st_ino != st2.st_ino
162       || st1.st_size != st2.st_size)
163     {
164       puts ("stat results do not match");
165       return 1;
166     }
167
168   /* Create a file descriptor which is closed again right away.  */
169   int dir_fd2 = dup (dir_fd);
170   if (dir_fd2 == -1)
171     {
172       puts ("dup failed");
173       return 1;
174     }
175   close (dir_fd2);
176
177   if (renameat (dir_fd2, "another-file", dir_fd, "some-file") == 0)
178     {
179       puts ("renameat with closed file descriptor succeeded");
180       return 1;
181     }
182   if (errno != EBADF)
183     {
184       puts ("error for renameat with closed file descriptor not EBADF");
185       return 1;
186     }
187
188   if (renameat (dir_fd, "another-file", dir_fd2, "some-file") == 0)
189     {
190       puts ("2nd renameat with closed file descriptor succeeded");
191       return 1;
192     }
193   if (errno != EBADF)
194     {
195       puts ("error for 2nd renameat with closed file descriptor not EBADF");
196       return 1;
197     }
198
199   if (unlinkat (dir_fd, "another-file", 0) != 0)
200     {
201       puts ("unlinkat failed");
202       return 1;
203     }
204
205   if (renameat (-1, "another-file", dir_fd, "some-file") == 0)
206     {
207       puts ("renameat with invalid file descriptor succeeded");
208       return 1;
209     }
210   if (errno != EBADF)
211     {
212       puts ("error for renameat with invalid file descriptor not EBADF");
213       return 1;
214     }
215
216   if (renameat (dir_fd, "another-file", -1, "some-file") == 0)
217     {
218       puts ("2nd renameat with invalid file descriptor succeeded");
219       return 1;
220     }
221   if (errno != EBADF)
222     {
223       puts ("error for 2nd renameat with invalid file descriptor not EBADF");
224       return 1;
225     }
226
227   close (dir_fd);
228
229   return 0;
230 }