#include <string.h>
#include <assert.h>
+#ifdef _WIN32
+
+#include <fcntl.h>
+#include <io.h>
+
+/* no eXecute on Win32 */
+#define X_OK 0
+#define W_OK 2
+#define R_OK 4
+
+#define O_RDWR _O_RDWR
+#define O_CREAT _O_CREAT
+#define O_EXCL _O_EXCL
+#define O_BINARY _O_BINARY
+
+/* remap some names, we are not in the POSIX world */
+#define access(fname, mode) _access(fname, mode)
+#define mktemp(tmpl) _mktemp(tmpl)
+#define open(fname, oflag, mode) _open(fname, oflag, mode)
+#define fdopen(fd, mode) _fdopen(fd, mode)
+#define popen(cmd, mode) _popen(cmd, mode)
+#define pclose(file) _pclose(file)
+
+#else
+#include <unistd.h>
+#define HAVE_MKSTEMP
+#endif
+
#ifndef WITH_LIBCORE
#define WITH_LIBCORE
#endif
#include "lexer.h"
#include "token_t.h"
+#include "types.h"
#include "type_hash.h"
#include "parser.h"
#include "ast2firm.h"
+#include "diagnostic.h"
+#include "lang_features.h"
+#include "driver/firm_opt.h"
+#include "driver/firm_cmdline.h"
#include "adt/error.h"
#include "write_fluffy.h"
+#include "revision.h"
#ifndef PREPROCESSOR
#define PREPROCESSOR "cpp -std=c99 -U__WCHAR_TYPE__ -D__WCHAR_TYPE__=int"
#endif
#ifndef LINKER
-#define LINKER "gcc"
+#define LINKER "gcc -m32"
#endif
-#ifdef _WIN32
-/* remap some names */
-#define popen(cmd, mode) _popen(cmd, mode)
-#define pclose(file) _pclose(file)
-#endif /* _WIN32 */
+#ifndef ASSEMBLER
+#define ASSEMBLER "as --32"
+#endif
-static int verbose;
-static bool do_dump;
+/** The current c mode/dialect. */
+unsigned int c_mode = _C89|_C99|_GNUC;
-static const ir_settings_if_conv_t *if_conv_info = NULL;
-static const backend_params *be_params = NULL;
+/** The 'machine size', 16, 32 or 64 bit, 32bit is the default. */
+unsigned int machine_size = 32;
-static void initialize_firm(void)
+/** true if the char type is signed. */
+bool char_is_signed = true;
+
+/** true for strict language checking. */
+bool strict_mode = false;
+
+static int verbose;
+static struct obstack cppflags_obst;
+
+#if defined(_DEBUG) || defined(FIRM_DEBUG)
+/**
+ * Debug printf implementation.
+ *
+ * @param fmt printf style format parameter
+ */
+void dbg_printf(const char *fmt, ...)
{
- be_opt_register();
- firm_init_options(NULL, 0, NULL);
-
- firm_parameter_t params;
- memset(¶ms, 0, sizeof(params));
-
- params.size = sizeof(params);
- params.enable_statistics = 0;
- params.initialize_local_func = uninitialized_local_var;
- params.cc_mask = 0;
- params.builtin_dbg = NULL;
-
- /* initialize backend */
- be_params = be_init();
- ir_set_debug_retrieve(retrieve_dbg);
- params.arch_op_settings = be_params->arch_op_settings;
- if_conv_info = be_params->if_conv_info;
-
- (void) if_conv_info; /* avoid unused warning */
-
- /* intialize firm itself */
- init_firm(¶ms);
- dbg_init(NULL, NULL, dbg_snprint);
-
- set_opt_constant_folding(1);
- set_opt_unreachable_code(1);
- set_opt_control_flow_straightening(1);
- set_opt_control_flow_weak_simplification(1);
- set_opt_control_flow_strong_simplification(1);
- set_opt_dyn_meth_dispatch(1);
- set_opt_normalize(1);
- set_opt_precise_exc_context(0);
- set_opt_strength_red(0);
- set_opt_fragile_ops(0);
- set_opt_optimize_class_casts(0);
- set_opt_suppress_downcast_optimization(0);
- set_opt_remove_confirm(1);
- set_opt_scalar_replacement(1);
- set_opt_ldst_only_null_ptr_exceptions(1);
- set_opt_alias_analysis(1);
+ va_list list;
- dump_consts_local(1);
- dump_keepalive_edges(1);
+ if (firm_dump.debug_print) {
+ va_start(list, fmt);
+ vprintf(fmt, list);
+ va_end(list);
+ } /* if */
}
+#endif /* defined(_DEBUG) || defined(FIRM_DEBUG) */
-static void dump(ir_graph *irg, const char *suffix)
+static void initialize_firm(void)
{
- if(do_dump) {
- dump_ir_block_graph(irg, suffix);
- }
+ firm_early_init();
+
+ dump_consts_local(1);
+ dump_keepalive_edges(1);
}
static void get_output_name(char *buf, size_t buflen, const char *inputname,
{
size_t last_dot = 0xffffffff;
size_t i = 0;
+
+ if(inputname == NULL) {
+ snprintf(buf, buflen, "a%s", newext);
+ return;
+ }
+
for(const char *c = inputname; *c != 0; ++c) {
if(*c == '.')
last_dot = i;
return unit;
}
-static void lextest(const char *fname)
+static void lextest(FILE *in, const char *fname)
{
- FILE *in = fopen(fname, "r");
- if(in == NULL) {
- fprintf(stderr, "Couldn't open '%s': %s\n", fname, strerror(errno));
- exit(1);
- }
-
lexer_open_stream(in, fname);
do {
print_token(stdout, &lexer_token);
puts("");
} while(lexer_token.type != T_EOF);
-
- fclose(in);
}
-static void backend(const char *inputname, const char *outname)
-{
- FILE *out = fopen(outname, "w");
- if(out == NULL) {
- fprintf(stderr, "couldn't open '%s' for writing: %s\n", outname,
- strerror(errno));
- exit(1);
- }
-
- be_main(out, inputname);
-
- fclose(out);
-}
-
-static void emit(const char *input_name, const char *out_name)
-{
- backend(input_name, out_name);
-}
-
-static FILE* preprocess(const char *in)
+static FILE* preprocess(FILE* in, const char *fname)
{
char buf[4096];
+ const char *flags = obstack_finish(&cppflags_obst);
- snprintf(buf, sizeof(buf), PREPROCESSOR " %s", in);
+ if(in != stdin) {
+ snprintf(buf, sizeof(buf), PREPROCESSOR " %s %s", flags, fname);
+ } else {
+ /* read from stdin */
+ snprintf(buf, sizeof(buf), PREPROCESSOR " %s -", flags);
+ }
if(verbose) {
puts(buf);
return f;
}
-static void link(const char *in, const char *out)
+static void do_link(const char *out, const char *in)
{
char buf[4096];
}
}
-static void assemble(const char *in, const char *out)
+static void assemble(const char *out, const char *in)
{
char buf[4096];
- snprintf(buf, sizeof(buf), "%s %s -c -o %s", LINKER, in, out);
+ snprintf(buf, sizeof(buf), "%s %s -o %s", ASSEMBLER, in, out);
if(verbose) {
puts(buf);
}
+
int err = system(buf);
if(err != 0) {
fprintf(stderr, "assembler reported an error\n");
}
}
-static void create_firm_prog(translation_unit_t *unit)
+static const char *try_dir(const char *dir)
{
- translation_unit_to_firm(unit);
+ if(dir == NULL)
+ return dir;
+ if(access(dir, R_OK | W_OK | X_OK) == 0)
+ return dir;
+ return NULL;
+}
+
+static const char *get_tempdir(void)
+{
+ static const char *tmpdir = NULL;
+
+ if(tmpdir != NULL)
+ return tmpdir;
+
+ if(tmpdir == NULL)
+ tmpdir = try_dir(getenv("TMPDIR"));
+ if(tmpdir == NULL)
+ tmpdir = try_dir(getenv("TMP"));
+ if(tmpdir == NULL)
+ tmpdir = try_dir(getenv("TEMP"));
+
+#ifdef P_tmpdir
+ if(tmpdir == NULL)
+ tmpdir = try_dir(P_tmpdir);
+#endif
+
+ if(tmpdir == NULL)
+ tmpdir = try_dir("/var/tmp");
+ if(tmpdir == NULL)
+ tmpdir = try_dir("/usr/tmp");
+ if(tmpdir == NULL)
+ tmpdir = try_dir("/tmp");
+
+ if(tmpdir == NULL)
+ tmpdir = ".";
+
+ return tmpdir;
+}
+
+#ifndef HAVE_MKSTEMP
+/* cheap and nasty mkstemp replacement */
+static int mkstemp(char *templ)
+{
+ mktemp(templ);
+ return open(templ, O_RDWR|O_CREAT|O_EXCL|O_BINARY, 0600);
+}
+#endif
+
+/**
+ * an own version of tmpnam, which: writes in a buffer, appends a user specified
+ * suffix, emits no warnings during linking (like glibc/gnu ld do for tmpnam)...
+ */
+static FILE *make_temp_file(char *buffer, size_t buflen,
+ const char *prefix, const char *suffix)
+{
+ const char *tempdir = get_tempdir();
+
+ /* oh well... mkstemp doesn't accept a suffix after XXXXXX... */
+ (void) suffix;
+ suffix = "";
- //dump_globals_as_text(dump_verbosity_max, "-globals");
+ snprintf(buffer, buflen, "%s/%sXXXXXX%s", tempdir, prefix, suffix);
- int n_irgs = get_irp_n_irgs();
- for(int i = 0; i < n_irgs; ++i) {
- ir_graph *const irg = get_irp_irg(i);
- dump(irg, "-start");
+ int fd = mkstemp(buffer);
+ if(fd == -1) {
+ fprintf(stderr, "couldn't create temporary file: %s\n",
+ strerror(errno));
+ exit(1);
}
+ FILE *out = fdopen(fd, "w");
+ if(out == NULL) {
+ fprintf(stderr, "couldn't create temporary file FILE*\n");
+ exit(1);
+ }
+
+ return out;
+}
+/**
+ * Do the necessary lowering for compound parameters.
+ */
+void lower_compound_params(void)
+{
lower_params_t params;
params.def_ptr_alignment = 4;
params.find_pointer_type = NULL;
params.ret_compound_in_regs = NULL;
lower_calls_with_compounds(¶ms);
-
- lower_highlevel();
-
- for(int i = 0; i < n_irgs; ++i) {
- ir_graph *const irg = get_irp_irg(i);
- dump(irg, "-lower");
- }
-}
-
-static void optimize(void)
-{
- int arr_len;
- ir_entity **keep_methods;
-
- cgana(&arr_len, &keep_methods);
- gc_irgs(arr_len, keep_methods);
- free(keep_methods);
-
- optimize_funccalls(0);
-
- lwrdw_param_t lwrdw_param = {
- 1,
- 1,
- mode_Ls, mode_Lu,
- mode_Is, mode_Iu,
- def_create_intrinsic_fkt,
- NULL
- };
- if (be_params->arch_create_intrinsic_fkt) {
- lwrdw_param.create_intrinsic = be_params->arch_create_intrinsic_fkt;
- lwrdw_param.ctx = be_params->create_intrinsic_ctx;
- }
-
- for(int i = 0; i < get_irp_n_irgs(); ++i) {
- ir_graph *irg = get_irp_irg(i);
-
- optimize_graph_df(irg);
- dump(irg, "-01-localopt");
- place_code(irg);
- dump(irg, "-02-place");
- optimize_cf(irg);
- dump(irg, "-03-cf");
- lower_dw_ops(&lwrdw_param);
- dump(irg, "-04-dw");
- optimize_graph_df(irg);
- dump(irg, "-05-localopt");
- optimize_cf(irg);
- dump(irg, "-06-cf");
- }
}
typedef enum compile_mode_t {
+ ParseOnly,
Compile,
+ CompileDump,
CompileAssemble,
CompileAssembleLink,
LexTest,
{
initialize_firm();
- init_symbol_table();
- init_tokens();
- init_types();
- init_typehash();
- init_lexer();
- init_ast();
- init_parser();
- init_ast2firm();
-
- const char *input = NULL;
- const char *outname = NULL;
- compile_mode_t mode = CompileAssembleLink;
+ const char *input = NULL;
+ const char *outname = NULL;
+ const char *dumpfunction = NULL;
+ compile_mode_t mode = CompileAssembleLink;
+
+ obstack_init(&cppflags_obst);
+
+#define GET_ARG_AFTER(def, args) \
+ def = &arg[sizeof(args)-1]; \
+ if(def[0] == '\0') { \
+ ++i; \
+ if(i >= argc) { \
+ fprintf(stderr, "error: expected argument after '" args "'\n"); \
+ argument_errors = true; \
+ break; \
+ } \
+ def = argv[i]; \
+ if(def[0] == '-' && def[1] != '\0') { \
+ fprintf(stderr, "error: expected argument after '" args "'\n"); \
+ argument_errors = true; \
+ continue; \
+ } \
+ }
+ bool help_displayed = false;
+ bool argument_errors = false;
for(int i = 1; i < argc; ++i) {
const char *arg = argv[i];
- if(strcmp(arg, "-o") == 0) {
- ++i;
- if(i >= argc) {
- usage(argv[0]);
- return 1;
- }
- outname = argv[i];
+ if(strncmp(arg, "-o", 2) == 0) {
+ GET_ARG_AFTER(outname, "-o");
} else if(strcmp(arg, "-c") == 0) {
mode = CompileAssemble;
} else if(strcmp(arg, "-S") == 0) {
mode = Compile;
+ } else if(strcmp(arg, "--gcc") == 0) {
+ c_mode |= _GNUC;
+ } else if(strcmp(arg, "--no-gcc") == 0) {
+ c_mode &= ~_GNUC;
+ } else if(strcmp(arg, "--ms") == 0) {
+ c_mode |= _MS;
+ } else if(strcmp(arg, "--signed-chars") == 0) {
+ char_is_signed = true;
+ } else if(strcmp(arg, "--unsigned-chars") == 0) {
+ char_is_signed = false;
+ } else if(strcmp(arg, "--strict") == 0) {
+ strict_mode = true;
+ } else if(strcmp(arg, "--no-ms") == 0) {
+ c_mode &= ~_MS;
} else if(strcmp(arg, "--lextest") == 0) {
mode = LexTest;
} else if(strcmp(arg, "--print-ast") == 0) {
mode = PrintAst;
} else if(strcmp(arg, "--print-fluffy") == 0) {
mode = PrintFluffy;
- } else if(strcmp(arg, "--dump") == 0) {
- do_dump = true;
+ } else if(strcmp(arg, "--version") == 0) {
+ firm_version_t ver;
+ firm_get_version(&ver);
+ printf("cparser (%d.%d %s) using libFirm (%u.%u", 0, 1, cparser_REVISION, ver.major, ver.minor);
+ if(ver.revision[0] != 0) {
+ putchar(' ');
+ fputs(ver.revision, stdout);
+ }
+ if(ver.build[0] != 0) {
+ putchar(' ');
+ fputs(ver.build, stdout);
+ }
+ puts(")\n");
+ exit(EXIT_SUCCESS);
+ } else if(strcmp(arg, "-fsyntax-only") == 0) {
+ mode = ParseOnly;
+ } else if(strncmp(arg, "-I", 2) == 0) {
+ const char *opt;
+ GET_ARG_AFTER(opt, "-I");
+ obstack_printf(&cppflags_obst, " -I%s", opt);
+ } else if(strncmp(arg, "-D", 2) == 0) {
+ const char *opt;
+ GET_ARG_AFTER(opt, "-D");
+ obstack_printf(&cppflags_obst, " -D%s", opt);
+ } else if(strncmp(arg, "-U", 2) == 0) {
+ const char *opt;
+ GET_ARG_AFTER(opt, "-U");
+ obstack_printf(&cppflags_obst, " -U%s", opt);
+ } else if(strcmp(arg, "--dump-function") == 0) {
+ ++i;
+ if(i >= argc) {
+ fprintf(stderr, "error: "
+ "expected argument after '--dump-function'\n");
+ argument_errors = true;
+ break;
+ }
+ dumpfunction = argv[i];
+ mode = CompileDump;
} else if(strcmp(arg, "-v") == 0) {
verbose = 1;
+ } else if(strcmp(arg, "-w") == 0) {
+ inhibit_all_warnings = true;
} else if(arg[0] == '-' && arg[1] == 'f') {
- const char *opt = &arg[2];
- if(opt[0] == 0) {
- ++i;
- if(i >= argc) {
- usage(argv[0]);
- return 1;
- }
- opt = argv[i];
- if(opt[0] == '-') {
- usage(argv[0]);
- return 1;
+ const char *opt;
+ GET_ARG_AFTER(opt, "-f");
+
+ if(strcmp(opt, "omit-frame-pointer") == 0) {
+ firm_be_option("omitfp");
+ } else if(strcmp(opt, "no-omit-frame-pointer") == 0) {
+ firm_be_option("omitfp=no");
+ } else {
+ int res = firm_option(opt);
+ if (res == 0) {
+ fprintf(stderr, "error: unknown Firm option '-f %s'\n",
+ opt);
+ argument_errors = true;
+ continue;
+ } else if (res == -1) {
+ help_displayed = true;
}
}
- //firm_option(opt);
} else if(arg[0] == '-' && arg[1] == 'b') {
- const char *opt = &arg[2];
- if(opt[0] == 0) {
- ++i;
- if(i >= argc) {
- usage(argv[0]);
- return 1;
- }
- opt = argv[i];
- if(opt[0] == '-') {
- usage(argv[0]);
- return 1;
- }
+ const char *opt;
+ GET_ARG_AFTER(opt, "-b");
+ int res = firm_be_option(opt);
+ if (res == 0) {
+ fprintf(stderr, "error: unknown Firm backend option '-b %s'\n",
+ opt);
+ argument_errors = true;
+ } else if (res == -1) {
+ help_displayed = true;
+ }
+ } else if(arg[0] == '-' && arg[1] == 'W') {
+ const char *opt;
+ GET_ARG_AFTER(opt, "-W");
+ if (strcmp(opt, "error") == 0) {
+ warnings_are_errors = true;
+ } else if (strcmp(opt, "fatal-errors") == 0) {
+ fatal_errors = true;
+ } else {
+ fprintf(stderr, "warning: ignoring gcc option -W%s\n", opt);
+ }
+ } else if(arg[0] == '-' && arg[1] == 'm') {
+ const char *opt;
+ GET_ARG_AFTER(opt, "-m");
+ char *endptr;
+ long int value = strtol(opt, &endptr, 10);
+ if (*endptr != '\0') {
+ fprintf(stderr, "error: wrong option '-m %s'\n", opt);
+ argument_errors = true;
+ }
+ if (value != 16 && value != 32 && value != 64) {
+ fprintf(stderr, "error: option -m supports only 16, 32 or 64\n");
+ argument_errors = true;
+ } else {
+ machine_size = (unsigned int)value;
}
- //firm_be_option(opt);
} else if(arg[0] == '-') {
- if (arg[1] == 'D' ||
- arg[1] == 'O' ||
- arg[1] == 'f' ||
- arg[1] == 'W' ||
+ if (arg[1] == '\0') {
+ if(input != NULL) {
+ fprintf(stderr, "error: multiple input files specified\n");
+ argument_errors = true;
+ } else {
+ input = arg;
+ }
+ } else if(strcmp(arg, "-pedantic") == 0) {
+ fprintf(stderr, "warning: ignoring gcc option '%s'\n", arg);
+ } else if(arg[1] == 'O' ||
arg[1] == 'g' ||
strncmp(arg + 1, "std=", 4) == 0) {
- fprintf(stderr, "Warning: Ignoring option '%s'\n", arg);
+ fprintf(stderr, "warning: ignoring gcc option '%s'\n", arg);
} else {
- usage(argv[0]);
- return 1;
+ fprintf(stderr, "error: unknown argument '%s'\n", arg);
+ argument_errors = true;
}
} else {
if(input != NULL) {
- fprintf(stderr, "Error: multiple input files specified\n");
- usage(argv[0]);
- return 1;
+ fprintf(stderr, "error: multiple input files specified\n");
+ argument_errors = true;
} else {
input = arg;
}
}
}
+ /* we do the lowering in ast2firm */
+ firm_opt.lower_bitfields = FALSE;
+
+ if(help_displayed) {
+ return !argument_errors;
+ }
+ if(argument_errors) {
+ usage(argv[0]);
+ return 1;
+ }
+
+ gen_firm_init();
+ init_symbol_table();
+ init_tokens();
+ init_types();
+ init_typehash();
+ init_basic_types();
+ init_lexer();
+ init_ast();
+ init_parser();
+ init_ast2firm();
+
+ FILE *out = NULL;
+ char outnamebuf[4096];
+ if(outname == NULL) {
+ switch(mode) {
+ case PrintAst:
+ case PrintFluffy:
+ case LexTest:
+ if(outname == NULL)
+ outname = "-";
+ break;
+ case ParseOnly:
+ break;
+ case Compile:
+ get_output_name(outnamebuf, sizeof(outnamebuf), input, ".s");
+ outname = outnamebuf;
+ break;
+ case CompileAssemble:
+ get_output_name(outnamebuf, sizeof(outnamebuf), input, ".o");
+ outname = outnamebuf;
+ break;
+ case CompileDump:
+ get_output_name(outnamebuf, sizeof(outnamebuf), dumpfunction,
+ ".vcg");
+ outname = outnamebuf;
+ break;
+ case CompileAssembleLink:
+ outname = "a.out";
+ break;
+ }
+ }
+
+ if(outname != NULL) {
+ if(strcmp(outname, "-") == 0) {
+ out = stdout;
+ } else {
+ out = fopen(outname, "w");
+ if(out == NULL) {
+ fprintf(stderr, "Couldn't open '%s' for writing: %s\n", outname,
+ strerror(errno));
+ return 1;
+ }
+ }
+ }
+
+ FILE *in;
if(input == NULL) {
fprintf(stderr, "%s: no input files\n", argv[0]);
return 1;
+ } else if(strcmp(input, "-") == 0) {
+ in = stdin;
+ input = "<stdin>";
+ } else {
+ in = fopen(input, "r");
+ if(in == NULL) {
+ fprintf(stderr, "Couldn't open '%s': %s\n", input, strerror(errno));
+ return 1;
+ }
}
if(mode == LexTest) {
- lextest(input);
+ lextest(in, input);
+ fclose(in);
return 0;
}
- FILE *const in = preprocess(input);
- translation_unit_t *const unit = do_parsing(in, input);
- pclose(in);
- if(unit == NULL)
- return 1;
+ FILE *preprocessed_in = preprocess(in, input);
+ translation_unit_t *const unit = do_parsing(preprocessed_in, input);
+ pclose(preprocessed_in);
+ if(unit == NULL) {
+ /* parsing failed because of errors */
+ fprintf(stderr, "%u error(s), %u warning(s)\n", error_count, warning_count);
+ return EXIT_FAILURE;
+ }
+ if (warning_count > 0) {
+ fprintf(stderr, "%u warning(s)\n", warning_count);
+ }
if(mode == PrintAst) {
+ type_set_output(out);
+ ast_set_output(out);
print_ast(unit);
return 0;
}
if(mode == PrintFluffy) {
- ast_set_output(stdout);
- write_fluffy_decls(unit);
+ type_set_output(out);
+ ast_set_output(out);
+ write_fluffy_decls(out, unit);
}
- char outsname[4096];
- const char *sname = NULL;
- if(mode == Compile) {
- sname = outname;
+ translation_unit_to_firm(unit);
+
+ if(mode == ParseOnly) {
+ return 0;
}
- if(sname == NULL) {
- get_output_name(outsname, sizeof(outsname), input, ".s");
- sname = outsname;
+
+ FILE *asm_out;
+ char asm_tempfile[1024];
+ if(mode == CompileDump) {
+ asm_out = NULL;
+ firm_be_opt.selection = BE_NONE;
+ } else if(mode == Compile) {
+ asm_out = out;
+ } else {
+ asm_out
+ = make_temp_file(asm_tempfile, sizeof(asm_tempfile), "cc", ".s");
+ }
+ gen_firm_finish(asm_out, input, /*c_mode=*/1, /*firm_const_exists=*/0);
+
+ if(mode == CompileDump) {
+ /* find irg */
+ ident *id = new_id_from_str(dumpfunction);
+ ir_graph *irg = NULL;
+ int n_irgs = get_irp_n_irgs();
+ for(int i = 0; i < n_irgs; ++i) {
+ ir_graph *tirg = get_irp_irg(i);
+ ident *irg_id = get_entity_ident(get_irg_entity(tirg));
+ if(irg_id == id) {
+ irg = tirg;
+ break;
+ }
+ }
+
+ if(irg == NULL) {
+ fprintf(stderr, "No graph for function '%s' found\n", dumpfunction);
+ return 1;
+ }
+
+ dump_ir_block_graph_file(irg, out);
+ fclose(out);
+ return 0;
}
- create_firm_prog(unit);
- optimize();
- emit(input, sname);
+ fclose(asm_out);
- if(mode == CompileAssemble) {
- char outoname[4096];
- const char *oname = outname;
- if(oname == NULL) {
- get_output_name(outoname, sizeof(outoname), input, ".o");
- oname = outoname;
+ /* assemble assembler and create object file */
+ char obj_tfile[1024];
+ if(mode == CompileAssemble || mode == CompileAssembleLink) {
+ const char *obj_outfile;
+ if(mode == CompileAssemble) {
+ fclose(out);
+ obj_outfile = outname;
+ } else {
+ FILE *tempf
+ = make_temp_file(obj_tfile, sizeof(obj_tfile), "cc", ".o");
+ fclose(tempf);
+ obj_outfile = obj_tfile;
}
- assemble(sname, oname);
- } else {
- assert(mode == CompileAssembleLink);
- if(outname == NULL)
- outname = "a.out";
+ assemble(obj_outfile, asm_tempfile);
+ }
- link(sname, outname);
+ /* link object file */
+ if(mode == CompileAssembleLink) {
+ do_link(outname, obj_tfile);
}
+ obstack_free(&cppflags_obst, NULL);
+
exit_ast2firm();
exit_parser();
exit_ast();