driver.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "toolchain/driver/driver.h"
  5. #include "common/vlog.h"
  6. #include "llvm/ADT/ArrayRef.h"
  7. #include "llvm/ADT/ScopeExit.h"
  8. #include "llvm/ADT/StringExtras.h"
  9. #include "llvm/ADT/StringRef.h"
  10. #include "llvm/ADT/StringSwitch.h"
  11. #include "llvm/IR/LLVMContext.h"
  12. #include "llvm/Support/Format.h"
  13. #include "toolchain/codegen/codegen.h"
  14. #include "toolchain/diagnostics/diagnostic_emitter.h"
  15. #include "toolchain/diagnostics/sorting_diagnostic_consumer.h"
  16. #include "toolchain/lexer/tokenized_buffer.h"
  17. #include "toolchain/lowering/lower_to_llvm.h"
  18. #include "toolchain/parser/parse_tree.h"
  19. #include "toolchain/semantics/semantics_ir.h"
  20. #include "toolchain/semantics/semantics_ir_formatter.h"
  21. #include "toolchain/source/source_buffer.h"
  22. namespace Carbon {
  23. namespace {
  24. enum class Subcommand {
  25. #define CARBON_SUBCOMMAND(Name, ...) Name,
  26. #include "toolchain/driver/flags.def"
  27. Unknown,
  28. };
  29. auto GetSubcommand(llvm::StringRef name) -> Subcommand {
  30. return llvm::StringSwitch<Subcommand>(name)
  31. #define CARBON_SUBCOMMAND(Name, Spelling, ...) .Case(Spelling, Subcommand::Name)
  32. #include "toolchain/driver/flags.def"
  33. .Default(Subcommand::Unknown);
  34. }
  35. } // namespace
  36. auto Driver::RunFullCommand(llvm::ArrayRef<llvm::StringRef> args) -> bool {
  37. StreamDiagnosticConsumer stream_consumer(error_stream_);
  38. DiagnosticConsumer* consumer = &stream_consumer;
  39. std::unique_ptr<SortingDiagnosticConsumer> sorting_consumer;
  40. // TODO: Figure out a command-line support library, this is temporary.
  41. if (!args.empty() && args[0] == "-v") {
  42. args = args.drop_front();
  43. // Note this implies streamed output in order to interleave.
  44. vlog_stream_ = &error_stream_;
  45. } else if (!args.empty() && args[0] == "--print-errors=streamed") {
  46. args = args.drop_front();
  47. } else {
  48. sorting_consumer = std::make_unique<SortingDiagnosticConsumer>(*consumer);
  49. consumer = sorting_consumer.get();
  50. }
  51. if (args.empty()) {
  52. error_stream_ << "ERROR: No subcommand specified.\n";
  53. return false;
  54. }
  55. llvm::StringRef subcommand_text = args[0];
  56. args = args.drop_front();
  57. switch (GetSubcommand(subcommand_text)) {
  58. case Subcommand::Unknown:
  59. error_stream_ << "ERROR: Unknown subcommand '" << subcommand_text
  60. << "'.\n";
  61. return false;
  62. #define CARBON_SUBCOMMAND(Name, ...) \
  63. case Subcommand::Name: \
  64. return Run##Name##Subcommand(*consumer, args);
  65. #include "toolchain/driver/flags.def"
  66. }
  67. llvm_unreachable("All subcommands handled!");
  68. }
  69. auto Driver::RunHelpSubcommand(DiagnosticConsumer& /*consumer*/,
  70. llvm::ArrayRef<llvm::StringRef> args) -> bool {
  71. // TODO: We should support getting detailed help on a subcommand by looking
  72. // for it as a positional parameter here.
  73. if (!args.empty()) {
  74. ReportExtraArgs("help", args);
  75. return false;
  76. }
  77. output_stream_ << "List of subcommands:\n\n";
  78. constexpr llvm::StringLiteral SubcommandsAndHelp[][2] = {
  79. #define CARBON_SUBCOMMAND(Name, Spelling, HelpText) {Spelling, HelpText},
  80. #include "toolchain/driver/flags.def"
  81. };
  82. int max_subcommand_width = 0;
  83. for (const auto* subcommand_and_help : SubcommandsAndHelp) {
  84. max_subcommand_width = std::max(
  85. max_subcommand_width, static_cast<int>(subcommand_and_help[0].size()));
  86. }
  87. for (const auto* subcommand_and_help : SubcommandsAndHelp) {
  88. llvm::StringRef subcommand_text = subcommand_and_help[0];
  89. // TODO: We should wrap this to the number of columns left after the
  90. // subcommand on the terminal, and using a hanging indent.
  91. llvm::StringRef help_text = subcommand_and_help[1];
  92. output_stream_ << " "
  93. << llvm::left_justify(subcommand_text, max_subcommand_width)
  94. << " - " << help_text << "\n";
  95. }
  96. output_stream_ << "\n";
  97. return true;
  98. }
  99. enum class DumpMode {
  100. TokenizedBuffer,
  101. ParseTree,
  102. RawSemanticsIR,
  103. SemanticsIR,
  104. LLVMIR,
  105. Assembly,
  106. ObjectCode,
  107. Unknown
  108. };
  109. auto Driver::RunDumpSubcommand(DiagnosticConsumer& consumer,
  110. llvm::ArrayRef<llvm::StringRef> args) -> bool {
  111. if (args.empty()) {
  112. error_stream_ << "ERROR: No dump mode specified.\n";
  113. return false;
  114. }
  115. auto dump_mode = llvm::StringSwitch<DumpMode>(args.front())
  116. .Case("tokens", DumpMode::TokenizedBuffer)
  117. .Case("parse-tree", DumpMode::ParseTree)
  118. .Case("raw-semantics-ir", DumpMode::RawSemanticsIR)
  119. .Case("semantics-ir", DumpMode::SemanticsIR)
  120. .Case("llvm-ir", DumpMode::LLVMIR)
  121. .Case("assembly", DumpMode::Assembly)
  122. .Case("objcode", DumpMode::ObjectCode)
  123. .Default(DumpMode::Unknown);
  124. if (dump_mode == DumpMode::Unknown) {
  125. error_stream_ << "ERROR: Dump mode should be one of tokens, parse-tree, "
  126. "semantics-ir, llvm-ir, assembly, or objcode.\n";
  127. return false;
  128. }
  129. args = args.drop_front();
  130. bool parse_tree_preorder = false;
  131. if (dump_mode == DumpMode::ParseTree && !args.empty() &&
  132. args.front() == "--preorder") {
  133. args = args.drop_front();
  134. parse_tree_preorder = true;
  135. }
  136. bool semantics_ir_include_raw = false;
  137. if (dump_mode == DumpMode::SemanticsIR && !args.empty() &&
  138. args.front() == "--include_raw") {
  139. args = args.drop_front();
  140. semantics_ir_include_raw = true;
  141. }
  142. bool semantics_ir_include_builtins = false;
  143. if ((dump_mode == DumpMode::RawSemanticsIR || semantics_ir_include_raw) &&
  144. !args.empty() && args.front() == "--include_builtins") {
  145. args = args.drop_front();
  146. semantics_ir_include_builtins = true;
  147. }
  148. llvm::StringRef target_triple;
  149. if (dump_mode == DumpMode::Assembly && !args.empty() &&
  150. args.front().starts_with("--target_triple=")) {
  151. target_triple = args.front().split("=").second;
  152. args = args.drop_front();
  153. }
  154. llvm::StringRef output_file;
  155. if (dump_mode == DumpMode::ObjectCode) {
  156. while (!args.empty()) {
  157. if (args.front().starts_with("--target_triple=")) {
  158. target_triple = args.front().split("=").second;
  159. args = args.drop_front();
  160. } else if (args.front().starts_with("--output_file=")) {
  161. output_file = args.front().split("=").second;
  162. args = args.drop_front();
  163. } else {
  164. break;
  165. }
  166. }
  167. if (output_file.empty()) {
  168. error_stream_ << "ERROR: Must provide an output file.\n";
  169. return false;
  170. }
  171. }
  172. if (args.empty()) {
  173. error_stream_ << "ERROR: No input file specified.\n";
  174. return false;
  175. }
  176. llvm::StringRef input_file_name = args.front();
  177. args = args.drop_front();
  178. if (!args.empty()) {
  179. ReportExtraArgs("dump", args);
  180. return false;
  181. }
  182. CARBON_VLOG() << "*** SourceBuffer::CreateFromFile ***\n";
  183. auto source = SourceBuffer::CreateFromFile(fs_, input_file_name);
  184. CARBON_VLOG() << "*** SourceBuffer::CreateFromFile done ***\n";
  185. // Require flushing the consumer before the source buffer is destroyed,
  186. // because diagnostics may reference the buffer.
  187. auto flush = llvm::make_scope_exit([&]() { consumer.Flush(); });
  188. if (!source.ok()) {
  189. error_stream_ << "ERROR: Unable to open input source file: "
  190. << source.error();
  191. return false;
  192. }
  193. bool has_errors = false;
  194. CARBON_VLOG() << "*** TokenizedBuffer::Lex ***\n";
  195. auto tokenized_source = TokenizedBuffer::Lex(*source, consumer);
  196. has_errors |= tokenized_source.has_errors();
  197. CARBON_VLOG() << "*** TokenizedBuffer::Lex done ***\n";
  198. if (dump_mode == DumpMode::TokenizedBuffer) {
  199. CARBON_VLOG() << "Finishing output.";
  200. output_stream_ << tokenized_source;
  201. return !has_errors;
  202. }
  203. CARBON_VLOG() << "tokenized_buffer: " << tokenized_source;
  204. CARBON_VLOG() << "*** ParseTree::Parse ***\n";
  205. auto parse_tree = ParseTree::Parse(tokenized_source, consumer, vlog_stream_);
  206. has_errors |= parse_tree.has_errors();
  207. CARBON_VLOG() << "*** ParseTree::Parse done ***\n";
  208. if (dump_mode == DumpMode::ParseTree) {
  209. parse_tree.Print(output_stream_, parse_tree_preorder);
  210. return !has_errors;
  211. }
  212. CARBON_VLOG() << "parse_tree: " << parse_tree;
  213. const SemanticsIR builtin_ir = SemanticsIR::MakeBuiltinIR();
  214. CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree ***\n";
  215. const SemanticsIR semantics_ir = SemanticsIR::MakeFromParseTree(
  216. builtin_ir, tokenized_source, parse_tree, consumer, vlog_stream_);
  217. has_errors |= semantics_ir.has_errors();
  218. CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree done ***\n";
  219. if (dump_mode == DumpMode::RawSemanticsIR) {
  220. semantics_ir.Print(output_stream_, semantics_ir_include_builtins);
  221. return !has_errors;
  222. }
  223. if (dump_mode == DumpMode::SemanticsIR) {
  224. if (semantics_ir_include_raw) {
  225. semantics_ir.Print(output_stream_, semantics_ir_include_builtins);
  226. output_stream_ << "\n";
  227. }
  228. FormatSemanticsIR(tokenized_source, parse_tree, semantics_ir,
  229. output_stream_);
  230. return !has_errors;
  231. }
  232. CARBON_VLOG() << "semantics_ir: " << semantics_ir;
  233. // Unlike previous steps, errors block further progress.
  234. if (has_errors) {
  235. CARBON_VLOG() << "Unable to dump llvm-ir due to prior errors.";
  236. return false;
  237. }
  238. CARBON_VLOG() << "*** LowerToLLVM ***\n";
  239. llvm::LLVMContext llvm_context;
  240. const std::unique_ptr<llvm::Module> module =
  241. LowerToLLVM(llvm_context, input_file_name, semantics_ir, vlog_stream_);
  242. CARBON_VLOG() << "*** LowerToLLVM done ***\n";
  243. if (dump_mode == DumpMode::LLVMIR) {
  244. module->print(output_stream_, /*AAW=*/nullptr,
  245. /*ShouldPreserveUseListOrder=*/true);
  246. return !has_errors;
  247. }
  248. if (vlog_stream_) {
  249. CARBON_VLOG() << "module: ";
  250. module->print(*vlog_stream_, /*AAW=*/nullptr,
  251. /*ShouldPreserveUseListOrder=*/false,
  252. /*IsForDebug=*/true);
  253. }
  254. if (dump_mode == DumpMode::Assembly) {
  255. CodeGen codegen(*module, target_triple, error_stream_, output_stream_);
  256. has_errors |= !codegen.PrintAssembly();
  257. return !has_errors;
  258. }
  259. if (dump_mode == DumpMode::ObjectCode) {
  260. std::error_code ec;
  261. llvm::raw_fd_ostream dest(output_file, ec, llvm::sys::fs::OF_None);
  262. if (ec) {
  263. error_stream_ << "Error: Could not open file: " << ec.message() << "\n";
  264. return false;
  265. }
  266. CodeGen codegen(*module, target_triple, error_stream_, dest);
  267. has_errors |= !codegen.GenerateObjectCode();
  268. if (!has_errors) {
  269. output_stream_ << "Success: Object file is generated!\n";
  270. }
  271. return !has_errors;
  272. }
  273. llvm_unreachable("should handle all dump modes");
  274. }
  275. auto Driver::ReportExtraArgs(llvm::StringRef subcommand_text,
  276. llvm::ArrayRef<llvm::StringRef> args) -> void {
  277. error_stream_ << "ERROR: Unexpected additional arguments to the '"
  278. << subcommand_text << "' subcommand:";
  279. for (auto arg : args) {
  280. error_stream_ << " " << arg;
  281. }
  282. error_stream_ << "\n";
  283. }
  284. } // namespace Carbon