1 /*
2 *
3 * Copyright (c) International Business Machines Corp., 2002
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
13 * the GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18 */
19
20 /* 11/01/2002 Port to LTP robbiew@us.ibm.com */
21 /* 06/30/2001 Port to Linux nsharoff@us.ibm.com */
22
23 /*inode02.c */
24 /*======================================================================
25 =================== TESTPLAN SEGMENT ===================
26 CALLS: mkdir, stat, open
27
28 Run with TERM mode.
29
30 >KEYS: < file system and I/O management, system resource constraints.
31 >WHAT: < Can the system handle a heavy load on the file system I/O
32 < functions?
33 >HOW: < Create several identical process that call inode02.c. This
34 < will simulate the multi-user environment, and hopefully uncover
35 < conflicts that might occur in "real life" use.
36 >BUGS: <
37 ======================================================================*/
38
39 #define PATH_STRING_LENGTH 1024
40 #define NAME_LENGTH 8
41 #define MAX_PATH_STRING_LENGTH (PATH_STRING_LENGTH - NAME_LENGTH - 40)
42 #define DIRECTORY_MODE 00777
43 #define FILE_MODE 00777
44
45 #define MKDIR_STRING_LENGTH (MAX_PATH_STRING_LENGTH + 7)
46
47 /* #define DEBUG you can watch the generation with this flag */
48
49 #define TRUE 1
50 #define FALSE 0
51 #define READ 0
52 #define WRITE 1
53
54 #include <stdio.h>
55 #include <errno.h>
56 #include <sys/types.h>
57 #include <sys/stat.h>
58 #include <signal.h>
59 #include <fcntl.h>
60 #include <errno.h>
61 #include <sys/wait.h>
62
63 #ifdef LINUX
64 #include <stdlib.h>
65 #include <unistd.h>
66 #include <string.h>
67 #endif
68
69 #define MAXCHILD 25
70 int allchild[MAXCHILD + 1];
71
72 char name[NAME_LENGTH + 1];
73 char path_string[PATH_STRING_LENGTH + 1];
74 char read_string[PATH_STRING_LENGTH + 1];
75 char write_string[PATH_STRING_LENGTH + 1];
76 char remove_string[PATH_STRING_LENGTH + 10];
77 int parent_pid;
78 int nchild;
79
80 FILE *list_stream = NULL;
81 int list_id;
82 int file_id;
83
84 int increment_name(), get_next_name(), mode(), escrivez(), massmurder();
85 int max_depth, max_breadth, file_length;
86 int bd_arg(char *);
87
88 #ifdef LINUX
89 void (*sigset(int, void (*)(int))) (int);
90 #endif
91
92 /** LTP Port **/
93 #include "test.h"
94
95 void setup(void);
96 void fail_exit(void);
97 void anyfail(void);
98 void ok_exit(void);
99 void forkfail(void);
100 void terror(char *);
101 int instress(void);
102
103 #define FAILED 0
104 #define PASSED 1
105
106 int local_flag = PASSED;
107 FILE *temp;
108
109 char *TCID = "inode02"; /* Test program identifier. */
110 int TST_TOTAL = 1; /* Total number of test cases. */
111 /**************/
112
main(int argc,char * argv[])113 int main(int argc, char *argv[])
114 {
115 int pid, tree(), p, status;
116 int count, child;
117 register int i;
118 int term();
119
120 setup();
121
122 parent_pid = getpid();
123
124 if (sigset(SIGTERM, (void (*)())term) == SIG_ERR) {
125 tst_resm(TBROK, "\tSIGTERM sigset set failed, errno=%d",
126 errno);
127 exit(1);
128 }
129
130 /************************************************/
131 /* */
132 /* Input the parameters for the directory--- */
133 /* file trees which are to be generated */
134 /* */
135 /************************************************/
136
137 if (argc < 2) {
138 max_depth = 6;
139 max_breadth = 5;
140 file_length = 8;
141 nchild = 5;
142 } else if (argc < 5) {
143 tst_resm(TCONF, "Bad argument count.");
144 printf
145 ("\tinode02 max_depth max_breadth file_length #children\n\tdefault: inode02 6 5 8 5\n");
146 exit(1);
147 } else {
148 i = 1;
149 if (sscanf(argv[i++], "%d", &max_depth) != 1)
150 bd_arg(argv[i - 1]);
151 if (sscanf(argv[i++], "%d", &max_breadth) != 1)
152 bd_arg(argv[i - 1]);
153 if (sscanf(argv[i++], "%d", &file_length) != 1)
154 bd_arg(argv[i - 1]);
155 if (sscanf(argv[i++], "%d", &nchild) != 1)
156 bd_arg(argv[i - 1]);
157 if (nchild > MAXCHILD) {
158 fprintf(temp, "too many children - max is %d\n",
159 MAXCHILD);
160 exit(1);
161 }
162 }
163
164 /************************************************/
165 /* */
166 /* Generate and check nchild trees */
167 /* */
168 /************************************************/
169
170 for (p = 0; p < nchild; p++) {
171 pid = fork();
172 if (pid == 0) {
173 tree();
174 } else {
175 if (pid < 1) {
176 terror
177 ("Fork failed (may be OK if under stress)");
178 massmurder();
179 if (instress()) {
180 ok_exit();
181 }
182 forkfail();
183 }
184 }
185 }
186
187 count = 0;
188 while ((child = wait(&status)) > 0) {
189 #ifdef DEBUG
190 tst_resm(TINFO, "Test %d exited status = %d\n", child, status);
191 #endif
192 if (status) {
193 fprintf(temp, "Test %d failed - expected 0 exit.\n",
194 child);
195 local_flag = FAILED;
196 }
197 count++;
198 }
199
200 if (count != nchild) {
201 tst_resm(TFAIL, "Wrong number of children waited on!");
202 tst_resm(TFAIL, "Saw %d, expected %d", count, nchild);
203 local_flag = FAILED;
204 }
205
206 /************************************************/
207 /* */
208 /* And report the results.......... */
209 /* */
210 /************************************************/
211
212 anyfail();
213 /** NOT REACHED **/
214 tst_exit();
215 }
216
bd_arg(char * str)217 int bd_arg(char *str)
218 {
219 fprintf(temp,
220 "Bad argument - %s - could not parse as number.\n\tinode02 [max_depth] [max_breadth] [file_length] [#children]\n\tdefault: inode02 6 5 8 5\n",
221 str);
222 exit(1);
223 }
224
tree(void)225 int tree(void)
226
227 /************************************************/
228 /* */
229 /* TREE */
230 /* */
231 /* generate a tree of directories and files */
232 /* and save the path names in the path_list */
233 /* file */
234 /* */
235 /* then, read the path names and attempt to */
236 /* access the corresponding directories and */
237 /* files */
238 /* */
239 /************************************************/
240 {
241 int gen_ret_val, ch_ret_val, exit_val, level;
242 int ret_val;
243 int generate(), check();
244 char path_list_string[PATH_STRING_LENGTH + 10];
245 int len;
246 int status;
247 int snp_ret;
248
249 /********************************/
250 /* */
251 /* make the root directory for */
252 /* the tree */
253 /* */
254 /********************************/
255
256 sprintf(path_string, "inode02.%d", getpid());
257
258 ret_val = mkdir(path_string, DIRECTORY_MODE);
259
260 if (ret_val == -1) {
261 tst_resm(TBROK,
262 "Reason: Impossible to create directory %s, errno=%d\n",
263 path_string, errno);
264 exit(-5);
265 }
266
267 strcpy(remove_string, "rm -rf ");
268 strcat(remove_string, path_string);
269
270 #ifdef DEBUG
271 tst_resm(TINFO, "\n%s\n", path_string);
272 #endif
273
274 /****************************************/
275 /* */
276 /* create the "path_list" file, in */
277 /* which the list of generated paths */
278 /* will be stored so that they later */
279 /* may be checked */
280 /* */
281 /****************************************/
282
283 snp_ret = snprintf(path_list_string, sizeof(path_list_string),
284 "%s/path_list", path_string);
285 if (snp_ret < 0 || snp_ret >= sizeof(path_list_string)) {
286 tst_resm(TBROK, "snprintf(path_list_string,..) returned %d",
287 snp_ret);
288 exit(-1);
289 }
290 list_id = creat(path_list_string, FILE_MODE);
291 if (list_id == -1) {
292 fprintf(temp,
293 "\nThe path_list file '%s' cannot be created, errno=%d\n",
294 path_list_string, errno);
295 exit(-7);
296 }
297
298 /****************************************/
299 /* */
300 /* and store its name in path_list */
301 /* */
302 /****************************************/
303
304 strcpy(write_string, path_string);
305 len = strlen(write_string);
306 write_string[len++] = 'D';
307 write_string[len] = '\0';
308 escrivez(write_string);
309
310 /****************************************/
311 /* */
312 /* generate the directory-file tree */
313 /* */
314 /****************************************/
315
316 level = 0;
317
318 #ifdef DEBUG
319 tst_resm(TINFO, "\n\t%s\n\n", "GENERATING:");
320 #endif
321
322 gen_ret_val = generate(path_string, level);
323 close(list_id);
324 list_id = open(path_list_string, READ);
325 if (list_id == -1) {
326 fprintf(temp,
327 "\nThe path_list file cannot be opened for reading, errno=%d\n",
328 errno);
329 exit(-8);
330 }
331 list_stream = fdopen(list_id, "r");
332
333 /****************************************/
334 /* */
335 /* check the directory-file tree */
336 /* for correctness */
337 /* */
338 /****************************************/
339
340 #ifdef DEBUG
341 tst_resm(TINFO, "\n\t%s\n\n", "CHECKING:");
342 #endif
343
344 ch_ret_val = check();
345
346 exit_val = MIN(ch_ret_val, gen_ret_val);
347
348 status = fclose(list_stream);
349 if (status != 0) {
350 fprintf(temp,
351 "Failed to close list_stream: ret=%d errno=%d (%s)\n",
352 status, errno, strerror(errno));
353 exit(-8);
354 }
355
356 /*
357 * Remove file.
358 */
359
360 status = system(remove_string);
361 if (status) {
362 fprintf(temp, "Caution - `%s' failed.\n", remove_string);
363 fprintf(temp, "Status returned %d.\n", status);
364 }
365
366 /****************************************/
367 /* */
368 /* .....and exit main */
369 /* */
370 /****************************************/
371
372 exit(exit_val);
373 }
374
generate(char * string,int level)375 int generate(char *string, int level)
376
377 /****************************************/
378 /* */
379 /* generate recursively a tree of */
380 /* directories and files: within */
381 /* created directory, an alternating */
382 /* series of files and directories */
383 /* are constructed---until tree */
384 /* breadth and depth limits are */
385 /* reached or an error occurs */
386 /* */
387 /****************************************/
388 /***************************/
389 /* string: */
390 /* the directory path */
391 /* string below which a */
392 /* tree is generated */
393 /* */
394 /***************************/
395
396 /***************************/
397 /* level: */
398 /* the tree depth variable */
399 /* */
400 /***************************/
401 {
402 int switch_flag;
403 int ret_val = 0;
404 int new_ret_val, len, ret_len;
405 char new_string[PATH_STRING_LENGTH + 1];
406 int new_level;
407 int i, j; /* iteration counters */
408 int snp_ret;
409
410 switch_flag = level & TRUE;
411 if (strlen(string) >= MAX_PATH_STRING_LENGTH) {
412
413 /********************************/
414 /* */
415 /* Maximum path name length */
416 /* reached */
417 /* */
418 /********************************/
419
420 fprintf(temp, "\nMaximum path_name length reached\n");
421 return (-1);
422 } else if (level < max_depth) {
423 for (i = 0; i <= max_breadth; i++) {
424 get_next_name();
425 snp_ret = snprintf(new_string, sizeof(new_string),
426 "%s/%s", string, name);
427 if (snp_ret < 0 || snp_ret >= sizeof(new_string)) {
428 tst_resm(TBROK, "snprintf(new_string,..) "
429 "returned %d", snp_ret);
430 exit(-1);
431 }
432
433 /****************************************/
434 /* */
435 /* switch between creating files */
436 /* and making directories */
437 /* */
438 /****************************************/
439
440 if (switch_flag) {
441 switch_flag = FALSE;
442
443 /****************************************/
444 /* */
445 /* create a new file */
446 /* */
447 /****************************************/
448
449 file_id = creat(new_string, FILE_MODE);
450 if (file_id == -1) {
451 fprintf(temp,
452 "\nImpossible to create file %s, errno=%d\n",
453 new_string, errno);
454 return (-2);
455 }
456 #ifdef DEBUG
457 tst_resm(TINFO, "%d %s F\n", level,
458 new_string);
459 #endif
460
461 /****************************************/
462 /* */
463 /* write to it */
464 /* */
465 /****************************************/
466
467 len = strlen(new_string);
468 for (j = 1; j <= file_length; j++) {
469 ret_len =
470 write(file_id, new_string, len);
471 if (ret_len != len) {
472 fprintf(temp,
473 "\nUnsuccessful write to file %s, errno=%d\n",
474 new_string, errno);
475 return (-3);
476 }
477 }
478 close(file_id);
479
480 /****************************************/
481 /* */
482 /* and store its name in path_list */
483 /* */
484 /****************************************/
485
486 strcpy(write_string, new_string);
487 len = strlen(write_string);
488 write_string[len++] = 'F';
489 write_string[len] = '\0';
490 escrivez(write_string);
491 } else {
492 switch_flag = TRUE;
493
494 /****************************************/
495 /* */
496 /* or make a directory */
497 /* */
498 /* (mknod can only be called when in */
499 /* super user mode) */
500 /* */
501 /****************************************/
502
503 ret_val = mkdir(new_string, DIRECTORY_MODE);
504
505 if (ret_val != 0) {
506 fprintf(temp,
507 "\nImpossible to create directory %s, errno=%d\n",
508 new_string, errno);
509 return (-5);
510 }
511 #ifdef DEBUG
512 tst_resm(TINFO, "%d %s D\n", level,
513 new_string);
514 #endif
515
516 /****************************************/
517 /* */
518 /* store its name in path_list */
519 /* */
520 /****************************************/
521
522 strcpy(write_string, new_string);
523 len = strlen(write_string);
524 write_string[len++] = 'D';
525 write_string[len] = '\0';
526 escrivez(write_string);
527
528 /****************************************/
529 /* */
530 /* and generate a new level */
531 /* */
532 /****************************************/
533
534 new_level = level + 1;
535 new_ret_val = generate(new_string, new_level);
536 if (new_ret_val < ret_val)
537 ret_val = new_ret_val;
538 }
539 }
540
541 /********************************/
542 /* */
543 /* Maximum breadth reached */
544 /* */
545 /********************************/
546
547 return (ret_val);
548 } else
549 /********************************/
550 /* */
551 /* Maximum depth reached */
552 /* */
553 /********************************/
554 return 0;
555 }
556
check(void)557 int check(void)
558
559 /****************************************/
560 /* */
561 /* check for file and directory */
562 /* correctness by reading records */
563 /* from the path_list and attempting */
564 /* to determine if the corresponding */
565 /* files or directories are as */
566 /* created */
567 /* */
568 /****************************************/
569 {
570 int len, path_mode, val, ret_len, j;
571
572 for (;;) {
573
574 /****************************************/
575 /* */
576 /* read a path string from path_list */
577 /* */
578 /****************************************/
579
580 if (fscanf(list_stream, "%s", path_string) == EOF) {
581
582 #ifdef DEBUG
583 tst_resm(TINFO, "\nEnd of path_list file reached \n");
584 #endif
585
586 return 0;
587 }
588 #ifdef DEBUG
589 tst_resm(TINFO, "%s\n", path_string);
590 #endif
591
592 len = strlen(path_string);
593 len--;
594 if (path_string[len] == 'F') {
595
596 /********************************/
597 /* */
598 /* this should be a file */
599 /* */
600 /********************************/
601
602 path_string[len] = '\0';
603 file_id = open(path_string, READ);
604 if (file_id <= 0) {
605 fprintf(temp,
606 "\nImpossible to open file %s, errno=%d\n",
607 path_string, errno);
608 return (-1);
609 }
610
611 else {
612 /********************************/
613 /* */
614 /* check its contents */
615 /* */
616 /********************************/
617
618 ret_len = 0;
619 len = strlen(path_string);
620 for (j = 1; j <= file_length; j++) {
621 ret_len =
622 read(file_id, read_string, len);
623 if (len != ret_len) {
624 fprintf(temp,
625 "\nFile read error for file %s, errno=%d\n",
626 path_string, errno);
627 return (-3);
628 }
629 read_string[len] = '\0';
630 val = strcmp(read_string, path_string);
631 if (val != 0) {
632 fprintf(temp,
633 "\nContents of file %s are different than expected: %s\n",
634 path_string,
635 read_string);
636 return (-4);
637 }
638 }
639 close(file_id);
640 } /* else for */
641 if (ret_len <= 0) {
642 fprintf(temp,
643 "\nImpossible to read file %s, errno=%d\n",
644 path_string, errno);
645 return (-2);
646 }
647 } else {
648
649 /********************************/
650 /* */
651 /* otherwise.......... */
652 /* it should be a directory */
653 /* */
654 /********************************/
655
656 path_string[len] = '\0';
657 path_mode = mode(path_string);
658 if (path_mode == -1) {
659 fprintf(temp,
660 "\nPreviously created directory path %s was not open\n",
661 path_string);
662 return (-4);
663 }
664 if ((040000 & path_mode) != 040000) {
665 fprintf(temp,
666 "\nPath %s was not recognized to be a directory\n",
667 path_string);
668 fprintf(temp, "Its mode is %o\n", path_mode);
669 return (-5);
670 }
671 }
672 } /* while */
673 }
674
get_next_name(void)675 int get_next_name(void)
676
677 /****************************************/
678 /* */
679 /* get the next---in a dictionary */
680 /* sense---file or directory name */
681 /* */
682 /****************************************/
683 {
684 static int k;
685 int i;
686 int last_position;
687
688 last_position = NAME_LENGTH - 1;
689 if (k == 0) {
690
691 /************************/
692 /* */
693 /* initialize name */
694 /* */
695 /************************/
696
697 for (i = 0; i < NAME_LENGTH; i++)
698 name[i] = 'a';
699 name[NAME_LENGTH] = '\0';
700 k++;
701 }
702 /********************************/
703 /* */
704 else
705 increment_name(last_position); /* i.e., beginning at the last */
706 /* position */
707 /* */
708 /********************************/
709 return 0;
710 }
711
increment_name(int position)712 int increment_name(int position)
713
714 /****************************************/
715 /* */
716 /* recursively revise the letters in */
717 /* a name to get the lexiographically */
718 /* next name */
719 /* */
720 /****************************************/
721 {
722 int next_position;
723
724 if (name[position] == 'z')
725 if (position == 0) {
726 fprintf(temp,
727 "ERROR: There are no more available names\n");
728 exit(-1);
729 } else {
730 name[position] = 'a'; /**********************/
731 next_position = --position; /* */
732 increment_name(next_position); /* increment the */
733 /* previous letter */
734 /* */
735 /**********************/
736 }
737 /*********************************/
738 /* */
739 else
740 name[position]++; /* otherwise, increment this one */
741 return 0; /* */
742 /*********************************/
743 }
744
mode(char * path_string)745 int mode(char *path_string)
746
747 /****************************************/
748 /* */
749 /* determine and return the mode of */
750 /* the file named by path_string */
751 /* */
752 /****************************************/
753 {
754 struct stat buf;
755 int ret_val, mod;
756
757 ret_val = stat(path_string, &buf);
758 if (ret_val == -1)
759 return (-1);
760 else {
761 mod = buf.st_mode;
762 return (mod);
763 }
764 }
765
escrivez(char * string)766 int escrivez(char *string)
767 {
768 char write_string[PATH_STRING_LENGTH + 1];
769 int len, ret_len;
770
771 strcpy(write_string, string);
772 len = strlen(write_string);
773 write_string[len] = '\n';
774 len++;
775 ret_len = write(list_id, write_string, len);
776 if (len != ret_len) {
777 fprintf(temp,
778 "A string of deviant length %d written to path_list, errno=%d\n",
779 ret_len, errno);
780 exit(-2);
781 }
782 return 0;
783 }
784
term(void)785 int term(void)
786 {
787 int status;
788
789 fflush(temp);
790 if (parent_pid == getpid()) {
791 massmurder(); /* kill kids */
792 fprintf(temp, "\term1 - SIGTERM received by parent.\n");
793 fflush(temp);
794 } else {
795 fprintf(temp, "\tchild - got SIGTERM signal.\n");
796 if (list_stream != NULL)
797 fclose(list_stream);
798 close(list_id);
799 close(file_id);
800 status = system(remove_string);
801 if (status) {
802 fprintf(temp, "Caution - ``%s'' returned status %d\n",
803 remove_string, status);
804 }
805 exit(0);
806 }
807 return 0;
808 }
809
massmurder(void)810 int massmurder(void)
811 {
812 int i;
813 for (i = 0; i < MAXCHILD; i++) {
814 if (allchild[i]) {
815 kill(allchild[i], SIGTERM);
816 }
817 }
818 return 0;
819 }
820
821 /** LTP Port **/
822 /*
823 * setup
824 *
825 * Do set up - here its a dummy function
826 */
setup(void)827 void setup(void)
828 {
829 tst_tmpdir();
830 temp = stderr;
831 }
832
833 /*
834 * fail_exit()
835 *
836 * Exit on failure
837 */
fail_exit(void)838 void fail_exit(void)
839 {
840 tst_brkm(TFAIL, tst_rmdir, "Test failed");
841 }
842
843 /*
844 *
845 * Function: anyfail()
846 *
847 * Description: Exit a test.
848 */
anyfail(void)849 void anyfail(void)
850 {
851 (local_flag == FAILED) ? tst_resm(TFAIL, "Test failed")
852 : tst_resm(TPASS, "Test passed");
853 tst_rmdir();
854 tst_exit();
855 }
856
857 /*
858 * ok_exit
859 *
860 * Calling block passed the test
861 */
ok_exit(void)862 void ok_exit(void)
863 {
864 local_flag = PASSED;
865 return;
866 }
867
868 /*
869 * forkfail()
870 *
871 * exit on failure
872 */
forkfail(void)873 void forkfail(void)
874 {
875 tst_brkm(TBROK, tst_rmdir, "Reason: %s", strerror(errno));
876 }
877
878 /*
879 * Function: terror
880 *
881 * Description: prints error message this may not be because some part of the
882 * test case failed, for example fork() failed. We will log this
883 * failure as TBROK instead of TFAIL.
884 */
terror(char * message)885 void terror(char *message)
886 {
887 tst_resm(TBROK, "Reason: %s:%s", message, strerror(errno));
888 return;
889 }
890
891 /*
892 * instress
893 *
894 * Assume that we are always running under stress, so this function will
895 * return > 0 value always.
896 */
instress(void)897 int instress(void)
898 {
899 tst_resm(TINFO, "System resource may be too low, fork() malloc()"
900 " etc are likely to fail.\n");
901 return 1;
902 }
903