driver.cpp 6.1 KB

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