pattern.cpp 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  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 "executable_semantics/ast/pattern.h"
  5. #include <string>
  6. #include "common/ostream.h"
  7. #include "executable_semantics/ast/expression.h"
  8. #include "executable_semantics/common/arena.h"
  9. #include "executable_semantics/common/error.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::TuplePattern: {
  26. const auto& tuple = cast<TuplePattern>(*this);
  27. out << "(";
  28. llvm::ListSeparator sep;
  29. for (Nonnull<const Pattern*> field : tuple.fields()) {
  30. out << sep << *field;
  31. }
  32. out << ")";
  33. break;
  34. }
  35. case PatternKind::AlternativePattern: {
  36. const auto& alternative = cast<AlternativePattern>(*this);
  37. out << alternative.choice_type() << "." << alternative.alternative_name()
  38. << alternative.arguments();
  39. break;
  40. }
  41. case PatternKind::ExpressionPattern:
  42. out << cast<ExpressionPattern>(*this).expression();
  43. break;
  44. case PatternKind::VarPattern:
  45. out << "var" << cast<VarPattern>(*this).pattern();
  46. break;
  47. }
  48. }
  49. // Equivalent to `GetBindings`, but stores its output in `bindings` instead of
  50. // returning it.
  51. static void GetBindingsImpl(
  52. const Pattern& pattern,
  53. std::vector<Nonnull<const BindingPattern*>>& bindings) {
  54. switch (pattern.kind()) {
  55. case PatternKind::BindingPattern:
  56. bindings.push_back(&cast<BindingPattern>(pattern));
  57. return;
  58. case PatternKind::TuplePattern:
  59. for (const Pattern* field : cast<TuplePattern>(pattern).fields()) {
  60. GetBindingsImpl(*field, bindings);
  61. }
  62. return;
  63. case PatternKind::AlternativePattern:
  64. GetBindingsImpl(cast<AlternativePattern>(pattern).arguments(), bindings);
  65. return;
  66. case PatternKind::AutoPattern:
  67. case PatternKind::ExpressionPattern:
  68. return;
  69. case PatternKind::VarPattern:
  70. GetBindingsImpl(cast<VarPattern>(pattern).pattern(), bindings);
  71. return;
  72. }
  73. }
  74. auto GetBindings(const Pattern& pattern)
  75. -> std::vector<Nonnull<const BindingPattern*>> {
  76. std::vector<Nonnull<const BindingPattern*>> result;
  77. GetBindingsImpl(pattern, result);
  78. return result;
  79. }
  80. auto PatternFromParenContents(Nonnull<Arena*> arena, SourceLocation source_loc,
  81. const ParenContents<Pattern>& paren_contents)
  82. -> Nonnull<Pattern*> {
  83. std::optional<Nonnull<Pattern*>> single_term = paren_contents.SingleTerm();
  84. if (single_term.has_value()) {
  85. return *single_term;
  86. } else {
  87. return TuplePatternFromParenContents(arena, source_loc, paren_contents);
  88. }
  89. }
  90. auto TuplePatternFromParenContents(Nonnull<Arena*> arena,
  91. SourceLocation source_loc,
  92. const ParenContents<Pattern>& paren_contents)
  93. -> Nonnull<TuplePattern*> {
  94. return arena->New<TuplePattern>(source_loc, paren_contents.elements);
  95. }
  96. // Used by AlternativePattern for constructor initialization. Produces a helpful
  97. // error for incorrect expressions, rather than letting a default cast error
  98. // apply.
  99. auto AlternativePattern::RequireFieldAccess(Nonnull<Expression*> alternative)
  100. -> ErrorOr<Nonnull<FieldAccessExpression*>> {
  101. if (alternative->kind() != ExpressionKind::FieldAccessExpression) {
  102. return FATAL_PROGRAM_ERROR(alternative->source_loc())
  103. << "Alternative pattern must have the form of a field access.";
  104. }
  105. return &cast<FieldAccessExpression>(*alternative);
  106. }
  107. auto ParenExpressionToParenPattern(Nonnull<Arena*> arena,
  108. const ParenContents<Expression>& contents)
  109. -> ParenContents<Pattern> {
  110. ParenContents<Pattern> result = {
  111. .elements = {}, .has_trailing_comma = contents.has_trailing_comma};
  112. for (const auto& element : contents.elements) {
  113. result.elements.push_back(arena->New<ExpressionPattern>(element));
  114. }
  115. return result;
  116. }
  117. } // namespace Carbon