209 lines
5.4 KiB
C++
209 lines
5.4 KiB
C++
// go.cc -- Go frontend main file for gcc.
|
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
#include "go-system.h"
|
|
|
|
#include "go-c.h"
|
|
#include "go-diagnostics.h"
|
|
|
|
#include "lex.h"
|
|
#include "parse.h"
|
|
#include "backend.h"
|
|
#include "gogo.h"
|
|
|
|
// The data structures we build to represent the file.
|
|
static Gogo* gogo;
|
|
|
|
// Create the main IR data structure.
|
|
|
|
GO_EXTERN_C
|
|
void
|
|
go_create_gogo(const struct go_create_gogo_args* args)
|
|
{
|
|
go_assert(::gogo == NULL);
|
|
::gogo = new Gogo(args->backend, args->linemap, args->int_type_size,
|
|
args->pointer_size);
|
|
|
|
if (args->pkgpath != NULL)
|
|
::gogo->set_pkgpath(args->pkgpath);
|
|
else if (args->prefix != NULL)
|
|
::gogo->set_prefix(args->prefix);
|
|
|
|
if (args->relative_import_path != NULL)
|
|
::gogo->set_relative_import_path(args->relative_import_path);
|
|
::gogo->set_check_divide_by_zero(args->check_divide_by_zero);
|
|
::gogo->set_check_divide_overflow(args->check_divide_overflow);
|
|
if (args->compiling_runtime)
|
|
::gogo->set_compiling_runtime(args->compiling_runtime);
|
|
if (args->c_header != NULL)
|
|
::gogo->set_c_header(args->c_header);
|
|
if (args->embedcfg != NULL)
|
|
::gogo->read_embedcfg(args->embedcfg);
|
|
::gogo->set_debug_escape_level(args->debug_escape_level);
|
|
if (args->debug_escape_hash != NULL)
|
|
::gogo->set_debug_escape_hash(args->debug_escape_hash);
|
|
::gogo->set_nil_check_size_threshold(args->nil_check_size_threshold);
|
|
if (args->debug_optimization)
|
|
::gogo->set_debug_optimization(args->debug_optimization);
|
|
if (args->need_eqtype)
|
|
::gogo->set_need_eqtype(args->need_eqtype);
|
|
}
|
|
|
|
// Parse the input files.
|
|
|
|
GO_EXTERN_C
|
|
void
|
|
go_parse_input_files(const char** filenames, unsigned int filename_count,
|
|
bool only_check_syntax, bool)
|
|
{
|
|
go_assert(filename_count > 0);
|
|
|
|
Lex::Linknames all_linknames;
|
|
for (unsigned int i = 0; i < filename_count; ++i)
|
|
{
|
|
if (i > 0)
|
|
::gogo->clear_file_scope();
|
|
|
|
const char* filename = filenames[i];
|
|
FILE* file;
|
|
if (strcmp(filename, "-") == 0)
|
|
file = stdin;
|
|
else
|
|
{
|
|
file = fopen(filename, "r");
|
|
if (file == NULL)
|
|
go_fatal_error(Linemap::unknown_location(),
|
|
"cannot open %s: %m", filename);
|
|
}
|
|
|
|
Lex lexer(filename, file, ::gogo->linemap());
|
|
|
|
Parse parse(&lexer, ::gogo);
|
|
parse.program();
|
|
|
|
if (strcmp(filename, "-") != 0)
|
|
fclose(file);
|
|
|
|
Lex::Linknames* linknames = lexer.get_and_clear_linknames();
|
|
if (linknames != NULL)
|
|
{
|
|
if (!::gogo->current_file_imported_unsafe())
|
|
{
|
|
for (Lex::Linknames::const_iterator p = linknames->begin();
|
|
p != linknames->end();
|
|
++p)
|
|
go_error_at(p->second.loc,
|
|
("%<//go:linkname%> only allowed in Go files that "
|
|
"import \"unsafe\""));
|
|
}
|
|
all_linknames.insert(linknames->begin(), linknames->end());
|
|
}
|
|
}
|
|
|
|
::gogo->clear_file_scope();
|
|
|
|
// If the global predeclared names are referenced but not defined,
|
|
// define them now.
|
|
::gogo->define_global_names();
|
|
|
|
// Apply any go:linkname directives.
|
|
for (Lex::Linknames::const_iterator p = all_linknames.begin();
|
|
p != all_linknames.end();
|
|
++p)
|
|
::gogo->add_linkname(p->first, p->second.is_exported, p->second.ext_name,
|
|
p->second.loc);
|
|
|
|
// Finalize method lists and build stub methods for named types.
|
|
::gogo->finalize_methods();
|
|
|
|
// Check that functions have a terminating statement.
|
|
::gogo->check_return_statements();
|
|
|
|
// Now that we have seen all the names, lower the parse tree into a
|
|
// form which is easier to use.
|
|
::gogo->lower_parse_tree();
|
|
|
|
// At this point we have handled all inline functions, so we no
|
|
// longer need the linemap.
|
|
::gogo->linemap()->stop();
|
|
|
|
// Create function descriptors as needed.
|
|
::gogo->create_function_descriptors();
|
|
|
|
// Now that we have seen all the names, verify that types are
|
|
// correct.
|
|
::gogo->verify_types();
|
|
|
|
// Work out types of unspecified constants and variables.
|
|
::gogo->determine_types();
|
|
|
|
// Check types and issue errors as appropriate.
|
|
::gogo->check_types();
|
|
|
|
if (only_check_syntax)
|
|
return;
|
|
|
|
// Do simple deadcode elimination.
|
|
::gogo->remove_deadcode();
|
|
|
|
// Make implicit type conversions explicit.
|
|
::gogo->add_conversions();
|
|
|
|
// Analyze the program flow for escape information.
|
|
::gogo->analyze_escape();
|
|
|
|
// Export global identifiers as appropriate.
|
|
::gogo->do_exports();
|
|
|
|
// Use temporary variables to force order of evaluation.
|
|
::gogo->order_evaluations();
|
|
|
|
// Turn short-cut operators (&&, ||) into explicit if statements.
|
|
::gogo->remove_shortcuts();
|
|
|
|
// Convert named types to backend representation.
|
|
::gogo->convert_named_types();
|
|
|
|
// Build thunks for functions which call recover.
|
|
::gogo->build_recover_thunks();
|
|
|
|
// Convert complicated go and defer statements into simpler ones.
|
|
::gogo->simplify_thunk_statements();
|
|
|
|
// Write out queued up functions for hash and comparison of types.
|
|
::gogo->write_specific_type_functions();
|
|
|
|
// Add write barriers.
|
|
::gogo->add_write_barriers();
|
|
|
|
// Flatten the parse tree.
|
|
::gogo->flatten();
|
|
|
|
// Reclaim memory of escape analysis Nodes.
|
|
::gogo->reclaim_escape_nodes();
|
|
|
|
// Dump ast, use filename[0] as the base name
|
|
::gogo->dump_ast(filenames[0]);
|
|
}
|
|
|
|
// Write out globals.
|
|
|
|
GO_EXTERN_C
|
|
void
|
|
go_write_globals()
|
|
{
|
|
return ::gogo->write_globals();
|
|
}
|
|
|
|
// Return the global IR structure. This is used by some of the
|
|
// langhooks to pass to other code.
|
|
|
|
Gogo*
|
|
go_get_gogo()
|
|
{
|
|
return ::gogo;
|
|
}
|