// // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. /* * fail.c * testObjects * * Created by Blaine Garst on 9/16/08. * */ #include <stdio.h> #include <unistd.h> #include <fcntl.h> #include <string.h> #include <stdlib.h> #include <stdbool.h> bool readfile(char *buffer, const char *from) { int fd = open(from, 0); if (fd < 0) return false; int count = read(fd, buffer, 512); if (count < 0) return false; buffer[count] = 0; // zap newline return true; } // basic idea, take compiler args, run compiler, and verify that expected failure matches any existing one int main(int argc, char *argv[]) { if (argc == 1) return 0; char *copy[argc+1]; // make a copy // find and strip off -e "errorfile" char *errorfile = NULL; int counter = 0, i = 0; for (i = 1; i < argc; ++i) { // skip 0 arg which is "fail" if (!strncmp(argv[i], "-e", 2)) { errorfile = argv[++i]; } else { copy[counter++] = argv[i]; } } copy[counter] = NULL; pid_t child = fork(); char buffer[512]; if (child == 0) { // in child sprintf(buffer, "/tmp/errorfile_%d", getpid()); close(1); int fd = creat(buffer, 0777); if (fd != 1) { fprintf(stderr, "didn't open custom error file %s as 1, got %d\n", buffer, fd); exit(1); } close(2); dup(1); int result = execv(copy[0], copy); exit(10); } if (child < 0) { printf("fork failed\n"); exit(1); } int status = 0; pid_t deadchild = wait(&status); if (deadchild != child) { printf("wait got %d instead of %d\n", deadchild, child); exit(1); } if (WEXITSTATUS(status) == 0) { printf("compiler exited normally, not good under these circumstances\n"); exit(1); } //printf("exit status of child %d was %d\n", child, WEXITSTATUS(status)); sprintf(buffer, "/tmp/errorfile_%d", child); if (errorfile) { //printf("ignoring error file: %s\n", errorfile); char desired[512]; char got[512]; bool gotErrorFile = readfile(desired, errorfile); bool gotOutput = readfile(got, buffer); if (!gotErrorFile && gotOutput) { printf("didn't read errorfile %s, it should have something from\n*****\n%s\n*****\nin it.\n", errorfile, got); exit(1); } else if (gotErrorFile && gotOutput) { char *where = strstr(got, desired); if (!where) { printf("didn't find contents of %s in %s\n", errorfile, buffer); exit(1); } } else { printf("errorfile %s and output %s inconsistent\n", errorfile, buffer); exit(1); } } unlink(buffer); printf("success\n"); exit(0); }