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 "methodDefinition.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include "checker/TSchecker.h"
193af6ab5fSopenharmony_ci#include "compiler/core/ETSGen.h"
203af6ab5fSopenharmony_ci#include "compiler/core/pandagen.h"
213af6ab5fSopenharmony_ci#include "ir/astDump.h"
223af6ab5fSopenharmony_ci#include "ir/srcDump.h"
233af6ab5fSopenharmony_ci
243af6ab5fSopenharmony_cinamespace ark::es2panda::ir {
253af6ab5fSopenharmony_ci
263af6ab5fSopenharmony_ciScriptFunction *MethodDefinition::Function()
273af6ab5fSopenharmony_ci{
283af6ab5fSopenharmony_ci    return value_->IsFunctionExpression() ? value_->AsFunctionExpression()->Function() : nullptr;
293af6ab5fSopenharmony_ci}
303af6ab5fSopenharmony_ci
313af6ab5fSopenharmony_ciconst ScriptFunction *MethodDefinition::Function() const
323af6ab5fSopenharmony_ci{
333af6ab5fSopenharmony_ci    return value_->IsFunctionExpression() ? value_->AsFunctionExpression()->Function() : nullptr;
343af6ab5fSopenharmony_ci}
353af6ab5fSopenharmony_ci
363af6ab5fSopenharmony_ciPrivateFieldKind MethodDefinition::ToPrivateFieldKind(bool const isStatic) const
373af6ab5fSopenharmony_ci{
383af6ab5fSopenharmony_ci    switch (kind_) {
393af6ab5fSopenharmony_ci        case MethodDefinitionKind::METHOD: {
403af6ab5fSopenharmony_ci            return isStatic ? PrivateFieldKind::STATIC_METHOD : PrivateFieldKind::METHOD;
413af6ab5fSopenharmony_ci        }
423af6ab5fSopenharmony_ci        case MethodDefinitionKind::GET: {
433af6ab5fSopenharmony_ci            return isStatic ? PrivateFieldKind::STATIC_GET : PrivateFieldKind::GET;
443af6ab5fSopenharmony_ci        }
453af6ab5fSopenharmony_ci        case MethodDefinitionKind::SET: {
463af6ab5fSopenharmony_ci            return isStatic ? PrivateFieldKind::STATIC_SET : PrivateFieldKind::SET;
473af6ab5fSopenharmony_ci        }
483af6ab5fSopenharmony_ci        default: {
493af6ab5fSopenharmony_ci            UNREACHABLE();
503af6ab5fSopenharmony_ci        }
513af6ab5fSopenharmony_ci    }
523af6ab5fSopenharmony_ci}
533af6ab5fSopenharmony_ci
543af6ab5fSopenharmony_civoid MethodDefinition::ResolveReferences(const NodeTraverser &cb) const
553af6ab5fSopenharmony_ci{
563af6ab5fSopenharmony_ci    cb(key_);
573af6ab5fSopenharmony_ci    cb(value_);
583af6ab5fSopenharmony_ci
593af6ab5fSopenharmony_ci    for (auto *it : overloads_) {
603af6ab5fSopenharmony_ci        cb(it);
613af6ab5fSopenharmony_ci    }
623af6ab5fSopenharmony_ci
633af6ab5fSopenharmony_ci    for (auto *it : decorators_) {
643af6ab5fSopenharmony_ci        cb(it);
653af6ab5fSopenharmony_ci    }
663af6ab5fSopenharmony_ci}
673af6ab5fSopenharmony_ci
683af6ab5fSopenharmony_civoid MethodDefinition::Iterate(const NodeTraverser &cb) const
693af6ab5fSopenharmony_ci{
703af6ab5fSopenharmony_ci    cb(key_);
713af6ab5fSopenharmony_ci    cb(value_);
723af6ab5fSopenharmony_ci
733af6ab5fSopenharmony_ci    for (auto *it : overloads_) {
743af6ab5fSopenharmony_ci        if (it->Parent() == this) {
753af6ab5fSopenharmony_ci            cb(it);
763af6ab5fSopenharmony_ci        }
773af6ab5fSopenharmony_ci    }
783af6ab5fSopenharmony_ci
793af6ab5fSopenharmony_ci    for (auto *it : decorators_) {
803af6ab5fSopenharmony_ci        cb(it);
813af6ab5fSopenharmony_ci    }
823af6ab5fSopenharmony_ci}
833af6ab5fSopenharmony_ci
843af6ab5fSopenharmony_civoid MethodDefinition::TransformChildren(const NodeTransformer &cb, std::string_view const transformationName)
853af6ab5fSopenharmony_ci{
863af6ab5fSopenharmony_ci    if (auto *transformedNode = cb(key_); key_ != transformedNode) {
873af6ab5fSopenharmony_ci        key_->SetTransformedNode(transformationName, transformedNode);
883af6ab5fSopenharmony_ci        key_ = transformedNode->AsExpression();
893af6ab5fSopenharmony_ci    }
903af6ab5fSopenharmony_ci
913af6ab5fSopenharmony_ci    if (auto *transformedNode = cb(value_); value_ != transformedNode) {
923af6ab5fSopenharmony_ci        value_->SetTransformedNode(transformationName, transformedNode);
933af6ab5fSopenharmony_ci        value_ = transformedNode->AsExpression();
943af6ab5fSopenharmony_ci    }
953af6ab5fSopenharmony_ci
963af6ab5fSopenharmony_ci    for (auto *&it : overloads_) {
973af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(it); it != transformedNode) {
983af6ab5fSopenharmony_ci            it->SetTransformedNode(transformationName, transformedNode);
993af6ab5fSopenharmony_ci            it = transformedNode->AsMethodDefinition();
1003af6ab5fSopenharmony_ci        }
1013af6ab5fSopenharmony_ci    }
1023af6ab5fSopenharmony_ci
1033af6ab5fSopenharmony_ci    for (auto *&it : decorators_) {
1043af6ab5fSopenharmony_ci        if (auto *transformedNode = cb(it); it != transformedNode) {
1053af6ab5fSopenharmony_ci            it->SetTransformedNode(transformationName, transformedNode);
1063af6ab5fSopenharmony_ci            it = transformedNode->AsDecorator();
1073af6ab5fSopenharmony_ci        }
1083af6ab5fSopenharmony_ci    }
1093af6ab5fSopenharmony_ci}
1103af6ab5fSopenharmony_ci
1113af6ab5fSopenharmony_civoid MethodDefinition::Dump(ir::AstDumper *dumper) const
1123af6ab5fSopenharmony_ci{
1133af6ab5fSopenharmony_ci    const char *kind = nullptr;
1143af6ab5fSopenharmony_ci
1153af6ab5fSopenharmony_ci    switch (kind_) {
1163af6ab5fSopenharmony_ci        case MethodDefinitionKind::CONSTRUCTOR: {
1173af6ab5fSopenharmony_ci            kind = "constructor";
1183af6ab5fSopenharmony_ci            break;
1193af6ab5fSopenharmony_ci        }
1203af6ab5fSopenharmony_ci        case MethodDefinitionKind::METHOD: {
1213af6ab5fSopenharmony_ci            kind = "method";
1223af6ab5fSopenharmony_ci            break;
1233af6ab5fSopenharmony_ci        }
1243af6ab5fSopenharmony_ci        case MethodDefinitionKind::EXTENSION_METHOD: {
1253af6ab5fSopenharmony_ci            kind = "extensionmethod";
1263af6ab5fSopenharmony_ci            break;
1273af6ab5fSopenharmony_ci        }
1283af6ab5fSopenharmony_ci        case MethodDefinitionKind::GET: {
1293af6ab5fSopenharmony_ci            kind = "get";
1303af6ab5fSopenharmony_ci            break;
1313af6ab5fSopenharmony_ci        }
1323af6ab5fSopenharmony_ci        case MethodDefinitionKind::SET: {
1333af6ab5fSopenharmony_ci            kind = "set";
1343af6ab5fSopenharmony_ci            break;
1353af6ab5fSopenharmony_ci        }
1363af6ab5fSopenharmony_ci        default: {
1373af6ab5fSopenharmony_ci            UNREACHABLE();
1383af6ab5fSopenharmony_ci        }
1393af6ab5fSopenharmony_ci    }
1403af6ab5fSopenharmony_ci
1413af6ab5fSopenharmony_ci    dumper->Add({{"type", "MethodDefinition"},
1423af6ab5fSopenharmony_ci                 {"key", key_},
1433af6ab5fSopenharmony_ci                 {"kind", kind},
1443af6ab5fSopenharmony_ci                 {"accessibility", AstDumper::Optional(AstDumper::ModifierToString(flags_))},
1453af6ab5fSopenharmony_ci                 {"static", IsStatic()},
1463af6ab5fSopenharmony_ci                 {"optional", IsOptionalDeclaration()},
1473af6ab5fSopenharmony_ci                 {"computed", isComputed_},
1483af6ab5fSopenharmony_ci                 {"value", value_},
1493af6ab5fSopenharmony_ci                 {"overloads", overloads_},
1503af6ab5fSopenharmony_ci                 {"decorators", decorators_}});
1513af6ab5fSopenharmony_ci}
1523af6ab5fSopenharmony_ci
1533af6ab5fSopenharmony_civoid MethodDefinition::DumpPrefix(ir::SrcDumper *dumper) const
1543af6ab5fSopenharmony_ci{
1553af6ab5fSopenharmony_ci    if (Parent() != nullptr && Parent()->IsClassDefinition() && !Parent()->AsClassDefinition()->IsLocal()) {
1563af6ab5fSopenharmony_ci        if (IsPrivate()) {
1573af6ab5fSopenharmony_ci            dumper->Add("private ");
1583af6ab5fSopenharmony_ci        } else if (IsProtected()) {
1593af6ab5fSopenharmony_ci            dumper->Add("protected ");
1603af6ab5fSopenharmony_ci        } else if (IsInternal()) {
1613af6ab5fSopenharmony_ci            dumper->Add("internal ");
1623af6ab5fSopenharmony_ci        } else {
1633af6ab5fSopenharmony_ci            dumper->Add("public ");
1643af6ab5fSopenharmony_ci        }
1653af6ab5fSopenharmony_ci    }
1663af6ab5fSopenharmony_ci
1673af6ab5fSopenharmony_ci    if (IsStatic()) {
1683af6ab5fSopenharmony_ci        dumper->Add("static ");
1693af6ab5fSopenharmony_ci    }
1703af6ab5fSopenharmony_ci
1713af6ab5fSopenharmony_ci    if (IsAbstract()) {
1723af6ab5fSopenharmony_ci        dumper->Add("abstract ");
1733af6ab5fSopenharmony_ci    }
1743af6ab5fSopenharmony_ci
1753af6ab5fSopenharmony_ci    if (IsFinal()) {
1763af6ab5fSopenharmony_ci        dumper->Add("final ");
1773af6ab5fSopenharmony_ci    }
1783af6ab5fSopenharmony_ci
1793af6ab5fSopenharmony_ci    if (IsNative()) {
1803af6ab5fSopenharmony_ci        dumper->Add("native ");
1813af6ab5fSopenharmony_ci    }
1823af6ab5fSopenharmony_ci
1833af6ab5fSopenharmony_ci    if (IsAsync()) {
1843af6ab5fSopenharmony_ci        dumper->Add("async ");
1853af6ab5fSopenharmony_ci    }
1863af6ab5fSopenharmony_ci
1873af6ab5fSopenharmony_ci    if (IsOverride()) {
1883af6ab5fSopenharmony_ci        dumper->Add("override ");
1893af6ab5fSopenharmony_ci    }
1903af6ab5fSopenharmony_ci
1913af6ab5fSopenharmony_ci    if (kind_ == MethodDefinitionKind::GET) {
1923af6ab5fSopenharmony_ci        dumper->Add("get ");
1933af6ab5fSopenharmony_ci    } else if (kind_ == MethodDefinitionKind::SET) {
1943af6ab5fSopenharmony_ci        dumper->Add("set ");
1953af6ab5fSopenharmony_ci    }
1963af6ab5fSopenharmony_ci}
1973af6ab5fSopenharmony_ci
1983af6ab5fSopenharmony_civoid MethodDefinition::Dump(ir::SrcDumper *dumper) const
1993af6ab5fSopenharmony_ci{
2003af6ab5fSopenharmony_ci    for (auto method : overloads_) {
2013af6ab5fSopenharmony_ci        method->Dump(dumper);
2023af6ab5fSopenharmony_ci        dumper->Endl();
2033af6ab5fSopenharmony_ci    }
2043af6ab5fSopenharmony_ci
2053af6ab5fSopenharmony_ci    // Do not dump default constructor
2063af6ab5fSopenharmony_ci    if (Parent() != nullptr && Parent()->IsClassDefinition() && value_->IsFunctionExpression() &&
2073af6ab5fSopenharmony_ci        value_->AsFunctionExpression()->Function() != nullptr &&
2083af6ab5fSopenharmony_ci        value_->AsFunctionExpression()->Function()->IsImplicitSuperCallNeeded()) {
2093af6ab5fSopenharmony_ci        return;
2103af6ab5fSopenharmony_ci    }
2113af6ab5fSopenharmony_ci
2123af6ab5fSopenharmony_ci    DumpPrefix(dumper);
2133af6ab5fSopenharmony_ci
2143af6ab5fSopenharmony_ci    if (key_ != nullptr) {
2153af6ab5fSopenharmony_ci        key_->Dump(dumper);
2163af6ab5fSopenharmony_ci    }
2173af6ab5fSopenharmony_ci
2183af6ab5fSopenharmony_ci    if (value_ != nullptr) {
2193af6ab5fSopenharmony_ci        value_->Dump(dumper);
2203af6ab5fSopenharmony_ci    }
2213af6ab5fSopenharmony_ci}
2223af6ab5fSopenharmony_ci
2233af6ab5fSopenharmony_civoid MethodDefinition::Compile(compiler::PandaGen *pg) const
2243af6ab5fSopenharmony_ci{
2253af6ab5fSopenharmony_ci    pg->GetAstCompiler()->Compile(this);
2263af6ab5fSopenharmony_ci}
2273af6ab5fSopenharmony_ci
2283af6ab5fSopenharmony_civoid MethodDefinition::Compile(compiler::ETSGen *etsg) const
2293af6ab5fSopenharmony_ci{
2303af6ab5fSopenharmony_ci    etsg->GetAstCompiler()->Compile(this);
2313af6ab5fSopenharmony_ci}
2323af6ab5fSopenharmony_ci
2333af6ab5fSopenharmony_cichecker::Type *MethodDefinition::Check(checker::TSChecker *checker)
2343af6ab5fSopenharmony_ci{
2353af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2363af6ab5fSopenharmony_ci}
2373af6ab5fSopenharmony_ci
2383af6ab5fSopenharmony_cichecker::Type *MethodDefinition::Check(checker::ETSChecker *checker)
2393af6ab5fSopenharmony_ci{
2403af6ab5fSopenharmony_ci    return checker->GetAnalyzer()->Check(this);
2413af6ab5fSopenharmony_ci}
2423af6ab5fSopenharmony_ci
2433af6ab5fSopenharmony_ciMethodDefinition *MethodDefinition::Clone(ArenaAllocator *const allocator, AstNode *const parent)
2443af6ab5fSopenharmony_ci{
2453af6ab5fSopenharmony_ci    auto *const key = key_ != nullptr ? key_->Clone(allocator, nullptr)->AsExpression() : nullptr;
2463af6ab5fSopenharmony_ci    auto *const value = value_ != nullptr ? value_->Clone(allocator, nullptr)->AsExpression() : nullptr;
2473af6ab5fSopenharmony_ci
2483af6ab5fSopenharmony_ci    if (auto *const clone = allocator->New<MethodDefinition>(kind_, key, value, flags_, allocator, isComputed_);
2493af6ab5fSopenharmony_ci        clone != nullptr) {
2503af6ab5fSopenharmony_ci        if (parent != nullptr) {
2513af6ab5fSopenharmony_ci            clone->SetParent(parent);
2523af6ab5fSopenharmony_ci        }
2533af6ab5fSopenharmony_ci
2543af6ab5fSopenharmony_ci        if (key != nullptr) {
2553af6ab5fSopenharmony_ci            key->SetParent(clone);
2563af6ab5fSopenharmony_ci        }
2573af6ab5fSopenharmony_ci
2583af6ab5fSopenharmony_ci        if (value != nullptr) {
2593af6ab5fSopenharmony_ci            value->SetParent(clone);
2603af6ab5fSopenharmony_ci        }
2613af6ab5fSopenharmony_ci
2623af6ab5fSopenharmony_ci        for (auto *const decorator : decorators_) {
2633af6ab5fSopenharmony_ci            clone->AddDecorator(decorator->Clone(allocator, clone));
2643af6ab5fSopenharmony_ci        }
2653af6ab5fSopenharmony_ci
2663af6ab5fSopenharmony_ci        for (auto *const overloads : overloads_) {
2673af6ab5fSopenharmony_ci            clone->AddOverload(overloads->Clone(allocator, clone));
2683af6ab5fSopenharmony_ci        }
2693af6ab5fSopenharmony_ci
2703af6ab5fSopenharmony_ci        return clone;
2713af6ab5fSopenharmony_ci    }
2723af6ab5fSopenharmony_ci
2733af6ab5fSopenharmony_ci    throw Error(ErrorType::GENERIC, "", CLONE_ALLOCATION_ERROR);
2743af6ab5fSopenharmony_ci}
2753af6ab5fSopenharmony_ci}  // namespace ark::es2panda::ir
276