| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "toolchain/driver/driver.h"
- #include "common/vlog.h"
- #include "llvm/ADT/ArrayRef.h"
- #include "llvm/ADT/ScopeExit.h"
- #include "llvm/ADT/StringExtras.h"
- #include "llvm/ADT/StringRef.h"
- #include "llvm/ADT/StringSwitch.h"
- #include "llvm/IR/LLVMContext.h"
- #include "llvm/Support/Format.h"
- #include "toolchain/codegen/codegen.h"
- #include "toolchain/diagnostics/diagnostic_emitter.h"
- #include "toolchain/diagnostics/sorting_diagnostic_consumer.h"
- #include "toolchain/lexer/tokenized_buffer.h"
- #include "toolchain/lowering/lower_to_llvm.h"
- #include "toolchain/parser/parse_tree.h"
- #include "toolchain/semantics/semantics_ir.h"
- #include "toolchain/source/source_buffer.h"
- namespace Carbon {
- namespace {
- enum class Subcommand {
- #define CARBON_SUBCOMMAND(Name, ...) Name,
- #include "toolchain/driver/flags.def"
- Unknown,
- };
- auto GetSubcommand(llvm::StringRef name) -> Subcommand {
- return llvm::StringSwitch<Subcommand>(name)
- #define CARBON_SUBCOMMAND(Name, Spelling, ...) .Case(Spelling, Subcommand::Name)
- #include "toolchain/driver/flags.def"
- .Default(Subcommand::Unknown);
- }
- } // namespace
- auto Driver::RunFullCommand(llvm::ArrayRef<llvm::StringRef> args) -> bool {
- StreamDiagnosticConsumer stream_consumer(error_stream_);
- DiagnosticConsumer* consumer = &stream_consumer;
- std::unique_ptr<SortingDiagnosticConsumer> sorting_consumer;
- // TODO: Figure out a command-line support library, this is temporary.
- if (!args.empty() && args[0] == "-v") {
- args = args.drop_front();
- // Note this implies streamed output in order to interleave.
- vlog_stream_ = &error_stream_;
- } else if (!args.empty() && args[0] == "--print-errors=streamed") {
- args = args.drop_front();
- } else {
- sorting_consumer = std::make_unique<SortingDiagnosticConsumer>(*consumer);
- consumer = sorting_consumer.get();
- }
- if (args.empty()) {
- error_stream_ << "ERROR: No subcommand specified.\n";
- return false;
- }
- llvm::StringRef subcommand_text = args[0];
- args = args.drop_front();
- switch (GetSubcommand(subcommand_text)) {
- case Subcommand::Unknown:
- error_stream_ << "ERROR: Unknown subcommand '" << subcommand_text
- << "'.\n";
- return false;
- #define CARBON_SUBCOMMAND(Name, ...) \
- case Subcommand::Name: \
- return Run##Name##Subcommand(*consumer, args);
- #include "toolchain/driver/flags.def"
- }
- llvm_unreachable("All subcommands handled!");
- }
- auto Driver::RunHelpSubcommand(DiagnosticConsumer& /*consumer*/,
- llvm::ArrayRef<llvm::StringRef> args) -> bool {
- // TODO: We should support getting detailed help on a subcommand by looking
- // for it as a positional parameter here.
- if (!args.empty()) {
- ReportExtraArgs("help", args);
- return false;
- }
- output_stream_ << "List of subcommands:\n\n";
- constexpr llvm::StringLiteral SubcommandsAndHelp[][2] = {
- #define CARBON_SUBCOMMAND(Name, Spelling, HelpText) {Spelling, HelpText},
- #include "toolchain/driver/flags.def"
- };
- int max_subcommand_width = 0;
- for (const auto* subcommand_and_help : SubcommandsAndHelp) {
- max_subcommand_width = std::max(
- max_subcommand_width, static_cast<int>(subcommand_and_help[0].size()));
- }
- for (const auto* subcommand_and_help : SubcommandsAndHelp) {
- llvm::StringRef subcommand_text = subcommand_and_help[0];
- // TODO: We should wrap this to the number of columns left after the
- // subcommand on the terminal, and using a hanging indent.
- llvm::StringRef help_text = subcommand_and_help[1];
- output_stream_ << " "
- << llvm::left_justify(subcommand_text, max_subcommand_width)
- << " - " << help_text << "\n";
- }
- output_stream_ << "\n";
- return true;
- }
- enum class DumpMode {
- TokenizedBuffer,
- ParseTree,
- SemanticsIR,
- LLVMIR,
- Assembly,
- ObjectCode,
- Unknown
- };
- auto Driver::RunDumpSubcommand(DiagnosticConsumer& consumer,
- llvm::ArrayRef<llvm::StringRef> args) -> bool {
- if (args.empty()) {
- error_stream_ << "ERROR: No dump mode specified.\n";
- return false;
- }
- auto dump_mode = llvm::StringSwitch<DumpMode>(args.front())
- .Case("tokens", DumpMode::TokenizedBuffer)
- .Case("parse-tree", DumpMode::ParseTree)
- .Case("semantics-ir", DumpMode::SemanticsIR)
- .Case("llvm-ir", DumpMode::LLVMIR)
- .Case("assembly", DumpMode::Assembly)
- .Case("objcode", DumpMode::ObjectCode)
- .Default(DumpMode::Unknown);
- if (dump_mode == DumpMode::Unknown) {
- error_stream_ << "ERROR: Dump mode should be one of tokens, parse-tree, "
- "semantics-ir, llvm-ir, assembly, or objcode.\n";
- return false;
- }
- args = args.drop_front();
- bool parse_tree_preorder = false;
- if (dump_mode == DumpMode::ParseTree && !args.empty() &&
- args.front() == "--preorder") {
- args = args.drop_front();
- parse_tree_preorder = true;
- }
- bool semantics_ir_include_builtins = false;
- if (dump_mode == DumpMode::SemanticsIR && !args.empty() &&
- args.front() == "--include_builtins") {
- args = args.drop_front();
- semantics_ir_include_builtins = true;
- }
- llvm::StringRef target_triple;
- if (dump_mode == DumpMode::Assembly && !args.empty() &&
- args.front().starts_with("--target_triple=")) {
- target_triple = args.front().split("=").second;
- args = args.drop_front();
- }
- llvm::StringRef output_file;
- if (dump_mode == DumpMode::ObjectCode) {
- while (!args.empty()) {
- if (args.front().starts_with("--target_triple=")) {
- target_triple = args.front().split("=").second;
- args = args.drop_front();
- } else if (args.front().starts_with("--output_file=")) {
- output_file = args.front().split("=").second;
- args = args.drop_front();
- } else {
- break;
- }
- }
- if (output_file.empty()) {
- error_stream_ << "ERROR: Must provide an output file.\n";
- return false;
- }
- }
- if (args.empty()) {
- error_stream_ << "ERROR: No input file specified.\n";
- return false;
- }
- llvm::StringRef input_file_name = args.front();
- args = args.drop_front();
- if (!args.empty()) {
- ReportExtraArgs("dump", args);
- return false;
- }
- CARBON_VLOG() << "*** SourceBuffer::CreateFromFile ***\n";
- auto source = SourceBuffer::CreateFromFile(fs_, input_file_name);
- CARBON_VLOG() << "*** SourceBuffer::CreateFromFile done ***\n";
- // Require flushing the consumer before the source buffer is destroyed,
- // because diagnostics may reference the buffer.
- auto flush = llvm::make_scope_exit([&]() { consumer.Flush(); });
- if (!source.ok()) {
- error_stream_ << "ERROR: Unable to open input source file: "
- << source.error();
- return false;
- }
- bool has_errors = false;
- CARBON_VLOG() << "*** TokenizedBuffer::Lex ***\n";
- auto tokenized_source = TokenizedBuffer::Lex(*source, consumer);
- has_errors |= tokenized_source.has_errors();
- CARBON_VLOG() << "*** TokenizedBuffer::Lex done ***\n";
- if (dump_mode == DumpMode::TokenizedBuffer) {
- CARBON_VLOG() << "Finishing output.";
- output_stream_ << tokenized_source;
- return !has_errors;
- }
- CARBON_VLOG() << "tokenized_buffer: " << tokenized_source;
- CARBON_VLOG() << "*** ParseTree::Parse ***\n";
- auto parse_tree = ParseTree::Parse(tokenized_source, consumer, vlog_stream_);
- has_errors |= parse_tree.has_errors();
- CARBON_VLOG() << "*** ParseTree::Parse done ***\n";
- if (dump_mode == DumpMode::ParseTree) {
- parse_tree.Print(output_stream_, parse_tree_preorder);
- return !has_errors;
- }
- CARBON_VLOG() << "parse_tree: " << parse_tree;
- const SemanticsIR builtin_ir = SemanticsIR::MakeBuiltinIR();
- CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree ***\n";
- const SemanticsIR semantics_ir = SemanticsIR::MakeFromParseTree(
- builtin_ir, tokenized_source, parse_tree, consumer, vlog_stream_);
- has_errors |= semantics_ir.has_errors();
- CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree done ***\n";
- if (dump_mode == DumpMode::SemanticsIR) {
- semantics_ir.Print(output_stream_, semantics_ir_include_builtins);
- return !has_errors;
- }
- CARBON_VLOG() << "semantics_ir: " << semantics_ir;
- // Unlike previous steps, errors block further progress.
- if (has_errors) {
- CARBON_VLOG() << "Unable to dump llvm-ir due to prior errors.";
- return false;
- }
- CARBON_VLOG() << "*** LowerToLLVM ***\n";
- llvm::LLVMContext llvm_context;
- const std::unique_ptr<llvm::Module> module =
- LowerToLLVM(llvm_context, input_file_name, semantics_ir, vlog_stream_);
- CARBON_VLOG() << "*** LowerToLLVM done ***\n";
- if (dump_mode == DumpMode::LLVMIR) {
- module->print(output_stream_, /*AAW=*/nullptr,
- /*ShouldPreserveUseListOrder=*/true);
- return !has_errors;
- }
- if (vlog_stream_) {
- CARBON_VLOG() << "module: ";
- module->print(*vlog_stream_, /*AAW=*/nullptr,
- /*ShouldPreserveUseListOrder=*/false,
- /*IsForDebug=*/true);
- }
- if (dump_mode == DumpMode::Assembly) {
- CodeGen codegen(*module, target_triple, error_stream_, output_stream_);
- has_errors |= !codegen.PrintAssembly();
- return !has_errors;
- }
- if (dump_mode == DumpMode::ObjectCode) {
- std::error_code ec;
- llvm::raw_fd_ostream dest(output_file, ec, llvm::sys::fs::OF_None);
- if (ec) {
- error_stream_ << "Error: Could not open file: " << ec.message() << "\n";
- return false;
- }
- CodeGen codegen(*module, target_triple, error_stream_, dest);
- has_errors |= !codegen.GenerateObjectCode();
- if (!has_errors) {
- output_stream_ << "Success: Object file is generated!\n";
- }
- return !has_errors;
- }
- llvm_unreachable("should handle all dump modes");
- }
- auto Driver::ReportExtraArgs(llvm::StringRef subcommand_text,
- llvm::ArrayRef<llvm::StringRef> args) -> void {
- error_stream_ << "ERROR: Unexpected additional arguments to the '"
- << subcommand_text << "' subcommand:";
- for (auto arg : args) {
- error_stream_ << " " << arg;
- }
- error_stream_ << "\n";
- }
- } // namespace Carbon
|