Fix large-file macro usage in bits/statvfs.h
[kopensolaris-gnu/glibc.git] / io / tst-symlinkat.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-symlinkat.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   close (fd);
107
108   if (symlinkat ("some-file", dir_fd, "another-file") != 0)
109     {
110       if (errno == ENOSYS)
111         {
112           puts ("symlinkat function not supported");
113           return 0;
114         }
115       puts ("symlinkat failed");
116       return 1;
117     }
118
119   struct stat64 st2;
120   if (fstatat64 (dir_fd, "some-file", &st2, 0) != 0)
121     {
122       puts ("fstatat64 failed");
123       return 1;
124     }
125   if (st1.st_dev != st2.st_dev || st1.st_ino != st2.st_ino)
126     {
127       puts ("file changed after symlinkat");
128       return 1;
129     }
130
131   if (fstatat64 (dir_fd, "another-file", &st2, AT_SYMLINK_NOFOLLOW) != 0)
132     {
133       puts ("2nd fstatat64 failed");
134       return 1;
135     }
136   if (!S_ISLNK (st2.st_mode))
137     {
138       puts ("2nd fstatat64 does not show file is a symlink");
139       return 1;
140     }
141
142   if (fstatat64 (dir_fd, "another-file", &st2, 0) != 0)
143     {
144       puts ("3rd fstatat64 failed");
145       return 1;
146     }
147   if (st1.st_dev != st2.st_dev
148       || st1.st_ino != st2.st_ino
149       || st1.st_size != st2.st_size)
150     {
151       puts ("stat results do not match");
152       return 1;
153     }
154
155   if (unlinkat (dir_fd, "another-file", 0) != 0)
156     {
157       puts ("unlinkat failed");
158       return 1;
159     }
160   if (unlinkat (dir_fd, "some-file", 0) != 0)
161     {
162       puts ("2nd unlinkat failed");
163       return 1;
164     }
165
166   close (dir_fd);
167
168   return 0;
169 }