| // |
| // 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); |
| } |
| |