[9] | 1 | /* Many of the tests require 1.6.n, n > 16, so we may as well assume that
|
---|
| 2 | * POSIX signals are implemented.
|
---|
| 3 | */
|
---|
| 4 | #define SIGACTION
|
---|
| 5 |
|
---|
| 6 | /* test38: read(), write() Author: Jan-Mark Wams (jms@cs.vu.nl) */
|
---|
| 7 |
|
---|
| 8 | #include <sys/types.h>
|
---|
| 9 | #include <sys/stat.h>
|
---|
| 10 | #include <sys/wait.h>
|
---|
| 11 | #include <stdlib.h>
|
---|
| 12 | #include <unistd.h>
|
---|
| 13 | #include <string.h>
|
---|
| 14 | #include <fcntl.h>
|
---|
| 15 | #include <limits.h>
|
---|
| 16 | #include <errno.h>
|
---|
| 17 | #include <time.h>
|
---|
| 18 | #include <signal.h>
|
---|
| 19 | #include <stdio.h>
|
---|
| 20 |
|
---|
| 21 | #define MAX_ERROR 4
|
---|
| 22 | #define ITERATIONS 3
|
---|
| 23 | #define BUF_SIZE 1024
|
---|
| 24 |
|
---|
| 25 | #define System(cmd) if (system(cmd) != 0) printf("``%s'' failed\n", cmd)
|
---|
| 26 | #define Chdir(dir) if (chdir(dir) != 0) printf("Can't goto %s\n", dir)
|
---|
| 27 | #define Stat(a,b) if (stat(a,b) != 0) printf("Can't stat %s\n", a)
|
---|
| 28 |
|
---|
| 29 | int errct = 0;
|
---|
| 30 | int subtest = 1;
|
---|
| 31 | int superuser;
|
---|
| 32 | int signumber = 0;
|
---|
| 33 |
|
---|
| 34 | _PROTOTYPE(void main, (int argc, char *argv[]));
|
---|
| 35 | _PROTOTYPE(void test38a, (void));
|
---|
| 36 | _PROTOTYPE(void test38b, (void));
|
---|
| 37 | _PROTOTYPE(void test38c, (void));
|
---|
| 38 | _PROTOTYPE(void setsignumber, (int _signumber));
|
---|
| 39 | _PROTOTYPE(void e, (int number));
|
---|
| 40 | _PROTOTYPE(void quit, (void));
|
---|
| 41 |
|
---|
| 42 | void main(argc, argv)
|
---|
| 43 | int argc;
|
---|
| 44 | char *argv[];
|
---|
| 45 | {
|
---|
| 46 | int i, m = 0xFFFF;
|
---|
| 47 |
|
---|
| 48 | sync();
|
---|
| 49 | if (argc == 2) m = atoi(argv[1]);
|
---|
| 50 | printf("Test 38 ");
|
---|
| 51 | fflush(stdout);
|
---|
| 52 | System("rm -rf DIR_38; mkdir DIR_38");
|
---|
| 53 | Chdir("DIR_38");
|
---|
| 54 | superuser = (geteuid() == 0);
|
---|
| 55 | umask(0000);
|
---|
| 56 |
|
---|
| 57 | for (i = 0; i < ITERATIONS; i++) {
|
---|
| 58 | if (m & 0001) test38a();
|
---|
| 59 | if (m & 0002) test38b();
|
---|
| 60 | if (m & 0004) test38c();
|
---|
| 61 | }
|
---|
| 62 | quit();
|
---|
| 63 | }
|
---|
| 64 |
|
---|
| 65 | void test38a()
|
---|
| 66 | { /* Try normal operation. */
|
---|
| 67 | int fd1;
|
---|
| 68 | struct stat st1, st2;
|
---|
| 69 | time_t time1;
|
---|
| 70 | char buf[BUF_SIZE];
|
---|
| 71 | int stat_loc;
|
---|
| 72 | int i, j;
|
---|
| 73 | int tube[2];
|
---|
| 74 |
|
---|
| 75 | subtest = 1;
|
---|
| 76 | System("rm -rf ../DIR_38/*");
|
---|
| 77 |
|
---|
| 78 | /* Let's open bar. */
|
---|
| 79 | if ((fd1 = open("bar", O_RDWR | O_CREAT, 0777)) != 3) e(1);
|
---|
| 80 | Stat("bar", &st1);
|
---|
| 81 |
|
---|
| 82 | /* Writing nothing should not affect the file at all. */
|
---|
| 83 | if (write(fd1, "", 0) != 0) e(2);
|
---|
| 84 | Stat("bar", &st2);
|
---|
| 85 | if (st1.st_uid != st2.st_uid) e(3);
|
---|
| 86 | if (st1.st_gid != st2.st_gid) e(4); /* should be same */
|
---|
| 87 | if (st1.st_mode != st2.st_mode) e(5);
|
---|
| 88 | if (st1.st_size != st2.st_size) e(6);
|
---|
| 89 | if (st1.st_nlink != st2.st_nlink) e(7);
|
---|
| 90 | if (st1.st_mtime != st2.st_mtime) e(8);
|
---|
| 91 | if (st1.st_ctime != st2.st_ctime) e(9);
|
---|
| 92 | if (st1.st_atime != st2.st_atime) e(10);
|
---|
| 93 |
|
---|
| 94 | /* A write should update some status fields. */
|
---|
| 95 | time(&time1);
|
---|
| 96 | while (time1 >= time((time_t *)0))
|
---|
| 97 | ;
|
---|
| 98 | if (write(fd1, "foo", 4) != 4) e(11);
|
---|
| 99 | Stat("bar", &st2);
|
---|
| 100 | if (st1.st_mode != st2.st_mode) e(12);
|
---|
| 101 | if (st1.st_size >= st2.st_size) e(13);
|
---|
| 102 | if ((off_t) 4 != st2.st_size) e(14);
|
---|
| 103 | if (st1.st_nlink != st2.st_nlink) e(15);
|
---|
| 104 | if (st1.st_mtime >= st2.st_mtime) e(16);
|
---|
| 105 | if (st1.st_ctime >= st2.st_ctime) e(17);
|
---|
| 106 | if (st1.st_atime != st2.st_atime) e(18);
|
---|
| 107 |
|
---|
| 108 | /* Lseeks should not change the file status. */
|
---|
| 109 | if (lseek(fd1, (off_t) - 2, SEEK_END) != 2) e(19);
|
---|
| 110 | Stat("bar", &st1);
|
---|
| 111 | if (st1.st_mode != st2.st_mode) e(20);
|
---|
| 112 | if (st1.st_size != st2.st_size) e(21);
|
---|
| 113 | if (st1.st_nlink != st2.st_nlink) e(22);
|
---|
| 114 | if (st1.st_mtime != st2.st_mtime) e(23);
|
---|
| 115 | if (st1.st_ctime != st2.st_ctime) e(24);
|
---|
| 116 | if (st1.st_atime != st2.st_atime) e(25);
|
---|
| 117 |
|
---|
| 118 | /* Writing should start at the current (2) position. */
|
---|
| 119 | if (write(fd1, "foo", 4) != 4) e(26);
|
---|
| 120 | Stat("bar", &st2);
|
---|
| 121 | if (st1.st_mode != st2.st_mode) e(27);
|
---|
| 122 | if (st1.st_size >= st2.st_size) e(28);
|
---|
| 123 | if ((off_t) 6 != st2.st_size) e(29);
|
---|
| 124 | if (st1.st_nlink != st2.st_nlink) e(30);
|
---|
| 125 | if (st1.st_mtime > st2.st_mtime) e(31);
|
---|
| 126 | if (st1.st_ctime > st2.st_ctime) e(32);
|
---|
| 127 | if (st1.st_atime != st2.st_atime) e(33);
|
---|
| 128 |
|
---|
| 129 | /* A read of zero bytes should not affect anything. */
|
---|
| 130 | if (read(fd1, buf, 0) != 0) e(34);
|
---|
| 131 | Stat("bar", &st1);
|
---|
| 132 | if (st1.st_uid != st2.st_uid) e(35);
|
---|
| 133 | if (st1.st_gid != st2.st_gid) e(36); /* should be same */
|
---|
| 134 | if (st1.st_mode != st2.st_mode) e(37);
|
---|
| 135 | if (st1.st_size != st2.st_size) e(38);
|
---|
| 136 | if (st1.st_nlink != st2.st_nlink) e(39);
|
---|
| 137 | if (st1.st_mtime != st2.st_mtime) e(40);
|
---|
| 138 | if (st1.st_ctime != st2.st_ctime) e(41);
|
---|
| 139 | if (st1.st_atime != st2.st_atime) e(42);
|
---|
| 140 |
|
---|
| 141 | /* The file now should contain ``fofoo\0'' Let's check that. */
|
---|
| 142 | if (lseek(fd1, (off_t) 0, SEEK_SET) != 0) e(43);
|
---|
| 143 | if (read(fd1, buf, BUF_SIZE) != 6) e(44);
|
---|
| 144 | if (strcmp(buf, "fofoo") != 0) e(45);
|
---|
| 145 |
|
---|
| 146 | /* Only the Access Time should be updated. */
|
---|
| 147 | Stat("bar", &st2);
|
---|
| 148 | if (st1.st_mtime != st2.st_mtime) e(46);
|
---|
| 149 | if (st1.st_ctime != st2.st_ctime) e(47);
|
---|
| 150 | if (st1.st_atime >= st2.st_atime) e(48);
|
---|
| 151 |
|
---|
| 152 | /* A read of zero bytes should do nothing even at the end of the file. */
|
---|
| 153 | time(&time1);
|
---|
| 154 | while (time1 >= time((time_t *)0))
|
---|
| 155 | ;
|
---|
| 156 | if (read(fd1, buf, 0) != 0) e(49);
|
---|
| 157 | Stat("bar", &st1);
|
---|
| 158 | if (st1.st_size != st2.st_size) e(50);
|
---|
| 159 | if (st1.st_mtime != st2.st_mtime) e(51);
|
---|
| 160 | if (st1.st_ctime != st2.st_ctime) e(52);
|
---|
| 161 | if (st1.st_atime != st2.st_atime) e(53);
|
---|
| 162 |
|
---|
| 163 | /* Reading should be done from the current offset. */
|
---|
| 164 | if (read(fd1, buf, BUF_SIZE) != 0) e(54);
|
---|
| 165 | if (lseek(fd1, (off_t) 2, SEEK_SET) != 2) e(55);
|
---|
| 166 | if (read(fd1, buf, BUF_SIZE) != 4) e(56);
|
---|
| 167 | if (strcmp(buf, "foo") != 0) e(57);
|
---|
| 168 |
|
---|
| 169 | /* Reading should effect the current file position. */
|
---|
| 170 | if (lseek(fd1, (off_t) 2, SEEK_SET) != 2) e(58);
|
---|
| 171 | if (read(fd1, buf, 1) != 1) e(59);
|
---|
| 172 | if (*buf != 'f') e(60);
|
---|
| 173 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 3) e(61);
|
---|
| 174 | if (read(fd1, buf, 1) != 1) e(62);
|
---|
| 175 | if (*buf != 'o') e(63);
|
---|
| 176 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 4) e(64);
|
---|
| 177 | if (read(fd1, buf, 1) != 1) e(65);
|
---|
| 178 | if (*buf != 'o') e(66);
|
---|
| 179 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 5) e(67);
|
---|
| 180 | if (read(fd1, buf, 1) != 1) e(68);
|
---|
| 181 | if (*buf != '\0') e(69);
|
---|
| 182 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(70);
|
---|
| 183 |
|
---|
| 184 | /* Read's at EOF should return 0. */
|
---|
| 185 | if (read(fd1, buf, BUF_SIZE) != 0) e(71);
|
---|
| 186 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(72);
|
---|
| 187 | if (read(fd1, buf, BUF_SIZE) != 0) e(73);
|
---|
| 188 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(74);
|
---|
| 189 | if (read(fd1, buf, BUF_SIZE) != 0) e(75);
|
---|
| 190 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(76);
|
---|
| 191 | if (read(fd1, buf, BUF_SIZE) != 0) e(77);
|
---|
| 192 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(78);
|
---|
| 193 | if (read(fd1, buf, BUF_SIZE) != 0) e(79);
|
---|
| 194 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 6) e(80);
|
---|
| 195 |
|
---|
| 196 | /* Writing should not always change the file size. */
|
---|
| 197 | if (lseek(fd1, (off_t) 2, SEEK_SET) != 2) e(81);
|
---|
| 198 | if (write(fd1, "ba", 2) != 2) e(82);
|
---|
| 199 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 4) e(83);
|
---|
| 200 | Stat("bar", &st1);
|
---|
| 201 | if (st1.st_size != 6) e(84);
|
---|
| 202 |
|
---|
| 203 | /* Kill the \0 at the end. */
|
---|
| 204 | if (lseek(fd1, (off_t) 5, SEEK_SET) != 5) e(85);
|
---|
| 205 | if (write(fd1, "x", 1) != 1) e(86);
|
---|
| 206 |
|
---|
| 207 | /* And close the bar. */
|
---|
| 208 | if (close(fd1) != 0) e(87);
|
---|
| 209 |
|
---|
| 210 | /* Try some stuff with O_APPEND. Bar contains ``fobaox'' */
|
---|
| 211 | if ((fd1 = open("bar", O_RDWR | O_APPEND)) != 3) e(88);
|
---|
| 212 |
|
---|
| 213 | /* No matter what the file position is. Writes should append. */
|
---|
| 214 | if (lseek(fd1, (off_t) 2, SEEK_SET) != 2) e(89);
|
---|
| 215 | if (write(fd1, "y", 1) != 1) e(90);
|
---|
| 216 | Stat("bar", &st1);
|
---|
| 217 | if (st1.st_size != (off_t) 7) e(91);
|
---|
| 218 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 7) e(92);
|
---|
| 219 | if (lseek(fd1, (off_t) 2, SEEK_SET) != 2) e(93);
|
---|
| 220 | if (write(fd1, "z", 2) != 2) e(94);
|
---|
| 221 |
|
---|
| 222 | /* The file should contain ``fobaoxyz\0'' == 9 chars long. */
|
---|
| 223 | Stat("bar", &st1);
|
---|
| 224 | if (st1.st_size != (off_t) 9) e(95);
|
---|
| 225 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 9) e(96);
|
---|
| 226 |
|
---|
| 227 | /* Reading on a O_APPEND flag should be from the current offset. */
|
---|
| 228 | if (lseek(fd1, (off_t) 0, SEEK_SET) != 0) e(97);
|
---|
| 229 | if (read(fd1, buf, BUF_SIZE) != 9) e(98);
|
---|
| 230 | if (strcmp(buf, "fobaoxyz") != 0) e(99);
|
---|
| 231 | if (lseek(fd1, (off_t) 0, SEEK_CUR) != 9) e(100);
|
---|
| 232 |
|
---|
| 233 | if (close(fd1) != 0) e(101);
|
---|
| 234 |
|
---|
| 235 | /* Let's test fifo writes. First blocking. */
|
---|
| 236 | if (mkfifo("fifo", 0777) != 0) e(102);
|
---|
| 237 |
|
---|
| 238 | /* Read from fifo but no writer. */
|
---|
| 239 | System("rm -rf /tmp/sema.38a");
|
---|
| 240 | switch (fork()) {
|
---|
| 241 | case -1: printf("Can't fork\n"); break;
|
---|
| 242 |
|
---|
| 243 | case 0:
|
---|
| 244 | alarm(20);
|
---|
| 245 | if ((fd1 = open("fifo", O_RDONLY)) != 3) e(103);
|
---|
| 246 | system("> /tmp/sema.38a");
|
---|
| 247 | system("while test -f /tmp/sema.38a; do sleep 1; done");
|
---|
| 248 | errno =0;
|
---|
| 249 | if (read(fd1, buf, BUF_SIZE) != 0) e(104);
|
---|
| 250 | if (read(fd1, buf, BUF_SIZE) != 0) e(105);
|
---|
| 251 | if (read(fd1, buf, BUF_SIZE) != 0) e(106);
|
---|
| 252 | if (close(fd1) != 0) e(107);
|
---|
| 253 | exit(0);
|
---|
| 254 |
|
---|
| 255 | default:
|
---|
| 256 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(108);
|
---|
| 257 | while (stat("/tmp/sema.38a", &st1) != 0) sleep(1);
|
---|
| 258 | if (close(fd1) != 0) e(109);
|
---|
| 259 | unlink("/tmp/sema.38a");
|
---|
| 260 | if (wait(&stat_loc) == -1) e(110);
|
---|
| 261 | if (stat_loc != 0) e(111); /* Alarm? */
|
---|
| 262 | }
|
---|
| 263 |
|
---|
| 264 | /* Read from fifo should wait for writer. */
|
---|
| 265 | switch (fork()) {
|
---|
| 266 | case -1: printf("Can't fork\n"); break;
|
---|
| 267 |
|
---|
| 268 | case 0:
|
---|
| 269 | alarm(20);
|
---|
| 270 | if ((fd1 = open("fifo", O_RDONLY)) != 3) e(112);
|
---|
| 271 | if (read(fd1, buf, BUF_SIZE) != 10) e(113);
|
---|
| 272 | if (strcmp(buf, "Hi reader") != 0) e(114);
|
---|
| 273 | if (close(fd1) != 0) e(115);
|
---|
| 274 | exit(0);
|
---|
| 275 |
|
---|
| 276 | default:
|
---|
| 277 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(116);
|
---|
| 278 | sleep(1);
|
---|
| 279 | if (write(fd1, "Hi reader", 10) != 10) e(117);
|
---|
| 280 | if (close(fd1) != 0) e(118);
|
---|
| 281 | if (wait(&stat_loc) == -1) e(119);
|
---|
| 282 | if (stat_loc != 0) e(120); /* Alarm? */
|
---|
| 283 | }
|
---|
| 284 |
|
---|
| 285 | #if DEAD_CODE
|
---|
| 286 | /* Does this test test what it is supposed to test??? */
|
---|
| 287 |
|
---|
| 288 | /* Read from fifo should wait for all writers to close. */
|
---|
| 289 | switch (fork()) {
|
---|
| 290 | case -1: printf("Can't fork\n"); break;
|
---|
| 291 |
|
---|
| 292 | case 0:
|
---|
| 293 | alarm(60);
|
---|
| 294 | switch (fork()) {
|
---|
| 295 | case -1: printf("Can't fork\n"); break;
|
---|
| 296 | case 0:
|
---|
| 297 | alarm(20);
|
---|
| 298 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(121);
|
---|
| 299 | printf("C2 did open\n");
|
---|
| 300 | if (close(fd1) != 0) e(122);
|
---|
| 301 | printf("C2 did close\n");
|
---|
| 302 | exit(0);
|
---|
| 303 | default:
|
---|
| 304 | printf("C1 scheduled\n");
|
---|
| 305 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(123);
|
---|
| 306 | printf("C1 did open\n");
|
---|
| 307 | sleep(2);
|
---|
| 308 | if (close(fd1) != 0) e(124);
|
---|
| 309 | printf("C1 did close\n");
|
---|
| 310 | sleep(1);
|
---|
| 311 | if (wait(&stat_loc) == -1) e(125);
|
---|
| 312 | if (stat_loc != 0) e(126); /* Alarm? */
|
---|
| 313 | }
|
---|
| 314 | exit(stat_loc);
|
---|
| 315 |
|
---|
| 316 | default: {
|
---|
| 317 | int wait_status;
|
---|
| 318 | printf("Parent running\n");
|
---|
| 319 | sleep(1); /* open in childs first */
|
---|
| 320 | if ((fd1 = open("fifo", O_RDONLY)) != 3) e(127);
|
---|
| 321 | if (read(fd1, buf, BUF_SIZE) != 0) e(128);
|
---|
| 322 | if (close(fd1) != 0) e(129);
|
---|
| 323 | printf("Parent closed\n");
|
---|
| 324 | if ((wait_status=wait(&stat_loc)) == -1) e(130);
|
---|
| 325 |
|
---|
| 326 | printf("wait_status %d, stat_loc %d:", wait_status, stat_loc);
|
---|
| 327 | if (WIFSIGNALED(stat_loc)) {
|
---|
| 328 | printf(" killed, signal number %d\n", WTERMSIG(stat_loc));
|
---|
| 329 | }
|
---|
| 330 | else if (WIFEXITED(stat_loc)) {
|
---|
| 331 | printf(" normal exit, status %d\n", WEXITSTATUS(stat_loc));
|
---|
| 332 | }
|
---|
| 333 |
|
---|
| 334 | if (stat_loc != 0) e(131); /* Alarm? */
|
---|
| 335 | }
|
---|
| 336 | }
|
---|
| 337 | #endif
|
---|
| 338 |
|
---|
| 339 | /* PIPE_BUF has to have a nice value. */
|
---|
| 340 | if (PIPE_BUF < 5) e(132);
|
---|
| 341 | if (BUF_SIZE < 1000) e(133);
|
---|
| 342 |
|
---|
| 343 | /* Writes of blocks smaller than PIPE_BUF should be atomic. */
|
---|
| 344 | System("rm -rf /tmp/sema.38b;> /tmp/sema.38b");
|
---|
| 345 | switch (fork()) {
|
---|
| 346 | case -1: printf("Can't fork\n"); break;
|
---|
| 347 |
|
---|
| 348 | case 0:
|
---|
| 349 | alarm(20);
|
---|
| 350 | switch (fork()) {
|
---|
| 351 | case -1: printf("Can't fork\n"); break;
|
---|
| 352 |
|
---|
| 353 | case 0:
|
---|
| 354 | alarm(20);
|
---|
| 355 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(134);
|
---|
| 356 | for (i = 0; i < 100; i++) write(fd1, "1234 ", 5);
|
---|
| 357 | system("while test -f /tmp/sema.38b; do sleep 1; done");
|
---|
| 358 | if (close(fd1) != 0) e(135);
|
---|
| 359 | exit(0);
|
---|
| 360 |
|
---|
| 361 | default:
|
---|
| 362 | if ((fd1 = open("fifo", O_WRONLY)) != 3) e(136);
|
---|
| 363 | for (i = 0; i < 100; i++) write(fd1, "1234 ", 5);
|
---|
| 364 | while (stat("/tmp/sema.38b", &st1) == 0) sleep(1);
|
---|
| 365 | if (close(fd1) != 0) e(137);
|
---|
| 366 | if (wait(&stat_loc) == -1) e(138);
|
---|
| 367 | if (stat_loc != 0) e(139); /* Alarm? */
|
---|
| 368 | }
|
---|
| 369 | exit(stat_loc);
|
---|
| 370 |
|
---|
| 371 | default:
|
---|
| 372 | if ((fd1 = open("fifo", O_RDONLY)) != 3) e(140);
|
---|
| 373 | i = 0;
|
---|
| 374 | memset(buf, '\0', BUF_SIZE);
|
---|
| 375 |
|
---|
| 376 | /* Read buffer full or till EOF. */
|
---|
| 377 | do {
|
---|
| 378 | j = read(fd1, buf + i, BUF_SIZE - i);
|
---|
| 379 | if (j > 0) {
|
---|
| 380 | if (j % 5 != 0) e(141);
|
---|
| 381 | i += j;
|
---|
| 382 | }
|
---|
| 383 | } while (j > 0 && i < 1000);
|
---|
| 384 |
|
---|
| 385 | /* Signal the children to close write ends. This should not be */
|
---|
| 386 | /* Necessary. But due to a bug in 1.16.6 this is necessary. */
|
---|
| 387 | unlink("/tmp/sema.38b");
|
---|
| 388 | if (j < 0) e(142);
|
---|
| 389 | if (i != 1000) e(143);
|
---|
| 390 | if (wait(&stat_loc) == -1) e(144);
|
---|
| 391 | if (stat_loc != 0) e(145); /* Alarm? */
|
---|
| 392 |
|
---|
| 393 | /* Check 200 times 1234. */
|
---|
| 394 | for (i = 0; i < 200; i++)
|
---|
| 395 | if (strncmp(buf + (i * 5), "1234 ", 5) != 0) break;
|
---|
| 396 | if (i != 200) e(146);
|
---|
| 397 | if (buf[1000] != '\0') e(147);
|
---|
| 398 | if (buf[1005] != '\0') e(148);
|
---|
| 399 | if (buf[1010] != '\0') e(149);
|
---|
| 400 | if (read(fd1, buf, BUF_SIZE) != 0) e(150);
|
---|
| 401 | if (close(fd1) != 0) e(151);
|
---|
| 402 | }
|
---|
| 403 |
|
---|
| 404 | /* Read from pipe should wait for writer. */
|
---|
| 405 | if (pipe(tube) != 0) e(152);
|
---|
| 406 | switch (fork()) {
|
---|
| 407 | case -1: printf("Can't fork\n"); break;
|
---|
| 408 | case 0:
|
---|
| 409 | alarm(20);
|
---|
| 410 | if (close(tube[1]) != 0) e(153);
|
---|
| 411 | if (read(tube[0], buf, BUF_SIZE) != 10) e(154);
|
---|
| 412 | if (strcmp(buf, "Hi reader") != 0) e(155);
|
---|
| 413 | if (close(tube[0]) != 0) e(156);
|
---|
| 414 | exit(0);
|
---|
| 415 | default:
|
---|
| 416 | if (close(tube[0]) != 0) e(157);
|
---|
| 417 | sleep(1);
|
---|
| 418 | if (write(tube[1], "Hi reader", 10) != 10) e(158);
|
---|
| 419 | if (close(tube[1]) != 0) e(159);
|
---|
| 420 | if (wait(&stat_loc) == -1) e(160);
|
---|
| 421 | if (stat_loc != 0) e(161); /* Alarm? */
|
---|
| 422 | }
|
---|
| 423 |
|
---|
| 424 | /* Read from pipe should wait for all writers to close. */
|
---|
| 425 | if (pipe(tube) != 0) e(162);
|
---|
| 426 | switch (fork()) {
|
---|
| 427 | case -1: printf("Can't fork\n"); break;
|
---|
| 428 | case 0:
|
---|
| 429 | alarm(20);
|
---|
| 430 | if (close(tube[0]) != 0) e(163);
|
---|
| 431 | switch (fork()) {
|
---|
| 432 | case -1: printf("Can't fork\n"); break;
|
---|
| 433 | case 0:
|
---|
| 434 | alarm(20);
|
---|
| 435 | if (close(tube[1]) != 0) e(164);
|
---|
| 436 | exit(0);
|
---|
| 437 | default:
|
---|
| 438 | sleep(1);
|
---|
| 439 | if (close(tube[1]) != 0) e(165);
|
---|
| 440 | if (wait(&stat_loc) == -1) e(166);
|
---|
| 441 | if (stat_loc != 0) e(167); /* Alarm? */
|
---|
| 442 | }
|
---|
| 443 | exit(stat_loc);
|
---|
| 444 | default:
|
---|
| 445 | if (close(tube[1]) != 0) e(168);
|
---|
| 446 | if (read(tube[0], buf, BUF_SIZE) != 0) e(169);
|
---|
| 447 | if (close(tube[0]) != 0) e(170);
|
---|
| 448 | if (wait(&stat_loc) == -1) e(171);
|
---|
| 449 | if (stat_loc != 0) e(172); /* Alarm? */
|
---|
| 450 | }
|
---|
| 451 |
|
---|
| 452 | /* Writes of blocks smaller than PIPE_BUF should be atomic. */
|
---|
| 453 | System("rm -rf /tmp/sema.38c;> /tmp/sema.38c");
|
---|
| 454 | if (pipe(tube) != 0) e(173);
|
---|
| 455 | switch (fork()) {
|
---|
| 456 | case -1: printf("Can't fork\n"); break;
|
---|
| 457 | case 0:
|
---|
| 458 | alarm(20);
|
---|
| 459 | if (close(tube[0]) != 0) e(174);
|
---|
| 460 | switch (fork()) {
|
---|
| 461 | case -1: printf("Can't fork\n"); break;
|
---|
| 462 | case 0:
|
---|
| 463 | alarm(20);
|
---|
| 464 | for (i = 0; i < 100; i++) write(tube[1], "1234 ", 5);
|
---|
| 465 | system("while test -f /tmp/sema.38c; do sleep 1; done");
|
---|
| 466 | if (close(tube[1]) != 0) e(175);
|
---|
| 467 | exit(0);
|
---|
| 468 | default:
|
---|
| 469 | for (i = 0; i < 100; i++) write(tube[1], "1234 ", 5);
|
---|
| 470 | while (stat("/tmp/sema.38c", &st1) == 0) sleep(1);
|
---|
| 471 | if (close(tube[1]) != 0) e(176);
|
---|
| 472 | if (wait(&stat_loc) == -1) e(177);
|
---|
| 473 | if (stat_loc != 0) e(178); /* Alarm? */
|
---|
| 474 | }
|
---|
| 475 | exit(stat_loc);
|
---|
| 476 | default:
|
---|
| 477 | i = 0;
|
---|
| 478 | if (close(tube[1]) != 0) e(179);
|
---|
| 479 | memset(buf, '\0', BUF_SIZE);
|
---|
| 480 | do {
|
---|
| 481 | j = read(tube[0], buf + i, BUF_SIZE - i);
|
---|
| 482 | if (j > 0) {
|
---|
| 483 | if (j % 5 != 0) e(180);
|
---|
| 484 | i += j;
|
---|
| 485 | } else
|
---|
| 486 | break; /* EOF seen. */
|
---|
| 487 | } while (i < 1000);
|
---|
| 488 | unlink("/tmp/sema.38c");
|
---|
| 489 | if (j < 0) e(181);
|
---|
| 490 | if (i != 1000) e(182);
|
---|
| 491 | if (close(tube[0]) != 0) e(183);
|
---|
| 492 | if (wait(&stat_loc) == -1) e(184);
|
---|
| 493 | if (stat_loc != 0) e(185); /* Alarm? */
|
---|
| 494 |
|
---|
| 495 | /* Check 200 times 1234. */
|
---|
| 496 | for (i = 0; i < 200; i++)
|
---|
| 497 | if (strncmp(buf + (i * 5), "1234 ", 5) != 0) break;
|
---|
| 498 | if (i != 200) e(186);
|
---|
| 499 | }
|
---|
| 500 | }
|
---|
| 501 |
|
---|
| 502 | void test38b()
|
---|
| 503 | {
|
---|
| 504 | int i, fd, stat_loc;
|
---|
| 505 | char buf[BUF_SIZE];
|
---|
| 506 | char buf2[BUF_SIZE];
|
---|
| 507 | struct stat st;
|
---|
| 508 |
|
---|
| 509 | subtest = 2;
|
---|
| 510 | System("rm -rf ../DIR_38/*");
|
---|
| 511 |
|
---|
| 512 | /* Lets try sequential writes. */
|
---|
| 513 | system("rm -rf /tmp/sema.38d");
|
---|
| 514 | System("> testing");
|
---|
| 515 | switch (fork()) {
|
---|
| 516 | case -1: printf("Can't fork\n"); break;
|
---|
| 517 | case 0:
|
---|
| 518 | alarm(20);
|
---|
| 519 | if ((fd = open("testing", O_WRONLY | O_APPEND)) != 3) e(1);
|
---|
| 520 | if (write(fd, "one ", 4) != 4) e(2);
|
---|
| 521 | if (close(fd) != 0) e(3);
|
---|
| 522 | system("> /tmp/sema.38d");
|
---|
| 523 | system("while test -f /tmp/sema.38d; do sleep 1; done");
|
---|
| 524 | if ((fd = open("testing", O_WRONLY | O_APPEND)) != 3) e(4);
|
---|
| 525 | if (write(fd, "three ", 6) != 6) e(5);
|
---|
| 526 | if (close(fd) != 0) e(6);
|
---|
| 527 | system("> /tmp/sema.38d");
|
---|
| 528 | exit(0);
|
---|
| 529 | default:
|
---|
| 530 | while (stat("/tmp/sema.38d", &st) != 0) sleep(1);
|
---|
| 531 | if ((fd = open("testing", O_WRONLY | O_APPEND)) != 3) e(7);
|
---|
| 532 | if (write(fd, "two ", 4) != 4) e(8);
|
---|
| 533 | if (close(fd) != 0) e(9);
|
---|
| 534 | unlink("/tmp/sema.38d");
|
---|
| 535 | while (stat("/tmp/sema.38d", &st) != 0) sleep(1);
|
---|
| 536 | if ((fd = open("testing", O_WRONLY | O_APPEND)) != 3) e(10);
|
---|
| 537 | if (write(fd, "four", 5) != 5) e(11);
|
---|
| 538 | if (close(fd) != 0) e(12);
|
---|
| 539 | if (wait(&stat_loc) == -1) e(13);
|
---|
| 540 | if (stat_loc != 0) e(14); /* The alarm went off? */
|
---|
| 541 | unlink("/tmp/sema.38d");
|
---|
| 542 | }
|
---|
| 543 | if ((fd = open("testing", O_RDONLY)) != 3) e(15);
|
---|
| 544 | if (read(fd, buf, BUF_SIZE) != 19) e(16);
|
---|
| 545 | if (strcmp(buf, "one two three four") != 0) e(17);
|
---|
| 546 | if (close(fd) != 0) e(18);
|
---|
| 547 |
|
---|
| 548 | /* Non written bytes in regular files should be zero. */
|
---|
| 549 | memset(buf2, '\0', BUF_SIZE);
|
---|
| 550 | if ((fd = open("bigfile", O_RDWR | O_CREAT, 0644)) != 3) e(19);
|
---|
| 551 | if (lseek(fd, (off_t) 102400, SEEK_SET) != (off_t) 102400L) e(20);
|
---|
| 552 | if (read(fd, buf, BUF_SIZE) != 0) e(21);
|
---|
| 553 | if (write(fd, ".", 1) != 1) e(22);
|
---|
| 554 | Stat("bigfile", &st);
|
---|
| 555 | if (st.st_size != (off_t) 102401) e(23);
|
---|
| 556 | if (lseek(fd, (off_t) 0, SEEK_SET) != 0) e(24);
|
---|
| 557 | for (i = 0; i < 102400 / BUF_SIZE; i++) {
|
---|
| 558 | if (read(fd, buf, BUF_SIZE) != BUF_SIZE) e(25);
|
---|
| 559 | if (memcmp(buf, buf2, BUF_SIZE) != 0) e(26);
|
---|
| 560 | }
|
---|
| 561 | if (close(fd) != 0) e(27);
|
---|
| 562 | }
|
---|
| 563 |
|
---|
| 564 | void test38c()
|
---|
| 565 | { /* Test correct error behavior. */
|
---|
| 566 | char buf[BUF_SIZE];
|
---|
| 567 | int fd, tube[2], stat_loc;
|
---|
| 568 | struct stat st;
|
---|
| 569 | pid_t pid;
|
---|
| 570 | #ifdef SIGACTION
|
---|
| 571 | struct sigaction act, oact;
|
---|
| 572 | #else
|
---|
| 573 | #if _ANSI
|
---|
| 574 | void (*oldfunc) (int);
|
---|
| 575 | #else
|
---|
| 576 | void (*oldfunc) ();
|
---|
| 577 | #endif
|
---|
| 578 | #endif
|
---|
| 579 |
|
---|
| 580 | subtest = 3;
|
---|
| 581 | System("rm -rf ../DIR_38/*");
|
---|
| 582 |
|
---|
| 583 | /* To test if writing processes on closed pipes are signumbered. */
|
---|
| 584 | #ifdef SIGACTION
|
---|
| 585 | act.sa_handler = setsignumber;
|
---|
| 586 | sigemptyset(&act.sa_mask);
|
---|
| 587 | act.sa_flags = 0;
|
---|
| 588 | if (sigaction(SIGPIPE, &act, &oact) != 0) e(1);
|
---|
| 589 | #else
|
---|
| 590 | oldfunc = signal(SIGPIPE, setsignumber);
|
---|
| 591 | #endif
|
---|
| 592 |
|
---|
| 593 | /* Non valid file descriptors should be an error. */
|
---|
| 594 | for (fd = -111; fd < 0; fd++) {
|
---|
| 595 | errno = 0;
|
---|
| 596 | if (read(fd, buf, BUF_SIZE) != -1) e(2);
|
---|
| 597 | if (errno != EBADF) e(3);
|
---|
| 598 | }
|
---|
| 599 | for (fd = 3; fd < 111; fd++) {
|
---|
| 600 | errno = 0;
|
---|
| 601 | if (read(fd, buf, BUF_SIZE) != -1) e(4);
|
---|
| 602 | if (errno != EBADF) e(5);
|
---|
| 603 | }
|
---|
| 604 | for (fd = -111; fd < 0; fd++) {
|
---|
| 605 | errno = 0;
|
---|
| 606 | if (write(fd, buf, BUF_SIZE) != -1) e(6);
|
---|
| 607 | if (errno != EBADF) e(7);
|
---|
| 608 | }
|
---|
| 609 | for (fd = 3; fd < 111; fd++) {
|
---|
| 610 | errno = 0;
|
---|
| 611 | if (write(fd, buf, BUF_SIZE) != -1) e(8);
|
---|
| 612 | if (errno != EBADF) e(9);
|
---|
| 613 | }
|
---|
| 614 |
|
---|
| 615 | /* Writing a pipe with no readers should trigger SIGPIPE. */
|
---|
| 616 | if (pipe(tube) != 0) e(10);
|
---|
| 617 | close(tube[0]);
|
---|
| 618 | switch (fork()) {
|
---|
| 619 | case -1: printf("Can't fork\n"); break;
|
---|
| 620 | case 0:
|
---|
| 621 | alarm(20);
|
---|
| 622 | signumber = 0;
|
---|
| 623 | if (write(tube[1], buf, BUF_SIZE) != -1) e(11);
|
---|
| 624 | if (errno != EPIPE) e(12);
|
---|
| 625 | if (signumber != SIGPIPE) e(13);
|
---|
| 626 | if (close(tube[1]) != 0) e(14);
|
---|
| 627 | exit(0);
|
---|
| 628 | default:
|
---|
| 629 | close(tube[1]);
|
---|
| 630 | if (wait(&stat_loc) == -1) e(15);
|
---|
| 631 | if (stat_loc != 0) e(16); /* Alarm? */
|
---|
| 632 | }
|
---|
| 633 |
|
---|
| 634 | /* Writing a fifo with no readers should trigger SIGPIPE. */
|
---|
| 635 | System("> /tmp/sema.38e");
|
---|
| 636 | if (mkfifo("fifo", 0666) != 0) e(17);
|
---|
| 637 | switch (fork()) {
|
---|
| 638 | case -1: printf("Can't fork\n"); break;
|
---|
| 639 | case 0:
|
---|
| 640 | alarm(20);
|
---|
| 641 | if ((fd = open("fifo", O_WRONLY)) != 3) e(18);
|
---|
| 642 | system("while test -f /tmp/sema.38e; do sleep 1; done");
|
---|
| 643 | signumber = 0;
|
---|
| 644 | if (write(fd, buf, BUF_SIZE) != -1) e(19);
|
---|
| 645 | if (errno != EPIPE) e(20);
|
---|
| 646 | if (signumber != SIGPIPE) e(21);
|
---|
| 647 | if (close(fd) != 0) e(22);
|
---|
| 648 | exit(0);
|
---|
| 649 | default:
|
---|
| 650 | if ((fd = open("fifo", O_RDONLY)) != 3) e(23);
|
---|
| 651 | if (close(fd) != 0) e(24);
|
---|
| 652 | unlink("/tmp/sema.38e");
|
---|
| 653 | if (wait(&stat_loc) == -1) e(25);
|
---|
| 654 | if (stat_loc != 0) e(26); /* Alarm? */
|
---|
| 655 | }
|
---|
| 656 |
|
---|
| 657 | #ifdef SIGACTION
|
---|
| 658 | /* Restore normal (re)action to SIGPIPE. */
|
---|
| 659 | if (sigaction(SIGPIPE, &oact, NULL) != 0) e(27);
|
---|
| 660 | #else
|
---|
| 661 | signal(SIGPIPE, oldfunc);
|
---|
| 662 | #endif
|
---|
| 663 |
|
---|
| 664 | /* Read from fifo should return -1 and set errno to EAGAIN. */
|
---|
| 665 | System("rm -rf /tmp/sema.38[fgh]");
|
---|
| 666 | switch (fork()) {
|
---|
| 667 | case -1: printf("Can't fork\n"); break;
|
---|
| 668 | case 0:
|
---|
| 669 | alarm(20);
|
---|
| 670 | system("while test ! -f /tmp/sema.38f; do sleep 1; done");
|
---|
| 671 | System("rm -rf /tmp/sema.38f");
|
---|
| 672 | if ((fd = open("fifo", O_WRONLY | O_NONBLOCK)) != 3) e(28);
|
---|
| 673 | close(creat("/tmp/sema.38g", 0666));
|
---|
| 674 | system("while test ! -f /tmp/sema.38h; do sleep 1; done");
|
---|
| 675 | if (close(fd) != 0) e(38);
|
---|
| 676 | System("rm -rf /tmp/sema.38h");
|
---|
| 677 | exit(0);
|
---|
| 678 | default:
|
---|
| 679 | if ((fd = open("fifo", O_RDONLY | O_NONBLOCK)) != 3) e(30);
|
---|
| 680 | close(creat("/tmp/sema.38f", 0666));
|
---|
| 681 | system("while test ! -f /tmp/sema.38g; do sleep 1; done");
|
---|
| 682 | System("rm -rf /tmp/sema.38g");
|
---|
| 683 | if (read(fd, buf, BUF_SIZE) != -1) e(31);
|
---|
| 684 | if (errno != EAGAIN) e(32);
|
---|
| 685 | if (read(fd, buf, BUF_SIZE) != -1) e(33);
|
---|
| 686 | if (errno != EAGAIN) e(34);
|
---|
| 687 | if (read(fd, buf, BUF_SIZE) != -1) e(35);
|
---|
| 688 | if (errno != EAGAIN) e(36);
|
---|
| 689 | close(creat("/tmp/sema.38h", 0666));
|
---|
| 690 | while (stat("/tmp/sema.38h", &st) == 0) sleep(1);
|
---|
| 691 | if (read(fd, buf, BUF_SIZE) != 0) e(37);
|
---|
| 692 | if (close(fd) != 0) e(38);
|
---|
| 693 | if (wait(&stat_loc) == -1) e(39);
|
---|
| 694 | if (stat_loc != 0) e(40); /* Alarm? */
|
---|
| 695 | }
|
---|
| 696 | System("rm -rf fifo");
|
---|
| 697 |
|
---|
| 698 | /* If a read is interrupted by a SIGNAL. */
|
---|
| 699 | if (pipe(tube) != 0) e(41);
|
---|
| 700 | switch (pid = fork()) {
|
---|
| 701 | case -1: printf("Can't fork\n"); break;
|
---|
| 702 | case 0:
|
---|
| 703 | alarm(20);
|
---|
| 704 | #ifdef SIGACTION
|
---|
| 705 | act.sa_handler = setsignumber;
|
---|
| 706 | sigemptyset(&act.sa_mask);
|
---|
| 707 | act.sa_flags = 0;
|
---|
| 708 | if (sigaction(SIGUSR1, &act, &oact) != 0) e(42);
|
---|
| 709 | #else
|
---|
| 710 | oldfunc = signal(SIGUSR1, setsignumber);
|
---|
| 711 | #endif
|
---|
| 712 | if (read(tube[0], buf, BUF_SIZE) != -1) e(43);
|
---|
| 713 | if (errno != EINTR) e(44);
|
---|
| 714 | if (signumber != SIGUSR1) e(45);
|
---|
| 715 | #ifdef SIGACTION
|
---|
| 716 | /* Restore normal (re)action to SIGPIPE. */
|
---|
| 717 | if (sigaction(SIGUSR1, &oact, NULL) != 0) e(46);
|
---|
| 718 | #else
|
---|
| 719 | signal(SIGUSR1, oldfunc);
|
---|
| 720 | #endif
|
---|
| 721 | close(tube[0]);
|
---|
| 722 | close(tube[1]);
|
---|
| 723 | exit(0);
|
---|
| 724 | default:
|
---|
| 725 | /* The sleep 1 should give the child time to start the read. */
|
---|
| 726 | sleep(1);
|
---|
| 727 | close(tube[0]);
|
---|
| 728 | kill(pid, SIGUSR1);
|
---|
| 729 | wait(&stat_loc);
|
---|
| 730 | if (stat_loc != 0) e(47); /* Alarm? */
|
---|
| 731 | close(tube[1]);
|
---|
| 732 | }
|
---|
| 733 | }
|
---|
| 734 |
|
---|
| 735 | void setsignumber(signum)
|
---|
| 736 | int signum;
|
---|
| 737 | {
|
---|
| 738 | signumber = signum;
|
---|
| 739 | }
|
---|
| 740 |
|
---|
| 741 | void e(n)
|
---|
| 742 | int n;
|
---|
| 743 | {
|
---|
| 744 | int err_num = errno; /* Save in case printf clobbers it. */
|
---|
| 745 |
|
---|
| 746 | printf("Subtest %d, error %d errno=%d: ", subtest, n, errno);
|
---|
| 747 | errno = err_num;
|
---|
| 748 | perror("");
|
---|
| 749 | if (errct++ > MAX_ERROR) {
|
---|
| 750 | printf("Too many errors; test aborted\n");
|
---|
| 751 | chdir("..");
|
---|
| 752 | system("rm -rf DIR*");
|
---|
| 753 | exit(1);
|
---|
| 754 | }
|
---|
| 755 | errno = 0;
|
---|
| 756 | }
|
---|
| 757 |
|
---|
| 758 | void quit()
|
---|
| 759 | {
|
---|
| 760 | Chdir("..");
|
---|
| 761 | System("rm -rf DIR_38");
|
---|
| 762 |
|
---|
| 763 | if (errct == 0) {
|
---|
| 764 | printf("ok\n");
|
---|
| 765 | exit(0);
|
---|
| 766 | } else {
|
---|
| 767 | printf("%d errors\n", errct);
|
---|
| 768 | exit(1);
|
---|
| 769 | }
|
---|
| 770 | }
|
---|