driver.cpp 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  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/StringExtras.h"
  8. #include "llvm/ADT/StringRef.h"
  9. #include "llvm/ADT/StringSwitch.h"
  10. #include "llvm/IR/LLVMContext.h"
  11. #include "llvm/Support/Error.h"
  12. #include "llvm/Support/ErrorHandling.h"
  13. #include "llvm/Support/Format.h"
  14. #include "toolchain/diagnostics/sorting_diagnostic_consumer.h"
  15. #include "toolchain/lexer/tokenized_buffer.h"
  16. #include "toolchain/lowering/lower_to_llvm.h"
  17. #include "toolchain/parser/parse_tree.h"
  18. #include "toolchain/semantics/semantics_ir.h"
  19. #include "toolchain/source/source_buffer.h"
  20. namespace Carbon {
  21. namespace {
  22. enum class Subcommand {
  23. #define CARBON_SUBCOMMAND(Name, ...) Name,
  24. #include "toolchain/driver/flags.def"
  25. Unknown,
  26. };
  27. auto GetSubcommand(llvm::StringRef name) -> Subcommand {
  28. return llvm::StringSwitch<Subcommand>(name)
  29. #define CARBON_SUBCOMMAND(Name, Spelling, ...) .Case(Spelling, Subcommand::Name)
  30. #include "toolchain/driver/flags.def"
  31. .Default(Subcommand::Unknown);
  32. }
  33. } // namespace
  34. auto Driver::RunFullCommand(llvm::ArrayRef<llvm::StringRef> args) -> bool {
  35. DiagnosticConsumer* consumer = &ConsoleDiagnosticConsumer();
  36. std::unique_ptr<SortingDiagnosticConsumer> sorting_consumer;
  37. // TODO: Figure out a command-line support library, this is temporary.
  38. if (!args.empty() && args[0] == "-v") {
  39. args = args.drop_front();
  40. // Note this implies streamed output in order to interleave.
  41. vlog_stream_ = &error_stream_;
  42. } else if (!args.empty() && args[0] == "--print-errors=streamed") {
  43. args = args.drop_front();
  44. } else {
  45. sorting_consumer = std::make_unique<SortingDiagnosticConsumer>(*consumer);
  46. consumer = sorting_consumer.get();
  47. }
  48. if (args.empty()) {
  49. error_stream_ << "ERROR: No subcommand specified.\n";
  50. return false;
  51. }
  52. llvm::StringRef subcommand_text = args[0];
  53. args = args.drop_front();
  54. switch (GetSubcommand(subcommand_text)) {
  55. case Subcommand::Unknown:
  56. error_stream_ << "ERROR: Unknown subcommand '" << subcommand_text
  57. << "'.\n";
  58. return false;
  59. #define CARBON_SUBCOMMAND(Name, ...) \
  60. case Subcommand::Name: \
  61. return Run##Name##Subcommand(*consumer, args);
  62. #include "toolchain/driver/flags.def"
  63. }
  64. llvm_unreachable("All subcommands handled!");
  65. }
  66. auto Driver::RunHelpSubcommand(DiagnosticConsumer& /*consumer*/,
  67. llvm::ArrayRef<llvm::StringRef> args) -> bool {
  68. // TODO: We should support getting detailed help on a subcommand by looking
  69. // for it as a positional parameter here.
  70. if (!args.empty()) {
  71. ReportExtraArgs("help", args);
  72. return false;
  73. }
  74. output_stream_ << "List of subcommands:\n\n";
  75. constexpr llvm::StringLiteral SubcommandsAndHelp[][2] = {
  76. #define CARBON_SUBCOMMAND(Name, Spelling, HelpText) {Spelling, HelpText},
  77. #include "toolchain/driver/flags.def"
  78. };
  79. int max_subcommand_width = 0;
  80. for (const auto* subcommand_and_help : SubcommandsAndHelp) {
  81. max_subcommand_width = std::max(
  82. max_subcommand_width, static_cast<int>(subcommand_and_help[0].size()));
  83. }
  84. for (const auto* subcommand_and_help : SubcommandsAndHelp) {
  85. llvm::StringRef subcommand_text = subcommand_and_help[0];
  86. // TODO: We should wrap this to the number of columns left after the
  87. // subcommand on the terminal, and using a hanging indent.
  88. llvm::StringRef help_text = subcommand_and_help[1];
  89. output_stream_ << " "
  90. << llvm::left_justify(subcommand_text, max_subcommand_width)
  91. << " - " << help_text << "\n";
  92. }
  93. output_stream_ << "\n";
  94. return true;
  95. }
  96. enum class DumpMode {
  97. TokenizedBuffer,
  98. ParseTree,
  99. SemanticsIR,
  100. LLVMIR,
  101. Unknown
  102. };
  103. auto Driver::RunDumpSubcommand(DiagnosticConsumer& consumer,
  104. llvm::ArrayRef<llvm::StringRef> args) -> bool {
  105. if (args.empty()) {
  106. error_stream_ << "ERROR: No dump mode specified.\n";
  107. return false;
  108. }
  109. auto dump_mode = llvm::StringSwitch<DumpMode>(args.front())
  110. .Case("tokens", DumpMode::TokenizedBuffer)
  111. .Case("parse-tree", DumpMode::ParseTree)
  112. .Case("semantics-ir", DumpMode::SemanticsIR)
  113. .Case("llvm-ir", DumpMode::LLVMIR)
  114. .Default(DumpMode::Unknown);
  115. if (dump_mode == DumpMode::Unknown) {
  116. error_stream_ << "ERROR: Dump mode should be one of tokens, parse-tree, or "
  117. "semantics-ir.\n";
  118. return false;
  119. }
  120. args = args.drop_front();
  121. bool parse_tree_preorder = false;
  122. if (dump_mode == DumpMode::ParseTree && !args.empty() &&
  123. args.front() == "--preorder") {
  124. args = args.drop_front();
  125. parse_tree_preorder = true;
  126. }
  127. bool semantics_ir_include_builtins = false;
  128. if (dump_mode == DumpMode::SemanticsIR && !args.empty() &&
  129. args.front() == "--include_builtins") {
  130. args = args.drop_front();
  131. semantics_ir_include_builtins = true;
  132. }
  133. if (args.empty()) {
  134. error_stream_ << "ERROR: No input file specified.\n";
  135. return false;
  136. }
  137. llvm::StringRef input_file_name = args.front();
  138. args = args.drop_front();
  139. if (!args.empty()) {
  140. ReportExtraArgs("dump", args);
  141. return false;
  142. }
  143. CARBON_VLOG() << "*** SourceBuffer::CreateFromFile ***\n";
  144. auto source = SourceBuffer::CreateFromFile(input_file_name);
  145. CARBON_VLOG() << "*** SourceBuffer::CreateFromFile done ***\n";
  146. if (!source) {
  147. error_stream_ << "ERROR: Unable to open input source file: ";
  148. llvm::handleAllErrors(source.takeError(),
  149. [&](const llvm::ErrorInfoBase& ei) {
  150. ei.log(error_stream_);
  151. error_stream_ << "\n";
  152. });
  153. return false;
  154. }
  155. bool has_errors = false;
  156. CARBON_VLOG() << "*** TokenizedBuffer::Lex ***\n";
  157. auto tokenized_source = TokenizedBuffer::Lex(*source, consumer);
  158. has_errors |= tokenized_source.has_errors();
  159. CARBON_VLOG() << "*** TokenizedBuffer::Lex done ***\n";
  160. if (dump_mode == DumpMode::TokenizedBuffer) {
  161. CARBON_VLOG() << "Finishing output.";
  162. consumer.Flush();
  163. output_stream_ << tokenized_source;
  164. return !has_errors;
  165. }
  166. CARBON_VLOG() << "tokenized_buffer: " << tokenized_source;
  167. CARBON_VLOG() << "*** ParseTree::Parse ***\n";
  168. auto parse_tree = ParseTree::Parse(tokenized_source, consumer, vlog_stream_);
  169. has_errors |= parse_tree.has_errors();
  170. CARBON_VLOG() << "*** ParseTree::Parse done ***\n";
  171. if (dump_mode == DumpMode::ParseTree) {
  172. consumer.Flush();
  173. parse_tree.Print(output_stream_, parse_tree_preorder);
  174. return !has_errors;
  175. }
  176. CARBON_VLOG() << "parse_tree: " << parse_tree;
  177. const SemanticsIR builtin_ir = SemanticsIR::MakeBuiltinIR();
  178. CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree ***\n";
  179. const SemanticsIR semantics_ir = SemanticsIR::MakeFromParseTree(
  180. builtin_ir, tokenized_source, parse_tree, consumer, vlog_stream_);
  181. has_errors |= semantics_ir.has_errors();
  182. CARBON_VLOG() << "*** SemanticsIR::MakeFromParseTree done ***\n";
  183. if (dump_mode == DumpMode::SemanticsIR) {
  184. consumer.Flush();
  185. semantics_ir.Print(output_stream_, semantics_ir_include_builtins);
  186. return !has_errors;
  187. }
  188. CARBON_VLOG() << "semantics_ir: " << semantics_ir;
  189. // Unlike previous steps, errors block further progress.
  190. if (has_errors) {
  191. CARBON_VLOG() << "Unable to dump llvm-ir due to prior errors.";
  192. return false;
  193. }
  194. CARBON_VLOG() << "*** LowerToLLVM ***\n";
  195. llvm::LLVMContext llvm_context;
  196. const std::unique_ptr<llvm::Module> module =
  197. LowerToLLVM(llvm_context, input_file_name, semantics_ir);
  198. CARBON_VLOG() << "*** LowerToLLVM done ***\n";
  199. if (dump_mode == DumpMode::LLVMIR) {
  200. consumer.Flush();
  201. module->print(output_stream_, /*AAW=*/nullptr,
  202. /*ShouldPreserveUseListOrder=*/true);
  203. return !has_errors;
  204. }
  205. if (vlog_stream_) {
  206. CARBON_VLOG() << "module: ";
  207. module->print(*vlog_stream_, /*AAW=*/nullptr,
  208. /*ShouldPreserveUseListOrder=*/false,
  209. /*IsForDebug=*/true);
  210. }
  211. llvm_unreachable("should handle all dump modes");
  212. }
  213. auto Driver::ReportExtraArgs(llvm::StringRef subcommand_text,
  214. llvm::ArrayRef<llvm::StringRef> args) -> void {
  215. error_stream_ << "ERROR: Unexpected additional arguments to the '"
  216. << subcommand_text << "' subcommand:";
  217. for (auto arg : args) {
  218. error_stream_ << " " << arg;
  219. }
  220. error_stream_ << "\n";
  221. }
  222. } // namespace Carbon