pattern.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  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 "explorer/ast/pattern.h"
  5. #include <string>
  6. #include "common/ostream.h"
  7. #include "explorer/ast/expression.h"
  8. #include "explorer/common/arena.h"
  9. #include "explorer/common/error_builders.h"
  10. #include "llvm/ADT/StringExtras.h"
  11. #include "llvm/Support/Casting.h"
  12. namespace Carbon {
  13. using llvm::cast;
  14. Pattern::~Pattern() = default;
  15. void Pattern::Print(llvm::raw_ostream& out) const {
  16. switch (kind()) {
  17. case PatternKind::AutoPattern:
  18. out << "auto";
  19. break;
  20. case PatternKind::BindingPattern: {
  21. const auto& binding = cast<BindingPattern>(*this);
  22. out << binding.name() << ": " << binding.type();
  23. break;
  24. }
  25. case PatternKind::GenericBinding: {
  26. const auto& binding = cast<GenericBinding>(*this);
  27. out << binding.name() << ":! " << binding.type();
  28. break;
  29. }
  30. case PatternKind::TuplePattern: {
  31. const auto& tuple = cast<TuplePattern>(*this);
  32. out << "(";
  33. llvm::ListSeparator sep;
  34. for (Nonnull<const Pattern*> field : tuple.fields()) {
  35. out << sep << *field;
  36. }
  37. out << ")";
  38. break;
  39. }
  40. case PatternKind::AlternativePattern: {
  41. const auto& alternative = cast<AlternativePattern>(*this);
  42. out << alternative.choice_type() << "." << alternative.alternative_name()
  43. << alternative.arguments();
  44. break;
  45. }
  46. case PatternKind::ExpressionPattern:
  47. out << cast<ExpressionPattern>(*this).expression();
  48. break;
  49. case PatternKind::VarPattern:
  50. out << "var" << cast<VarPattern>(*this).pattern();
  51. break;
  52. case PatternKind::AddrPattern:
  53. out << "addr" << cast<AddrPattern>(*this).binding();
  54. break;
  55. }
  56. }
  57. void Pattern::PrintID(llvm::raw_ostream& out) const {
  58. switch (kind()) {
  59. case PatternKind::AutoPattern:
  60. out << "auto";
  61. break;
  62. case PatternKind::BindingPattern: {
  63. const auto& binding = cast<BindingPattern>(*this);
  64. out << binding.name();
  65. break;
  66. }
  67. case PatternKind::GenericBinding: {
  68. const auto& binding = cast<GenericBinding>(*this);
  69. out << binding.name();
  70. break;
  71. }
  72. case PatternKind::TuplePattern: {
  73. out << "(...)";
  74. break;
  75. }
  76. case PatternKind::AlternativePattern: {
  77. const auto& alternative = cast<AlternativePattern>(*this);
  78. out << alternative.choice_type() << "." << alternative.alternative_name()
  79. << "(...)";
  80. break;
  81. }
  82. case PatternKind::VarPattern:
  83. out << "var ...";
  84. break;
  85. case PatternKind::AddrPattern:
  86. out << "addr ...";
  87. break;
  88. case PatternKind::ExpressionPattern:
  89. out << "...";
  90. break;
  91. }
  92. }
  93. auto VisitNestedPatterns(const Pattern& pattern,
  94. llvm::function_ref<bool(const Pattern&)> visitor)
  95. -> bool {
  96. if (!visitor(pattern)) {
  97. return false;
  98. }
  99. switch (pattern.kind()) {
  100. case PatternKind::TuplePattern:
  101. for (const Pattern* field : cast<TuplePattern>(pattern).fields()) {
  102. if (!VisitNestedPatterns(*field, visitor)) {
  103. return false;
  104. }
  105. }
  106. return true;
  107. case PatternKind::AlternativePattern:
  108. return VisitNestedPatterns(cast<AlternativePattern>(pattern).arguments(),
  109. visitor);
  110. case PatternKind::VarPattern:
  111. return VisitNestedPatterns(cast<VarPattern>(pattern).pattern(), visitor);
  112. case PatternKind::AddrPattern:
  113. return VisitNestedPatterns(cast<AddrPattern>(pattern).binding(), visitor);
  114. case PatternKind::BindingPattern:
  115. case PatternKind::AutoPattern:
  116. case PatternKind::ExpressionPattern:
  117. case PatternKind::GenericBinding:
  118. return true;
  119. }
  120. }
  121. auto PatternFromParenContents(Nonnull<Arena*> arena, SourceLocation source_loc,
  122. const ParenContents<Pattern>& paren_contents)
  123. -> Nonnull<Pattern*> {
  124. std::optional<Nonnull<Pattern*>> single_term = paren_contents.SingleTerm();
  125. if (single_term.has_value()) {
  126. return *single_term;
  127. } else {
  128. return TuplePatternFromParenContents(arena, source_loc, paren_contents);
  129. }
  130. }
  131. auto TuplePatternFromParenContents(Nonnull<Arena*> arena,
  132. SourceLocation source_loc,
  133. const ParenContents<Pattern>& paren_contents)
  134. -> Nonnull<TuplePattern*> {
  135. return arena->New<TuplePattern>(source_loc, paren_contents.elements);
  136. }
  137. // Used by AlternativePattern for constructor initialization. Produces a helpful
  138. // error for incorrect expressions, rather than letting a default cast error
  139. // apply.
  140. auto AlternativePattern::RequireSimpleMemberAccess(
  141. Nonnull<Expression*> alternative)
  142. -> ErrorOr<Nonnull<SimpleMemberAccessExpression*>> {
  143. if (alternative->kind() != ExpressionKind::SimpleMemberAccessExpression) {
  144. return ProgramError(alternative->source_loc())
  145. << "Alternative pattern must have the form of a field access.";
  146. }
  147. return &cast<SimpleMemberAccessExpression>(*alternative);
  148. }
  149. auto ParenExpressionToParenPattern(Nonnull<Arena*> arena,
  150. const ParenContents<Expression>& contents)
  151. -> ParenContents<Pattern> {
  152. ParenContents<Pattern> result = {
  153. .elements = {}, .has_trailing_comma = contents.has_trailing_comma};
  154. for (const auto& element : contents.elements) {
  155. result.elements.push_back(arena->New<ExpressionPattern>(element));
  156. }
  157. return result;
  158. }
  159. } // namespace Carbon