| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "executable_semantics/ast/pattern.h"
- #include <string>
- #include "common/ostream.h"
- #include "executable_semantics/ast/expression.h"
- #include "executable_semantics/common/arena.h"
- #include "executable_semantics/common/error.h"
- #include "llvm/ADT/StringExtras.h"
- #include "llvm/Support/Casting.h"
- namespace Carbon {
- using llvm::cast;
- void Pattern::Print(llvm::raw_ostream& out) const {
- switch (Tag()) {
- case Kind::AutoPattern:
- out << "auto";
- break;
- case Kind::BindingPattern: {
- const auto& binding = cast<BindingPattern>(*this);
- if (binding.Name().has_value()) {
- out << *binding.Name();
- } else {
- out << "_";
- }
- out << ": " << *binding.Type();
- break;
- }
- case Kind::TuplePattern: {
- const auto& tuple = cast<TuplePattern>(*this);
- out << "(";
- llvm::ListSeparator sep;
- for (const TuplePattern::Field& field : tuple.Fields()) {
- out << sep << field.name << " = " << field.pattern;
- }
- out << ")";
- break;
- }
- case Kind::AlternativePattern: {
- const auto& alternative = cast<AlternativePattern>(*this);
- out << alternative.ChoiceType() << "." << alternative.AlternativeName()
- << alternative.Arguments();
- break;
- }
- case Kind::ExpressionPattern:
- out << cast<ExpressionPattern>(*this).Expression();
- break;
- }
- }
- TuplePattern::TuplePattern(const Expression* tuple_literal)
- : Pattern(Kind::TuplePattern, tuple_literal->LineNumber()) {
- const auto& tuple = cast<TupleLiteral>(*tuple_literal);
- for (const FieldInitializer& init : tuple.Fields()) {
- fields.push_back(Field(
- init.name, global_arena->New<ExpressionPattern>(init.expression)));
- }
- }
- auto PatternFromParenContents(int line_num,
- const ParenContents<Pattern>& paren_contents)
- -> const Pattern* {
- std::optional<const Pattern*> single_term = paren_contents.SingleTerm();
- if (single_term.has_value()) {
- return *single_term;
- } else {
- return TuplePatternFromParenContents(line_num, paren_contents);
- }
- }
- auto TuplePatternFromParenContents(int line_num,
- const ParenContents<Pattern>& paren_contents)
- -> const TuplePattern* {
- return global_arena->New<TuplePattern>(
- line_num, paren_contents.TupleElements<TuplePattern::Field>(line_num));
- }
- AlternativePattern::AlternativePattern(int line_num,
- const Expression* alternative,
- const TuplePattern* arguments)
- : Pattern(Kind::AlternativePattern, line_num), arguments(arguments) {
- if (alternative->Tag() != Expression::Kind::FieldAccessExpression) {
- FATAL_PROGRAM_ERROR(alternative->LineNumber())
- << "Alternative pattern must have the form of a field access.";
- }
- const auto& field_access = cast<FieldAccessExpression>(*alternative);
- choice_type = field_access.Aggregate();
- alternative_name = field_access.Field();
- }
- auto ParenExpressionToParenPattern(const ParenContents<Expression>& contents)
- -> ParenContents<Pattern> {
- ParenContents<Pattern> result = {
- .elements = {}, .has_trailing_comma = contents.has_trailing_comma};
- for (const auto& element : contents.elements) {
- result.elements.push_back(
- {.name = element.name,
- .term = global_arena->New<ExpressionPattern>(element.term)});
- }
- return result;
- }
- } // namespace Carbon
|