driver.cpp 6.0 KB

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