ref: 52a4d88af911ccd62125f55e72155b3d76878a66
dir: /myrbuild/myrbuild.c/
#include <stdlib.h> #include <stdio.h> #include <stdint.h> #include <string.h> #include <assert.h> #include <sys/types.h> #include <sys/stat.h> #include <sys/wait.h> #include <sys/utsname.h> #include <ctype.h> #include <fcntl.h> #include <unistd.h> #include <regex.h> #include <err.h> #include "parse.h" #include "../config.h" /* make libparse happy */ Node *file; char *filename; /* binaries we call out to */ char *mc = "6m"; char *as = "as"; char *ar = "ar"; char *ld = "ld"; char *muse = "muse"; /* the name of the output file */ char *libname; char *binname; /* additional paths to search for packages */ char **incpaths; size_t nincpaths; /* libraries to link against. */ char **libs; size_t nlibs; /* the linker script to use */ char *ldscript; char *sysname; regex_t usepat; Htab *compiled; Htab *loopdetect; static void usage(char *prog) { printf("%s [-h] [-I path] [-l lib] [-b bin] inputs...\n", prog); printf("\t-h\tprint this help\n"); printf("\t-b bin\tBuild a binary called 'bin'\n"); printf("\t-l lib\tBuild a library called 'name'\n"); printf("\t-s script\tUse the linker script 'script' when linking\n"); printf("\t-I path\tAdd 'path' to use search path\n"); } int hassuffix(char *path, char *suffix) { int pathlen; int sufflen; pathlen = strlen(path); sufflen = strlen(suffix); if (sufflen > pathlen) return 0; return !strcmp(&path[pathlen-sufflen], suffix); } int isquoted(char *path) { return path[0] == '"' && path[strlen(path) - 1] == '"'; } char *usetomyr(char *path) { char buf[1024]; /* skip initial quote */ path++; if (!hassuffix(path, ".use\"")) { fprintf(stderr, "\"%s, should end with \".use\"\n", path); exit(1); } swapsuffix(buf, 1024, path, ".use\"", ".myr"); return strdup(buf); } void printl(char **lst) { printf("\t"); printf("%s\t", *lst++); while (*lst) printf("%s ", *lst++); printf("\n"); } void gencmd(char ***cmd, size_t *ncmd, char *bin, char *file, char **extra, size_t nextra) { size_t i; *cmd = NULL; *ncmd = 0; lappend(cmd, ncmd, bin); for (i = 0; i < nincpaths; i++) { lappend(cmd, ncmd, "-I"); lappend(cmd, ncmd, incpaths[i]); } for (i = 0; i < nextra; i++) lappend(cmd, ncmd, extra[i]); lappend(cmd, ncmd, file); lappend(cmd, ncmd, NULL); } void run(char **cmd) { pid_t pid; int status; printl(cmd); pid = fork(); status = 0; if (pid == -1) { err(1, "Could not fork"); } else if (pid == 0) { if (execvp(cmd[0], cmd) == -1) err(1, "Failed to exec %s", cmd[0]); } else { waitpid(pid, &status, 0); } if (WIFEXITED(status) && WEXITSTATUS(status) != 0) exit(WEXITSTATUS(status)); else if (WIFSIGNALED(status)) die("%s: exited with signal %d\n", cmd[0], WTERMSIG(status)); } int isfresh(char *from, char *to) { struct stat from_sb, to_sb; if (stat(from, &from_sb)) err(1, "Could not find %s", from); if (stat(to, &to_sb) == -1) return 0; return from_sb.st_mtime <= to_sb.st_mtime; } int inlist(char **list, size_t sz, char *str) { size_t i; for (i = 0; i < sz; i++) if (!strcmp(list[i], str)) return 1; return 0; } void getdeps(char *file, char **deps, size_t depsz, size_t *ndeps) { char buf[2048]; /* if you hit this limit, shoot yourself */ regmatch_t m[2]; size_t i; FILE *f; char *dep; f = fopen(file, "r"); if (!f) err(1, "Could not open file \"%s\"", file); i = 0; while (fgets(buf, sizeof buf, f)) { if (regexec(&usepat, buf, 2, m, 0) == REG_NOMATCH) continue; if (i == depsz) die("Too many deps for file %s", file); dep = strdupn(&buf[m[1].rm_so], m[1].rm_eo - m[1].rm_so); if (!inlist(deps, i, dep)) deps[i++] = dep; else free(dep); } fclose(f); *ndeps = i; } void compile(char *file) { size_t i, ndeps; char **cmd; size_t ncmd; char *s; char *localdep; char *deps[512]; char use[1024]; char obj[1024]; char *extra[] = {"-g", "-o", "" /* filename */}; if (hthas(compiled, file)) return; if (hthas(loopdetect, file)) die("Cycle in dependency graph, involving %s\n", file); htput(loopdetect, file, file); if (hassuffix(file, ".myr")) { swapsuffix(use, sizeof use, file, ".myr", ".use"); swapsuffix(obj, sizeof obj, file, ".myr", ".o"); getdeps(file, deps, 512, &ndeps); for (i = 0; i < ndeps; i++) { if (isquoted(deps[i])) { localdep = usetomyr(deps[i]); compile(localdep); free(localdep); } else if (!inlist(libs, nlibs, deps[i])) { lappend(&libs, &nlibs, deps[i]); } } if (isfresh(file, use)) goto done; if (isfresh(file, obj)) goto done; gencmd(&cmd, &ncmd, muse, file, NULL, 0); run(cmd); gencmd(&cmd, &ncmd, mc, file, NULL, 0); run(cmd); } else if (hassuffix(file, ".s")) { swapsuffix(obj, sizeof obj, file, ".s", ".o"); if (isfresh(file, obj)) goto done; extra[2] = obj; gencmd(&cmd, &ncmd, as, file, extra, 3); run(cmd); } done: s = strdup(file); htput(compiled, s, s); htdel(loopdetect, file); } void mergeuse(char **files, size_t nfiles) { char **args; size_t i, nargs; char buf[1024]; args = NULL; nargs = 0; lappend(&args, &nargs, strdup(muse)); lappend(&args, &nargs, strdup("-mo")); lappend(&args, &nargs, strdup(libname)); for (i = 0; i < nfiles; i++) { if (hassuffix(files[i], ".myr")) { swapsuffix(buf, sizeof buf, files[i], ".myr", ".use"); lappend(&args, &nargs, strdup(buf)); } else if (!hassuffix(files[i], ".s")) { die("Unknown file type %s", files[i]); } } lappend(&args, &nargs, NULL); run(args); for (i = 0; i < nargs; i++) free(args[i]); lfree(&args, &nargs); } void archive(char **files, size_t nfiles) { char **args; size_t i, nargs; char buf[1024]; args = NULL; nargs = 0; snprintf(buf, sizeof buf, "lib%s.a", libname); lappend(&args, &nargs, strdup(ar)); lappend(&args, &nargs, strdup("-rcs")); lappend(&args, &nargs, strdup(buf)); for (i = 0; i < nfiles; i++) { if (hassuffix(files[i], ".myr")) swapsuffix(buf, sizeof buf, files[i], ".myr", ".o"); else if (hassuffix(files[i], ".s")) swapsuffix(buf, sizeof buf, files[i], ".s", ".o"); else die("Unknown file type %s", files[i]); lappend(&args, &nargs, strdup(buf)); } lappend(&args, &nargs, NULL); run(args); for (i = 0; i < nargs; i++) free(args[i]); lfree(&args, &nargs); } void linkobj(char **files, size_t nfiles) { char **args; size_t i, nargs; char buf[1024]; if (!binname) binname = "a.out"; args = NULL; nargs = 0; /* ld -T ldscript -o outfile */ lappend(&args, &nargs, strdup(ld)); lappend(&args, &nargs, strdup("-o")); lappend(&args, &nargs, strdup(binname)); /* ld -T ldscript */ if (ldscript) { snprintf(buf, sizeof buf, "-T%s", ldscript); lappend(&args, &nargs, strdup(buf)); } /* ld -T ldscript -o outfile foo.o bar.o baz.o */ for (i = 0; i < nfiles; i++) { if (hassuffix(files[i], ".myr")) swapsuffix(buf, sizeof buf, files[i], ".myr", ".o"); else if (hassuffix(files[i], ".s")) swapsuffix(buf, sizeof buf, files[i], ".s", ".o"); else die("Unknown file type %s", files[i]); lappend(&args, &nargs, strdup(buf)); } /* ld -T ldscript -o outfile foo.o bar.o baz.o -L/path1 -L/path2 */ for (i = 0; i < nincpaths; i++) { snprintf(buf, sizeof buf, "-L%s", incpaths[i]); lappend(&args, &nargs, strdup(buf)); } snprintf(buf, sizeof buf, "-L%s%s", Instroot, "/lib/myr"); lappend(&args, &nargs, strdup(buf)); /* ld -T ldscript -o outfile foo.o bar.o baz.o -L/path1 -L/path2 -llib1 -llib2*/ for (i = 0; i < nlibs; i++) { snprintf(buf, sizeof buf, "-l%s", libs[i]); lappend(&args, &nargs, strdup(buf)); } /* OSX wants a minimum version specified to prevent warnings*/ if (!strcmp(sysname, "Darwin")) { lappend(&args, &nargs, strdup("-macosx_version_min")); lappend(&args, &nargs, strdup("10.6")); } /* the null terminator for exec() */ lappend(&args, &nargs, NULL); run(args); for (i = 0; i < nargs; i++) free(args[i]); lfree(&args, &nargs); } int main(int argc, char **argv) { int opt; int i; struct utsname name; if (uname(&name) == 0) sysname = strdup(name.sysname); while ((opt = getopt(argc, argv, "hb:l:s:I:C:A:M:L:R:")) != -1) { switch (opt) { case 'b': binname = optarg; break; case 'l': libname = optarg; break; case 's': ldscript = optarg; break; case 'C': mc = optarg; break; case 'A': as = optarg; break; case 'M': muse = optarg; break; case 'L': ld = optarg; break; case 'R': ar = optarg; break; case 'I': lappend(&incpaths, &nincpaths, optarg); break; case 'h': usage(argv[0]); exit(0); break; default: usage(argv[0]); exit(0); break; } } if (libname && binname) die("Can't specify both library and binary names"); compiled = mkht(strhash, streq); loopdetect = mkht(strhash, streq); regcomp(&usepat, "^[[:space:]]*use[[:space:]]+([^[:space:]]+)", REG_EXTENDED); for (i = optind; i < argc; i++) compile(argv[i]); if (libname) { mergeuse(&argv[optind], argc - optind); archive(&argv[optind], argc - optind); } else { linkobj(&argv[optind], argc - optind); } return 0; }