/* SPDX-License-Identifier: copyleft-next-0.3.1 */ /* Copyright 2024 Kim Kuparinen < kimi.h.kuparinen@gmail.com > */ /** * @file compiler.c * * Top level compiler implementation. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /** * Read whole file into a buffer and return pointer to buffer. * Possibly kind of silly to have both \p file and \p f. * Apparently there's no standardized way to get the file name of a * file pointer. * * @param file Name of file to read. * @param f File pointer. * @return Pointer to buffer with file contents. */ static char *read_file(const char *file, FILE *f) { fseek(f, 0, SEEK_END); /** @todo check how well standardized this actually is */ long s = ftell(f); if (s == LONG_MAX) { error("%s might be a directory", file); return NULL; } fseek(f, 0, SEEK_SET); char *buf = malloc((size_t)(s + 1)); if (!buf) return NULL; fread(buf, (size_t)(s + 1), 1, f); /* remember terminating null */ buf[s] = 0; return buf; } /** * Helper for process_file(), actually processes the file * after process_file() has done path lookups and working directory * changes and whatnot. * * @param parent Parent file context. \c NULL if root file. * @param public \c 1 if file is being imported publicly, \c 0 otherwise. * @param file File name to process. * @return \c 0 if processing was succesful, non-zero value otherwise. */ static int process(struct scope *scope, const char *file) { FILE *f = fopen(file, "rb"); if (!f) { error("failed opening %s: %s", file, strerror(errno)); return -1; } const char *buf = read_file(file, f); fclose(f); if (!buf) return -1; struct parser *p = create_parser(); if (!p) { free((void *)buf); return -1; } parse(p, file, buf); struct ast *tree = p->tree; bool failed = p->failed; destroy_parser(p); if (failed) { free((void *)buf); return -1; } ast_dump_list(0, tree); scope->fctx.fbuf = buf; scope->fctx.fname = strdup(file); if (analyze_root(scope, tree)) return -1; if (mvcheck_root(tree)) return -1; return 0; } #define MAP_KEY const char * #define MAP_TYPE struct scope * #define MAP_CMP(a, b) strcmp((a), (b)) #define MAP_HASH(a) CONTS_MAP_STR_HASH(a) #define MAP_NAME scopes #include /* ugly global for now */ static struct scopes scopes; static void destroy_scopes() { foreach(scopes, n, &scopes) { destroy_scope(n->data); free((void *)n->key); } scopes_destroy(&scopes); } struct scope *compile_file(const char *file) { struct scope *scope = NULL; const char *base = NULL, *dir = NULL, *cwd = NULL, *real = NULL; if (!(base = fwd_basename(file))) { error("couldn't get basename of %s", file); goto out; } if (!(dir = fwd_dirname(file))) { error("couldn't get dirname of %s", file); goto out; } if (!(cwd = fwd_cwdname())) { error("couldn't get current working dir"); goto out; } if (*dir != 0 && chdir(dir)) { error("couldn't change to directory %s: %s", dir, strerror(errno)); goto out; } if (!(real = realpath(base, NULL))) { error("no such file: %s", file); goto out; } struct scope **exists = scopes_find(&scopes, real); if (exists) { scope = *exists; goto out; } scope = create_scope(); scopes_insert(&scopes, strdup(real), scope); if (process(scope, base)) { scope = NULL; goto out; } if (chdir(cwd)) { error("couldn't change back to directory %s: %s", cwd, strerror(errno)); goto out; } out: free((void *)base); free((void *)dir); free((void *)cwd); free((void *)real); return scope; } int compile(const char *input) { scopes = scopes_create(1); int ret = -1; struct scope *root = compile_file(input); if (!root) { error("processing of %s stopped due to errors", input); goto out; } if ((ret = lower(root))) { error("lowering of %s failed due to errors", input); goto out; } out: destroy_scopes(); destroy_allocs(); return ret; }