driver.cpp 7.4 KB

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