pattern.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  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. void Pattern::Print(llvm::raw_ostream& out) const {
  15. switch (Tag()) {
  16. case Kind::AutoPattern:
  17. out << "auto";
  18. break;
  19. case Kind::BindingPattern: {
  20. const auto& binding = cast<BindingPattern>(*this);
  21. if (binding.Name().has_value()) {
  22. out << *binding.Name();
  23. } else {
  24. out << "_";
  25. }
  26. out << ": " << *binding.Type();
  27. break;
  28. }
  29. case Kind::TuplePattern: {
  30. const auto& tuple = cast<TuplePattern>(*this);
  31. out << "(";
  32. llvm::ListSeparator sep;
  33. for (const TuplePattern::Field& field : tuple.Fields()) {
  34. out << sep << field.name << " = " << *field.pattern;
  35. }
  36. out << ")";
  37. break;
  38. }
  39. case Kind::AlternativePattern: {
  40. const auto& alternative = cast<AlternativePattern>(*this);
  41. out << *alternative.ChoiceType() << "." << alternative.AlternativeName()
  42. << *alternative.Arguments();
  43. break;
  44. }
  45. case Kind::ExpressionPattern:
  46. out << *cast<ExpressionPattern>(*this).Expression();
  47. break;
  48. }
  49. }
  50. TuplePattern::TuplePattern(Nonnull<Arena*> arena,
  51. Nonnull<const Expression*> tuple_literal)
  52. : Pattern(Kind::TuplePattern, tuple_literal->SourceLoc()) {
  53. const auto& tuple = cast<TupleLiteral>(*tuple_literal);
  54. for (const FieldInitializer& init : tuple.Fields()) {
  55. fields.push_back(
  56. Field(init.name, arena->New<ExpressionPattern>(init.expression)));
  57. }
  58. }
  59. auto PatternFromParenContents(Nonnull<Arena*> arena, SourceLocation loc,
  60. const ParenContents<Pattern>& paren_contents)
  61. -> Nonnull<const Pattern*> {
  62. std::optional<Nonnull<const Pattern*>> single_term =
  63. paren_contents.SingleTerm();
  64. if (single_term.has_value()) {
  65. return *single_term;
  66. } else {
  67. return TuplePatternFromParenContents(arena, loc, paren_contents);
  68. }
  69. }
  70. auto TuplePatternFromParenContents(Nonnull<Arena*> arena, SourceLocation loc,
  71. const ParenContents<Pattern>& paren_contents)
  72. -> Nonnull<const TuplePattern*> {
  73. return arena->New<TuplePattern>(
  74. loc, paren_contents.TupleElements<TuplePattern::Field>(loc));
  75. }
  76. // Used by AlternativePattern for constructor initialization. Produces a helpful
  77. // error for incorrect expressions, rather than letting a default cast error
  78. // apply.
  79. static const FieldAccessExpression& RequireFieldAccess(
  80. Nonnull<const Expression*> alternative) {
  81. if (alternative->Tag() != Expression::Kind::FieldAccessExpression) {
  82. FATAL_PROGRAM_ERROR(alternative->SourceLoc())
  83. << "Alternative pattern must have the form of a field access.";
  84. }
  85. return cast<FieldAccessExpression>(*alternative);
  86. }
  87. AlternativePattern::AlternativePattern(SourceLocation loc,
  88. Nonnull<const Expression*> alternative,
  89. Nonnull<const TuplePattern*> arguments)
  90. : Pattern(Kind::AlternativePattern, loc),
  91. choice_type(RequireFieldAccess(alternative).Aggregate()),
  92. alternative_name(RequireFieldAccess(alternative).Field()),
  93. arguments(arguments) {}
  94. auto ParenExpressionToParenPattern(Nonnull<Arena*> arena,
  95. const ParenContents<Expression>& contents)
  96. -> ParenContents<Pattern> {
  97. ParenContents<Pattern> result = {
  98. .elements = {}, .has_trailing_comma = contents.has_trailing_comma};
  99. for (const auto& element : contents.elements) {
  100. result.elements.push_back(
  101. {.name = element.name,
  102. .term = arena->New<ExpressionPattern>(element.term)});
  103. }
  104. return result;
  105. }
  106. } // namespace Carbon