pattern.cpp 6.4 KB

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