13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021-2024 Huawei Device Co., Ltd.
33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License.
53af6ab5fSopenharmony_ci * You may obtain a copy of the License at
63af6ab5fSopenharmony_ci *
73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
83af6ab5fSopenharmony_ci *
93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and
133af6ab5fSopenharmony_ci * limitations under the License.
143af6ab5fSopenharmony_ci */
153af6ab5fSopenharmony_ci
163af6ab5fSopenharmony_ci#include "property.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include "es2panda.h"
193af6ab5fSopenharmony_ci#include "checker/TSchecker.h"
203af6ab5fSopenharmony_ci#include "compiler/core/ETSGen.h"
213af6ab5fSopenharmony_ci#include "compiler/core/pandagen.h"
223af6ab5fSopenharmony_ci#include "ir/astDump.h"
233af6ab5fSopenharmony_ci#include "ir/srcDump.h"
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_cinamespace ark::es2panda::ir {
263af6ab5fSopenharmony_ciProperty::Property([[maybe_unused]] Tag const tag, Property const &other, Expression *const key,
273af6ab5fSopenharmony_ci                   Expression *const value)
283af6ab5fSopenharmony_ci    : Property(other)
293af6ab5fSopenharmony_ci{
303af6ab5fSopenharmony_ci    key_ = key;
313af6ab5fSopenharmony_ci    value_ = value;
323af6ab5fSopenharmony_ci}
333af6ab5fSopenharmony_ci
343af6ab5fSopenharmony_ciProperty *Property::Clone(ArenaAllocator *const allocator, AstNode *const parent)
353af6ab5fSopenharmony_ci{
363af6ab5fSopenharmony_ci    auto *const key = key_ != nullptr ? key_->Clone(allocator, nullptr)->AsExpression() : nullptr;
373af6ab5fSopenharmony_ci    auto *const value = value_ != nullptr ? value_->Clone(allocator, nullptr)->AsExpression() : nullptr;
383af6ab5fSopenharmony_ci
393af6ab5fSopenharmony_ci    if (auto *const clone = allocator->New<Property>(Tag {}, *this, key, value); clone != nullptr) {
403af6ab5fSopenharmony_ci        if (key != nullptr) {
413af6ab5fSopenharmony_ci            key->SetParent(clone);
423af6ab5fSopenharmony_ci        }
433af6ab5fSopenharmony_ci        if (value != nullptr) {
443af6ab5fSopenharmony_ci            value->SetParent(clone);
453af6ab5fSopenharmony_ci        }
463af6ab5fSopenharmony_ci        if (parent != nullptr) {
473af6ab5fSopenharmony_ci            clone->SetParent(parent);
483af6ab5fSopenharmony_ci        }
493af6ab5fSopenharmony_ci        return clone;
503af6ab5fSopenharmony_ci    }
513af6ab5fSopenharmony_ci
523af6ab5fSopenharmony_ci    throw Error(ErrorType::GENERIC, "", CLONE_ALLOCATION_ERROR);
533af6ab5fSopenharmony_ci}
543af6ab5fSopenharmony_ci
553af6ab5fSopenharmony_cibool Property::ConvertibleToPatternProperty()
563af6ab5fSopenharmony_ci{
573af6ab5fSopenharmony_ci    // Object pattern can't contain getter or setter
583af6ab5fSopenharmony_ci    if (IsAccessor() || isMethod_) {
593af6ab5fSopenharmony_ci        return false;
603af6ab5fSopenharmony_ci    }
613af6ab5fSopenharmony_ci
623af6ab5fSopenharmony_ci    switch (value_->Type()) {
633af6ab5fSopenharmony_ci        case AstNodeType::OBJECT_EXPRESSION: {
643af6ab5fSopenharmony_ci            return value_->AsObjectExpression()->ConvertibleToObjectPattern();
653af6ab5fSopenharmony_ci        }
663af6ab5fSopenharmony_ci        case AstNodeType::ARRAY_EXPRESSION: {
673af6ab5fSopenharmony_ci            return value_->AsArrayExpression()->ConvertibleToArrayPattern();
683af6ab5fSopenharmony_ci        }
693af6ab5fSopenharmony_ci        case AstNodeType::ASSIGNMENT_EXPRESSION: {
703af6ab5fSopenharmony_ci            return value_->AsAssignmentExpression()->ConvertibleToAssignmentPattern();
713af6ab5fSopenharmony_ci        }
723af6ab5fSopenharmony_ci        case AstNodeType::IDENTIFIER:
733af6ab5fSopenharmony_ci        case AstNodeType::MEMBER_EXPRESSION:
743af6ab5fSopenharmony_ci        case AstNodeType::ARRAY_PATTERN:
753af6ab5fSopenharmony_ci        case AstNodeType::OBJECT_PATTERN:
763af6ab5fSopenharmony_ci        case AstNodeType::ASSIGNMENT_PATTERN: {
773af6ab5fSopenharmony_ci            break;
783af6ab5fSopenharmony_ci        }
793af6ab5fSopenharmony_ci        default: {
803af6ab5fSopenharmony_ci            if (isShorthand_) {
813af6ab5fSopenharmony_ci                break;
823af6ab5fSopenharmony_ci            }
833af6ab5fSopenharmony_ci
843af6ab5fSopenharmony_ci            return false;
853af6ab5fSopenharmony_ci        }
863af6ab5fSopenharmony_ci    }
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_ci    return true;
893af6ab5fSopenharmony_ci}
903af6ab5fSopenharmony_ci
913af6ab5fSopenharmony_ciValidationInfo Property::ValidateExpression()
923af6ab5fSopenharmony_ci{
933af6ab5fSopenharmony_ci    ValidationInfo info;
943af6ab5fSopenharmony_ci
953af6ab5fSopenharmony_ci    if (!IsComputed() && !IsMethod() && !IsAccessor() && !IsShorthand()) {
963af6ab5fSopenharmony_ci        bool currentIsProto = false;
973af6ab5fSopenharmony_ci
983af6ab5fSopenharmony_ci        if (key_->IsIdentifier()) {
993af6ab5fSopenharmony_ci            currentIsProto = key_->AsIdentifier()->Name().Is("__proto__");
1003af6ab5fSopenharmony_ci        } else if (key_->IsStringLiteral()) {
1013af6ab5fSopenharmony_ci            currentIsProto = key_->AsStringLiteral()->Str().Is("__proto__");
1023af6ab5fSopenharmony_ci        }
1033af6ab5fSopenharmony_ci
1043af6ab5fSopenharmony_ci        if (currentIsProto) {
1053af6ab5fSopenharmony_ci            kind_ = PropertyKind::PROTO;
1063af6ab5fSopenharmony_ci        }
1073af6ab5fSopenharmony_ci    }
1083af6ab5fSopenharmony_ci
1093af6ab5fSopenharmony_ci    if (value_ != nullptr) {
1103af6ab5fSopenharmony_ci        if (value_->IsAssignmentPattern()) {
1113af6ab5fSopenharmony_ci            return {"Invalid shorthand property initializer.", value_->Start()};
1123af6ab5fSopenharmony_ci        }
1133af6ab5fSopenharmony_ci
1143af6ab5fSopenharmony_ci        if (value_->IsObjectExpression()) {
1153af6ab5fSopenharmony_ci            info = value_->AsObjectExpression()->ValidateExpression();
1163af6ab5fSopenharmony_ci        } else if (value_->IsArrayExpression()) {
1173af6ab5fSopenharmony_ci            info = value_->AsArrayExpression()->ValidateExpression();
1183af6ab5fSopenharmony_ci        }
1193af6ab5fSopenharmony_ci    }
1203af6ab5fSopenharmony_ci
1213af6ab5fSopenharmony_ci    return info;
1223af6ab5fSopenharmony_ci}
1233af6ab5fSopenharmony_ci
1243af6ab5fSopenharmony_civoid Property::TransformChildren(const NodeTransformer &cb, std::string_view transformationName)
1253af6ab5fSopenharmony_ci{
1263af6ab5fSopenharmony_ci    if (auto *transformedNode = cb(key_); key_ != transformedNode) {
1273af6ab5fSopenharmony_ci        key_->SetTransformedNode(transformationName, transformedNode);
1283af6ab5fSopenharmony_ci        key_ = transformedNode->AsExpression();
1293af6ab5fSopenharmony_ci    }
1303af6ab5fSopenharmony_ci
1313af6ab5fSopenharmony_ci    if (auto *transformedNode = cb(value_); value_ != transformedNode) {
1323af6ab5fSopenharmony_ci        value_->SetTransformedNode(transformationName, transformedNode);
1333af6ab5fSopenharmony_ci        value_ = transformedNode->AsExpression();
1343af6ab5fSopenharmony_ci    }
1353af6ab5fSopenharmony_ci}
1363af6ab5fSopenharmony_ci
1373af6ab5fSopenharmony_civoid Property::Iterate(const NodeTraverser &cb) const
1383af6ab5fSopenharmony_ci{
1393af6ab5fSopenharmony_ci    cb(key_);
1403af6ab5fSopenharmony_ci    cb(value_);
1413af6ab5fSopenharmony_ci}
1423af6ab5fSopenharmony_ci
1433af6ab5fSopenharmony_civoid Property::Dump(ir::AstDumper *dumper) const
1443af6ab5fSopenharmony_ci{
1453af6ab5fSopenharmony_ci    const char *kind = nullptr;
1463af6ab5fSopenharmony_ci
1473af6ab5fSopenharmony_ci    switch (kind_) {
1483af6ab5fSopenharmony_ci        case PropertyKind::INIT: {
1493af6ab5fSopenharmony_ci            kind = "init";
1503af6ab5fSopenharmony_ci            break;
1513af6ab5fSopenharmony_ci        }
1523af6ab5fSopenharmony_ci        case PropertyKind::PROTO: {
1533af6ab5fSopenharmony_ci            kind = "proto";
1543af6ab5fSopenharmony_ci            break;
1553af6ab5fSopenharmony_ci        }
1563af6ab5fSopenharmony_ci        case PropertyKind::GET: {
1573af6ab5fSopenharmony_ci            kind = "get";
1583af6ab5fSopenharmony_ci            break;
1593af6ab5fSopenharmony_ci        }
1603af6ab5fSopenharmony_ci        case PropertyKind::SET: {
1613af6ab5fSopenharmony_ci            kind = "set";
1623af6ab5fSopenharmony_ci            break;
1633af6ab5fSopenharmony_ci        }
1643af6ab5fSopenharmony_ci        default: {
1653af6ab5fSopenharmony_ci            UNREACHABLE();
1663af6ab5fSopenharmony_ci        }
1673af6ab5fSopenharmony_ci    }
1683af6ab5fSopenharmony_ci
1693af6ab5fSopenharmony_ci    dumper->Add({{"type", "Property"},
1703af6ab5fSopenharmony_ci                 {"method", isMethod_},
1713af6ab5fSopenharmony_ci                 {"shorthand", isShorthand_},
1723af6ab5fSopenharmony_ci                 {"computed", isComputed_},
1733af6ab5fSopenharmony_ci                 {"key", key_},
1743af6ab5fSopenharmony_ci                 {"value", value_},
1753af6ab5fSopenharmony_ci                 {"kind", kind}});
1763af6ab5fSopenharmony_ci}
1773af6ab5fSopenharmony_ci
1783af6ab5fSopenharmony_civoid Property::Compile(compiler::PandaGen *pg) const
1793af6ab5fSopenharmony_ci{
1803af6ab5fSopenharmony_ci    pg->GetAstCompiler()->Compile(this);
1813af6ab5fSopenharmony_ci}
1823af6ab5fSopenharmony_ci
1833af6ab5fSopenharmony_civoid Property::Compile(compiler::ETSGen *etsg) const
1843af6ab5fSopenharmony_ci{
1853af6ab5fSopenharmony_ci    etsg->GetAstCompiler()->Compile(this);
1863af6ab5fSopenharmony_ci}
1873af6ab5fSopenharmony_ci
1883af6ab5fSopenharmony_civoid Property::Dump(ir::SrcDumper *dumper) const
1893af6ab5fSopenharmony_ci{
1903af6ab5fSopenharmony_ci    if (kind_ == PropertyKind::INIT) {
1913af6ab5fSopenharmony_ci        key_->Dump(dumper);
1923af6ab5fSopenharmony_ci        dumper->Add(": ");
1933af6ab5fSopenharmony_ci        value_->Dump(dumper);
1943af6ab5fSopenharmony_ci    }
1953af6ab5fSopenharmony_ci}
1963af6ab5fSopenharmony_ci
1973af6ab5fSopenharmony_cichecker::Type *Property::Check(checker::TSChecker *checker)
1983af6ab5fSopenharmony_ci{
1993af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2003af6ab5fSopenharmony_ci}
2013af6ab5fSopenharmony_ci
2023af6ab5fSopenharmony_cichecker::Type *Property::Check(checker::ETSChecker *checker)
2033af6ab5fSopenharmony_ci{
2043af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2053af6ab5fSopenharmony_ci}
2063af6ab5fSopenharmony_ci}  // namespace ark::es2panda::ir
207