13af6ab5fSopenharmony_ci/**
23af6ab5fSopenharmony_ci * Copyright (c) 2021 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 "classDefinition.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include "binder/binder.h"
193af6ab5fSopenharmony_ci#include "binder/scope.h"
203af6ab5fSopenharmony_ci#include "compiler/base/literals.h"
213af6ab5fSopenharmony_ci#include "compiler/base/lreference.h"
223af6ab5fSopenharmony_ci#include "compiler/core/pandagen.h"
233af6ab5fSopenharmony_ci#include "typescript/checker.h"
243af6ab5fSopenharmony_ci#include "ir/astDump.h"
253af6ab5fSopenharmony_ci#include "ir/base/classProperty.h"
263af6ab5fSopenharmony_ci#include "ir/base/methodDefinition.h"
273af6ab5fSopenharmony_ci#include "ir/base/scriptFunction.h"
283af6ab5fSopenharmony_ci#include "ir/expression.h"
293af6ab5fSopenharmony_ci#include "ir/expressions/functionExpression.h"
303af6ab5fSopenharmony_ci#include "ir/expressions/identifier.h"
313af6ab5fSopenharmony_ci#include "ir/expressions/literals/nullLiteral.h"
323af6ab5fSopenharmony_ci#include "ir/expressions/literals/numberLiteral.h"
333af6ab5fSopenharmony_ci#include "ir/expressions/literals/stringLiteral.h"
343af6ab5fSopenharmony_ci#include "ir/expressions/literals/taggedLiteral.h"
353af6ab5fSopenharmony_ci#include "ir/ts/tsClassImplements.h"
363af6ab5fSopenharmony_ci#include "ir/ts/tsIndexSignature.h"
373af6ab5fSopenharmony_ci#include "ir/ts/tsTypeParameter.h"
383af6ab5fSopenharmony_ci#include "ir/ts/tsTypeParameterDeclaration.h"
393af6ab5fSopenharmony_ci#include "ir/ts/tsTypeParameterInstantiation.h"
403af6ab5fSopenharmony_ci#include "ir/ts/tsUnionType.h"
413af6ab5fSopenharmony_ci#include "util/helpers.h"
423af6ab5fSopenharmony_ci
433af6ab5fSopenharmony_cinamespace panda::es2panda::ir {
443af6ab5fSopenharmony_ci
453af6ab5fSopenharmony_ciconst FunctionExpression *ClassDefinition::Ctor() const
463af6ab5fSopenharmony_ci{
473af6ab5fSopenharmony_ci    ASSERT(ctor_ != nullptr);
483af6ab5fSopenharmony_ci    return ctor_->Value();
493af6ab5fSopenharmony_ci}
503af6ab5fSopenharmony_ci
513af6ab5fSopenharmony_ciutil::StringView ClassDefinition::GetName() const
523af6ab5fSopenharmony_ci{
533af6ab5fSopenharmony_ci    if (ident_) {
543af6ab5fSopenharmony_ci        return ident_->Name();
553af6ab5fSopenharmony_ci    }
563af6ab5fSopenharmony_ci
573af6ab5fSopenharmony_ci    if (exportDefault_) {
583af6ab5fSopenharmony_ci        return parser::SourceTextModuleRecord::DEFAULT_LOCAL_NAME;
593af6ab5fSopenharmony_ci    }
603af6ab5fSopenharmony_ci
613af6ab5fSopenharmony_ci    return "";
623af6ab5fSopenharmony_ci}
633af6ab5fSopenharmony_ci
643af6ab5fSopenharmony_civoid ClassDefinition::Iterate(const NodeTraverser &cb) const
653af6ab5fSopenharmony_ci{
663af6ab5fSopenharmony_ci    if (ident_) {
673af6ab5fSopenharmony_ci        cb(ident_);
683af6ab5fSopenharmony_ci    }
693af6ab5fSopenharmony_ci
703af6ab5fSopenharmony_ci    if (typeParams_) {
713af6ab5fSopenharmony_ci        cb(typeParams_);
723af6ab5fSopenharmony_ci    }
733af6ab5fSopenharmony_ci
743af6ab5fSopenharmony_ci    if (superClass_) {
753af6ab5fSopenharmony_ci        cb(superClass_);
763af6ab5fSopenharmony_ci    }
773af6ab5fSopenharmony_ci
783af6ab5fSopenharmony_ci    if (superTypeParams_) {
793af6ab5fSopenharmony_ci        cb(superTypeParams_);
803af6ab5fSopenharmony_ci    }
813af6ab5fSopenharmony_ci
823af6ab5fSopenharmony_ci    for (auto *it : implements_) {
833af6ab5fSopenharmony_ci        cb(it);
843af6ab5fSopenharmony_ci    }
853af6ab5fSopenharmony_ci
863af6ab5fSopenharmony_ci    cb(ctor_);
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_ci    for (auto *it : body_) {
893af6ab5fSopenharmony_ci        cb(it);
903af6ab5fSopenharmony_ci    }
913af6ab5fSopenharmony_ci
923af6ab5fSopenharmony_ci    for (auto *it : indexSignatures_) {
933af6ab5fSopenharmony_ci        cb(it);
943af6ab5fSopenharmony_ci    }
953af6ab5fSopenharmony_ci}
963af6ab5fSopenharmony_ci
973af6ab5fSopenharmony_civoid ClassDefinition::Dump(ir::AstDumper *dumper) const
983af6ab5fSopenharmony_ci{
993af6ab5fSopenharmony_ci    dumper->Add({{"id", AstDumper::Nullable(ident_)},
1003af6ab5fSopenharmony_ci                 {"typeParameters", AstDumper::Optional(typeParams_)},
1013af6ab5fSopenharmony_ci                 {"superClass", AstDumper::Nullable(superClass_)},
1023af6ab5fSopenharmony_ci                 {"superTypeParameters", AstDumper::Optional(superTypeParams_)},
1033af6ab5fSopenharmony_ci                 {"implements", implements_},
1043af6ab5fSopenharmony_ci                 {"constructor", ctor_},
1053af6ab5fSopenharmony_ci                 {"body", body_},
1063af6ab5fSopenharmony_ci                 {"indexSignatures", indexSignatures_}});
1073af6ab5fSopenharmony_ci}
1083af6ab5fSopenharmony_ci
1093af6ab5fSopenharmony_cicompiler::VReg ClassDefinition::CompileHeritageClause(compiler::PandaGen *pg) const
1103af6ab5fSopenharmony_ci{
1113af6ab5fSopenharmony_ci    compiler::VReg baseReg = pg->AllocReg();
1123af6ab5fSopenharmony_ci
1133af6ab5fSopenharmony_ci    if (superClass_) {
1143af6ab5fSopenharmony_ci        superClass_->Compile(pg);
1153af6ab5fSopenharmony_ci    } else {
1163af6ab5fSopenharmony_ci        pg->LoadConst(this, compiler::Constant::JS_HOLE);
1173af6ab5fSopenharmony_ci    }
1183af6ab5fSopenharmony_ci
1193af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, baseReg);
1203af6ab5fSopenharmony_ci    return baseReg;
1213af6ab5fSopenharmony_ci}
1223af6ab5fSopenharmony_ci
1233af6ab5fSopenharmony_civoid ClassDefinition::InitializeClassName(compiler::PandaGen *pg) const
1243af6ab5fSopenharmony_ci{
1253af6ab5fSopenharmony_ci    if (!ident_) {
1263af6ab5fSopenharmony_ci        return;
1273af6ab5fSopenharmony_ci    }
1283af6ab5fSopenharmony_ci
1293af6ab5fSopenharmony_ci    compiler::LReference lref = compiler::LReference::CreateLRef(pg, ident_, true);
1303af6ab5fSopenharmony_ci    lref.SetValue();
1313af6ab5fSopenharmony_ci}
1323af6ab5fSopenharmony_ci
1333af6ab5fSopenharmony_ci// NOLINTNEXTLINE(google-runtime-references)
1343af6ab5fSopenharmony_ciint32_t ClassDefinition::CreateClassPublicBuffer(compiler::PandaGen *pg, util::BitSet &compiled,
1353af6ab5fSopenharmony_ci    int32_t fieldTypeBufIdx) const
1363af6ab5fSopenharmony_ci{
1373af6ab5fSopenharmony_ci    auto *buf = pg->NewLiteralBuffer();
1383af6ab5fSopenharmony_ci    compiler::LiteralBuffer staticBuf(pg->Allocator());
1393af6ab5fSopenharmony_ci    uint32_t instancePropertyCount = 0;
1403af6ab5fSopenharmony_ci    std::unordered_map<util::StringView, size_t> propNameMap;
1413af6ab5fSopenharmony_ci    std::unordered_map<util::StringView, size_t> staticPropNameMap;
1423af6ab5fSopenharmony_ci
1433af6ab5fSopenharmony_ci    const auto &properties = body_;
1443af6ab5fSopenharmony_ci
1453af6ab5fSopenharmony_ci    for (size_t i = 0; i < properties.size(); i++) {
1463af6ab5fSopenharmony_ci        if (!properties[i]->IsMethodDefinition()) {
1473af6ab5fSopenharmony_ci            continue;
1483af6ab5fSopenharmony_ci        }
1493af6ab5fSopenharmony_ci        const ir::MethodDefinition *prop = properties[i]->AsMethodDefinition();
1503af6ab5fSopenharmony_ci        /* If it's sendable, put the getters/setters into literal buffer.
1513af6ab5fSopenharmony_ci         * If not, break at getters/setters to be compatible with api10. */
1523af6ab5fSopenharmony_ci        if (prop->IsPrivate()) {
1533af6ab5fSopenharmony_ci            continue;
1543af6ab5fSopenharmony_ci        }
1553af6ab5fSopenharmony_ci
1563af6ab5fSopenharmony_ci        if (prop->Computed()) {
1573af6ab5fSopenharmony_ci            break;
1583af6ab5fSopenharmony_ci        }
1593af6ab5fSopenharmony_ci
1603af6ab5fSopenharmony_ci        if (prop->IsAccessor() && !isSendable_) {
1613af6ab5fSopenharmony_ci            break;
1623af6ab5fSopenharmony_ci        }
1633af6ab5fSopenharmony_ci
1643af6ab5fSopenharmony_ci        if (prop->IsAbstract()) {
1653af6ab5fSopenharmony_ci            compiled.Set(i);
1663af6ab5fSopenharmony_ci            continue;
1673af6ab5fSopenharmony_ci        }
1683af6ab5fSopenharmony_ci
1693af6ab5fSopenharmony_ci        if (prop->Value()->Function()->IsOverload()) {
1703af6ab5fSopenharmony_ci            compiled.Set(i);
1713af6ab5fSopenharmony_ci            continue;
1723af6ab5fSopenharmony_ci        }
1733af6ab5fSopenharmony_ci
1743af6ab5fSopenharmony_ci        util::StringView name = util::Helpers::LiteralToPropName(pg->Allocator(), prop->Key());
1753af6ab5fSopenharmony_ci        compiler::LiteralBuffer *literalBuf = prop->IsStatic() ? &staticBuf : buf;
1763af6ab5fSopenharmony_ci        auto &nameMap = prop->IsStatic() ? staticPropNameMap : propNameMap;
1773af6ab5fSopenharmony_ci
1783af6ab5fSopenharmony_ci        size_t bufferPos = literalBuf->Literals().size();
1793af6ab5fSopenharmony_ci        auto res = nameMap.insert({name, bufferPos});
1803af6ab5fSopenharmony_ci        if (res.second || isSendable_) {
1813af6ab5fSopenharmony_ci            if (!prop->IsStatic()) {
1823af6ab5fSopenharmony_ci                instancePropertyCount++;
1833af6ab5fSopenharmony_ci            }
1843af6ab5fSopenharmony_ci
1853af6ab5fSopenharmony_ci            literalBuf->Add(pg->Allocator()->New<StringLiteral>(name));
1863af6ab5fSopenharmony_ci            literalBuf->Add(nullptr); // save for method internalname
1873af6ab5fSopenharmony_ci            literalBuf->Add(nullptr); // save for method affiliate
1883af6ab5fSopenharmony_ci        } else {
1893af6ab5fSopenharmony_ci            bufferPos = res.first->second;
1903af6ab5fSopenharmony_ci        }
1913af6ab5fSopenharmony_ci
1923af6ab5fSopenharmony_ci        const ir::FunctionExpression *func = prop->Value()->AsFunctionExpression();
1933af6ab5fSopenharmony_ci        const util::StringView &internalName = func->Function()->Scope()->InternalName();
1943af6ab5fSopenharmony_ci
1953af6ab5fSopenharmony_ci        LiteralTag litTag = (prop->Kind() == MethodDefinitionKind::METHOD) ?  LiteralTag::METHOD :
1963af6ab5fSopenharmony_ci                            ((prop->Kind() == MethodDefinitionKind::SET) ? LiteralTag::SETTER : LiteralTag::GETTER);
1973af6ab5fSopenharmony_ci
1983af6ab5fSopenharmony_ci        Literal *value = pg->Allocator()->New<TaggedLiteral>(litTag, internalName);
1993af6ab5fSopenharmony_ci        literalBuf->ResetLiteral(bufferPos + 1, value);
2003af6ab5fSopenharmony_ci        Literal *methodAffiliate = pg->Allocator()->New<TaggedLiteral>(LiteralTag::METHODAFFILIATE,
2013af6ab5fSopenharmony_ci                                                                       func->Function()->FormalParamsLength());
2023af6ab5fSopenharmony_ci        literalBuf->ResetLiteral(bufferPos + 2, methodAffiliate); // bufferPos + 2 is saved for method affiliate
2033af6ab5fSopenharmony_ci        compiled.Set(i);
2043af6ab5fSopenharmony_ci    }
2053af6ab5fSopenharmony_ci
2063af6ab5fSopenharmony_ci    /* Static items are stored at the end of the buffer */
2073af6ab5fSopenharmony_ci    buf->Insert(&staticBuf);
2083af6ab5fSopenharmony_ci
2093af6ab5fSopenharmony_ci    /* The last literal item represents the offset of the first static property. The regular property literal count
2103af6ab5fSopenharmony_ci     * is divided by 2 as key/value pairs count as one. */
2113af6ab5fSopenharmony_ci    buf->Add(pg->Allocator()->New<NumberLiteral>(instancePropertyCount));
2123af6ab5fSopenharmony_ci
2133af6ab5fSopenharmony_ci    if (IsSendable()) {
2143af6ab5fSopenharmony_ci        std::string recordName = std::string(pg->Binder()->Program()->RecordName());
2153af6ab5fSopenharmony_ci        std::string fieldTypeIdxStr = recordName + "_" + std::to_string(fieldTypeBufIdx);
2163af6ab5fSopenharmony_ci        util::UString fieldTypeLitId(fieldTypeIdxStr, pg->Allocator());
2173af6ab5fSopenharmony_ci        buf->Add(pg->Allocator()->New<TaggedLiteral>(LiteralTag::LITERALARRAY, fieldTypeLitId.View()));
2183af6ab5fSopenharmony_ci    }
2193af6ab5fSopenharmony_ci    return pg->AddLiteralBuffer(buf);
2203af6ab5fSopenharmony_ci}
2213af6ab5fSopenharmony_ci
2223af6ab5fSopenharmony_ciint32_t ClassDefinition::CreateClassPrivateBuffer(compiler::PandaGen *pg) const
2233af6ab5fSopenharmony_ci{
2243af6ab5fSopenharmony_ci    auto *buf = pg->NewLiteralBuffer();
2253af6ab5fSopenharmony_ci    compiler::LiteralBuffer staticBuf(pg->Allocator());
2263af6ab5fSopenharmony_ci    uint32_t instancePropertyCount = 0;
2273af6ab5fSopenharmony_ci
2283af6ab5fSopenharmony_ci    for (const auto *prop : body_) {
2293af6ab5fSopenharmony_ci        if (!prop->IsMethodDefinition()) {
2303af6ab5fSopenharmony_ci            continue;
2313af6ab5fSopenharmony_ci        }
2323af6ab5fSopenharmony_ci
2333af6ab5fSopenharmony_ci        const auto *methodDef = prop->AsMethodDefinition();
2343af6ab5fSopenharmony_ci        if (!methodDef->IsPrivate()) {
2353af6ab5fSopenharmony_ci            continue;
2363af6ab5fSopenharmony_ci        }
2373af6ab5fSopenharmony_ci
2383af6ab5fSopenharmony_ci        compiler::LiteralBuffer *literalBuf = methodDef->IsStatic() ? &staticBuf : (instancePropertyCount++, buf);
2393af6ab5fSopenharmony_ci        const ir::FunctionExpression *func = methodDef->Value()->AsFunctionExpression();
2403af6ab5fSopenharmony_ci        const util::StringView &internalName = func->Function()->Scope()->InternalName();
2413af6ab5fSopenharmony_ci        Literal *value = nullptr;
2423af6ab5fSopenharmony_ci        Literal *methodAffiliate = pg->Allocator()->New<TaggedLiteral>(LiteralTag::METHODAFFILIATE,
2433af6ab5fSopenharmony_ci                                                                       func->Function()->FormalParamsLength());
2443af6ab5fSopenharmony_ci        switch (methodDef->Kind()) {
2453af6ab5fSopenharmony_ci            case MethodDefinitionKind::METHOD: {
2463af6ab5fSopenharmony_ci                value = pg->Allocator()->New<TaggedLiteral>(LiteralTag::METHOD, internalName);
2473af6ab5fSopenharmony_ci                break;
2483af6ab5fSopenharmony_ci            }
2493af6ab5fSopenharmony_ci            case MethodDefinitionKind::GET: {
2503af6ab5fSopenharmony_ci                value = pg->Allocator()->New<TaggedLiteral>(LiteralTag::GETTER, internalName);
2513af6ab5fSopenharmony_ci                break;
2523af6ab5fSopenharmony_ci            }
2533af6ab5fSopenharmony_ci            case MethodDefinitionKind::SET: {
2543af6ab5fSopenharmony_ci                value = pg->Allocator()->New<TaggedLiteral>(LiteralTag::SETTER, internalName);
2553af6ab5fSopenharmony_ci                break;
2563af6ab5fSopenharmony_ci            }
2573af6ab5fSopenharmony_ci            default: {
2583af6ab5fSopenharmony_ci                UNREACHABLE();
2593af6ab5fSopenharmony_ci            }
2603af6ab5fSopenharmony_ci        }
2613af6ab5fSopenharmony_ci        literalBuf->Add(value);
2623af6ab5fSopenharmony_ci        literalBuf->Add(methodAffiliate);
2633af6ab5fSopenharmony_ci    }
2643af6ab5fSopenharmony_ci
2653af6ab5fSopenharmony_ci    buf->Insert(&staticBuf);
2663af6ab5fSopenharmony_ci    buf->Add(pg->Allocator()->New<NumberLiteral>(instancePropertyCount));
2673af6ab5fSopenharmony_ci
2683af6ab5fSopenharmony_ci    return pg->AddLiteralBuffer(buf);
2693af6ab5fSopenharmony_ci}
2703af6ab5fSopenharmony_ci
2713af6ab5fSopenharmony_civoid ClassDefinition::CompileMissingProperties(compiler::PandaGen *pg, const util::BitSet &compiled,
2723af6ab5fSopenharmony_ci                                               compiler::VReg classReg) const
2733af6ab5fSopenharmony_ci{
2743af6ab5fSopenharmony_ci    const auto &properties = body_;
2753af6ab5fSopenharmony_ci
2763af6ab5fSopenharmony_ci    compiler::VReg protoReg = pg->AllocReg();
2773af6ab5fSopenharmony_ci
2783af6ab5fSopenharmony_ci    pg->LoadObjByName(this, classReg, "prototype");
2793af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, protoReg);
2803af6ab5fSopenharmony_ci
2813af6ab5fSopenharmony_ci    for (size_t i = 0; i < properties.size(); i++) {
2823af6ab5fSopenharmony_ci        if (!properties[i]->IsMethodDefinition() || compiled.Test(i)) {
2833af6ab5fSopenharmony_ci            continue;
2843af6ab5fSopenharmony_ci        }
2853af6ab5fSopenharmony_ci
2863af6ab5fSopenharmony_ci        const ir::MethodDefinition *prop = properties[i]->AsMethodDefinition();
2873af6ab5fSopenharmony_ci        if (prop->IsOptional() && prop->Value()->Function()->IsOverload()) {
2883af6ab5fSopenharmony_ci            continue;
2893af6ab5fSopenharmony_ci        }
2903af6ab5fSopenharmony_ci
2913af6ab5fSopenharmony_ci        if (prop->IsPrivate() || prop->IsAbstract()) {
2923af6ab5fSopenharmony_ci            continue;
2933af6ab5fSopenharmony_ci        }
2943af6ab5fSopenharmony_ci
2953af6ab5fSopenharmony_ci        compiler::VReg dest = prop->IsStatic() ? classReg : protoReg;
2963af6ab5fSopenharmony_ci        compiler::RegScope rs(pg);
2973af6ab5fSopenharmony_ci
2983af6ab5fSopenharmony_ci        switch (prop->Kind()) {
2993af6ab5fSopenharmony_ci            case ir::MethodDefinitionKind::METHOD: {
3003af6ab5fSopenharmony_ci                compiler::Operand key = prop->Computed() ? prop->KeyReg() :
3013af6ab5fSopenharmony_ci                    pg->ToPropertyKey(prop->Key(), false);
3023af6ab5fSopenharmony_ci
3033af6ab5fSopenharmony_ci                pg->LoadAccumulator(this, dest);
3043af6ab5fSopenharmony_ci                const ir::FunctionExpression *func = prop->Value()->AsFunctionExpression();
3053af6ab5fSopenharmony_ci                func->Compile(pg);
3063af6ab5fSopenharmony_ci
3073af6ab5fSopenharmony_ci                pg->StoreOwnProperty(prop->Value()->Parent(), dest, key, prop->Computed());
3083af6ab5fSopenharmony_ci                break;
3093af6ab5fSopenharmony_ci            }
3103af6ab5fSopenharmony_ci            case ir::MethodDefinitionKind::GET:
3113af6ab5fSopenharmony_ci            case ir::MethodDefinitionKind::SET: {
3123af6ab5fSopenharmony_ci                CompileGetterOrSetter(pg, dest, prop);
3133af6ab5fSopenharmony_ci                break;
3143af6ab5fSopenharmony_ci            }
3153af6ab5fSopenharmony_ci            default: {
3163af6ab5fSopenharmony_ci                UNREACHABLE();
3173af6ab5fSopenharmony_ci            }
3183af6ab5fSopenharmony_ci        }
3193af6ab5fSopenharmony_ci    }
3203af6ab5fSopenharmony_ci
3213af6ab5fSopenharmony_ci    if (NeedInstanceInitializer()) {
3223af6ab5fSopenharmony_ci        InstanceInitialize(pg, protoReg);
3233af6ab5fSopenharmony_ci    }
3243af6ab5fSopenharmony_ci}
3253af6ab5fSopenharmony_ci
3263af6ab5fSopenharmony_civoid ClassDefinition::StaticInitialize(compiler::PandaGen *pg, compiler::VReg classReg) const
3273af6ab5fSopenharmony_ci{
3283af6ab5fSopenharmony_ci    compiler::VReg callee = pg->AllocReg();
3293af6ab5fSopenharmony_ci    compiler::VReg thisReg = pg->AllocReg();
3303af6ab5fSopenharmony_ci
3313af6ab5fSopenharmony_ci    const ir::FunctionExpression *func = staticInitializer_->Value();
3323af6ab5fSopenharmony_ci    func->Compile(pg);
3333af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, callee);
3343af6ab5fSopenharmony_ci
3353af6ab5fSopenharmony_ci    pg->MoveVreg(this, thisReg, classReg);
3363af6ab5fSopenharmony_ci    pg->CallThis(this, callee, 1);
3373af6ab5fSopenharmony_ci
3383af6ab5fSopenharmony_ci    pg->LoadAccumulator(this, classReg);
3393af6ab5fSopenharmony_ci}
3403af6ab5fSopenharmony_ci
3413af6ab5fSopenharmony_civoid ClassDefinition::InstanceInitialize(compiler::PandaGen *pg, compiler::VReg protoReg) const
3423af6ab5fSopenharmony_ci{
3433af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, protoReg);
3443af6ab5fSopenharmony_ci    instanceInitializer_->Value()->Compile(pg);
3453af6ab5fSopenharmony_ci    pg->StoreLexicalVar(instanceInitializer_, 0, GetSlot(instanceInitializer_->Key()));
3463af6ab5fSopenharmony_ci}
3473af6ab5fSopenharmony_ci
3483af6ab5fSopenharmony_civoid ClassDefinition::CompileComputedKeys(compiler::PandaGen *pg) const
3493af6ab5fSopenharmony_ci{
3503af6ab5fSopenharmony_ci    for (const auto &stmt : body_) {
3513af6ab5fSopenharmony_ci        if (stmt->IsClassProperty()) {
3523af6ab5fSopenharmony_ci            const ir::ClassProperty *prop = stmt->AsClassProperty();
3533af6ab5fSopenharmony_ci
3543af6ab5fSopenharmony_ci            // Do not process non-static public fields when not using define semantic.
3553af6ab5fSopenharmony_ci            if (!prop->IsStatic() && !pg->Binder()->Program()->UseDefineSemantic()) {
3563af6ab5fSopenharmony_ci                continue;
3573af6ab5fSopenharmony_ci            }
3583af6ab5fSopenharmony_ci
3593af6ab5fSopenharmony_ci            if (prop->IsComputed() && prop->NeedCompileKey()) {
3603af6ab5fSopenharmony_ci                prop->Key()->Compile(pg);
3613af6ab5fSopenharmony_ci                pg->ToComputedPropertyKey(prop->Key());
3623af6ab5fSopenharmony_ci                pg->StoreLexicalVar(prop->Key(), 0, GetSlot(prop->Key()));
3633af6ab5fSopenharmony_ci            }
3643af6ab5fSopenharmony_ci        } else if (stmt->IsMethodDefinition()) {
3653af6ab5fSopenharmony_ci            auto *methodDef = stmt->AsMethodDefinition();
3663af6ab5fSopenharmony_ci            if (methodDef->Computed()) {
3673af6ab5fSopenharmony_ci                compiler::VReg keyReg = pg->AllocReg();
3683af6ab5fSopenharmony_ci                methodDef->SetKeyReg(keyReg);
3693af6ab5fSopenharmony_ci                methodDef->Key()->Compile(pg);
3703af6ab5fSopenharmony_ci                pg->ToComputedPropertyKey(methodDef->Key());
3713af6ab5fSopenharmony_ci                pg->StoreAccumulator(methodDef->Key(), keyReg);
3723af6ab5fSopenharmony_ci            }
3733af6ab5fSopenharmony_ci        }
3743af6ab5fSopenharmony_ci    }
3753af6ab5fSopenharmony_ci}
3763af6ab5fSopenharmony_ci
3773af6ab5fSopenharmony_civoid ClassDefinition::Compile(compiler::PandaGen *pg) const
3783af6ab5fSopenharmony_ci{
3793af6ab5fSopenharmony_ci    if (declare_) {
3803af6ab5fSopenharmony_ci        return;
3813af6ab5fSopenharmony_ci    }
3823af6ab5fSopenharmony_ci
3833af6ab5fSopenharmony_ci    if (isSendable_) {
3843af6ab5fSopenharmony_ci        CompileSendableClass(pg);
3853af6ab5fSopenharmony_ci        return;
3863af6ab5fSopenharmony_ci    }
3873af6ab5fSopenharmony_ci
3883af6ab5fSopenharmony_ci    compiler::RegScope rs(pg);
3893af6ab5fSopenharmony_ci    compiler::VReg classReg = pg->AllocReg();
3903af6ab5fSopenharmony_ci
3913af6ab5fSopenharmony_ci    compiler::LabelTarget target(pg);
3923af6ab5fSopenharmony_ci    compiler::VariableEnvScope classEnvScope(pg, scope_, target);
3933af6ab5fSopenharmony_ci    compiler::VReg baseReg = CompileHeritageClause(pg);
3943af6ab5fSopenharmony_ci    util::StringView ctorId = ctor_->Function()->Scope()->InternalName();
3953af6ab5fSopenharmony_ci    util::BitSet compiled(body_.size());
3963af6ab5fSopenharmony_ci
3973af6ab5fSopenharmony_ci    if (hasComputedKey_) {
3983af6ab5fSopenharmony_ci        CompileComputedKeys(pg);
3993af6ab5fSopenharmony_ci    }
4003af6ab5fSopenharmony_ci
4013af6ab5fSopenharmony_ci    int32_t bufIdx = CreateClassPublicBuffer(pg, compiled);
4023af6ab5fSopenharmony_ci    pg->DefineClassWithBuffer(this, ctorId, bufIdx, baseReg);
4033af6ab5fSopenharmony_ci
4043af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, classReg);
4053af6ab5fSopenharmony_ci
4063af6ab5fSopenharmony_ci    if (HasStaticPrivateMethod()) {
4073af6ab5fSopenharmony_ci        pg->StoreLexicalVar(this, 0, scope_->staticMethodValidation_);
4083af6ab5fSopenharmony_ci    }
4093af6ab5fSopenharmony_ci
4103af6ab5fSopenharmony_ci    InitializeClassName(pg);
4113af6ab5fSopenharmony_ci
4123af6ab5fSopenharmony_ci    CompileMissingProperties(pg, compiled, classReg);
4133af6ab5fSopenharmony_ci
4143af6ab5fSopenharmony_ci    if (hasPrivateElement_) {
4153af6ab5fSopenharmony_ci        int32_t bufIdx = CreateClassPrivateBuffer(pg);
4163af6ab5fSopenharmony_ci        pg->CreatePrivateProperty(this, scope_->privateFieldCnt_, bufIdx);
4173af6ab5fSopenharmony_ci    }
4183af6ab5fSopenharmony_ci
4193af6ab5fSopenharmony_ci    pg->LoadAccumulator(this, classReg);
4203af6ab5fSopenharmony_ci
4213af6ab5fSopenharmony_ci    if (NeedStaticInitializer()) {
4223af6ab5fSopenharmony_ci        StaticInitialize(pg, classReg);
4233af6ab5fSopenharmony_ci    }
4243af6ab5fSopenharmony_ci}
4253af6ab5fSopenharmony_ci
4263af6ab5fSopenharmony_cichecker::Type *ClassDefinition::Check(checker::Checker *checker) const
4273af6ab5fSopenharmony_ci{
4283af6ab5fSopenharmony_ci    return checker->GlobalAnyType();
4293af6ab5fSopenharmony_ci}
4303af6ab5fSopenharmony_ci
4313af6ab5fSopenharmony_civoid ClassDefinition::UpdateSelf(const NodeUpdater &cb, binder::Binder *binder)
4323af6ab5fSopenharmony_ci{
4333af6ab5fSopenharmony_ci    auto scopeCtx = binder::LexicalScope<binder::ClassScope>::Enter(binder, scope_);
4343af6ab5fSopenharmony_ci
4353af6ab5fSopenharmony_ci    if (ident_) {
4363af6ab5fSopenharmony_ci        ident_ = std::get<ir::AstNode *>(cb(ident_))->AsIdentifier();
4373af6ab5fSopenharmony_ci    }
4383af6ab5fSopenharmony_ci
4393af6ab5fSopenharmony_ci    if (typeParams_) {
4403af6ab5fSopenharmony_ci        typeParams_ = std::get<ir::AstNode *>(cb(typeParams_))->AsTSTypeParameterDeclaration();
4413af6ab5fSopenharmony_ci    }
4423af6ab5fSopenharmony_ci
4433af6ab5fSopenharmony_ci    if (superClass_) {
4443af6ab5fSopenharmony_ci        superClass_ = std::get<ir::AstNode *>(cb(superClass_))->AsExpression();
4453af6ab5fSopenharmony_ci    }
4463af6ab5fSopenharmony_ci
4473af6ab5fSopenharmony_ci    if (superTypeParams_) {
4483af6ab5fSopenharmony_ci        superTypeParams_ = std::get<ir::AstNode *>(cb(superTypeParams_))->AsTSTypeParameterInstantiation();
4493af6ab5fSopenharmony_ci    }
4503af6ab5fSopenharmony_ci
4513af6ab5fSopenharmony_ci    for (auto iter = implements_.begin(); iter != implements_.end(); iter++) {
4523af6ab5fSopenharmony_ci        *iter = std::get<ir::AstNode *>(cb(*iter))->AsTSClassImplements();
4533af6ab5fSopenharmony_ci    }
4543af6ab5fSopenharmony_ci
4553af6ab5fSopenharmony_ci    ctor_ = std::get<ir::AstNode *>(cb(ctor_))->AsMethodDefinition();
4563af6ab5fSopenharmony_ci
4573af6ab5fSopenharmony_ci    for (auto iter = body_.begin(); iter != body_.end(); iter++) {
4583af6ab5fSopenharmony_ci        *iter = std::get<ir::AstNode *>(cb(*iter))->AsStatement();
4593af6ab5fSopenharmony_ci    }
4603af6ab5fSopenharmony_ci
4613af6ab5fSopenharmony_ci    for (auto iter = indexSignatures_.begin(); iter != indexSignatures_.end(); iter++) {
4623af6ab5fSopenharmony_ci        *iter = std::get<ir::AstNode *>(cb(*iter))->AsTSIndexSignature();
4633af6ab5fSopenharmony_ci    }
4643af6ab5fSopenharmony_ci}
4653af6ab5fSopenharmony_ci
4663af6ab5fSopenharmony_ci
4673af6ab5fSopenharmony_civoid ClassDefinition::BuildClassEnvironment(bool useDefineSemantic)
4683af6ab5fSopenharmony_ci{
4693af6ab5fSopenharmony_ci    int instancePrivateMethodCnt = 0;
4703af6ab5fSopenharmony_ci    int staticPrivateMethodCnt = 0;
4713af6ab5fSopenharmony_ci    int privateFieldCnt = 0;
4723af6ab5fSopenharmony_ci    std::vector<const Statement *> privateProperties;
4733af6ab5fSopenharmony_ci    for (const auto *stmt : body_) {
4743af6ab5fSopenharmony_ci        if (stmt->IsMethodDefinition()) {
4753af6ab5fSopenharmony_ci            auto *methodDef = stmt->AsMethodDefinition();
4763af6ab5fSopenharmony_ci            if (methodDef->IsPrivate()) {
4773af6ab5fSopenharmony_ci                privateProperties.push_back(stmt);
4783af6ab5fSopenharmony_ci                methodDef->IsStatic() ? staticPrivateMethodCnt ++ : instancePrivateMethodCnt++;
4793af6ab5fSopenharmony_ci            } else if (methodDef->Computed()) {
4803af6ab5fSopenharmony_ci                hasComputedKey_ = true;
4813af6ab5fSopenharmony_ci            }
4823af6ab5fSopenharmony_ci            continue;
4833af6ab5fSopenharmony_ci        }
4843af6ab5fSopenharmony_ci
4853af6ab5fSopenharmony_ci        if (stmt->IsClassStaticBlock()) {
4863af6ab5fSopenharmony_ci            needStaticInitializer_ = true;
4873af6ab5fSopenharmony_ci            continue;
4883af6ab5fSopenharmony_ci        }
4893af6ab5fSopenharmony_ci
4903af6ab5fSopenharmony_ci        ASSERT(stmt->IsClassProperty());
4913af6ab5fSopenharmony_ci        const auto *prop = stmt->AsClassProperty();
4923af6ab5fSopenharmony_ci        // Do not process non-static public fields when not using define semantic.
4933af6ab5fSopenharmony_ci        if (!prop->IsPrivate() && !prop->IsStatic() && !useDefineSemantic) {
4943af6ab5fSopenharmony_ci            continue;
4953af6ab5fSopenharmony_ci        }
4963af6ab5fSopenharmony_ci
4973af6ab5fSopenharmony_ci        prop->IsStatic() ? needStaticInitializer_ = true : needInstanceInitializer_ = true;
4983af6ab5fSopenharmony_ci
4993af6ab5fSopenharmony_ci        if (prop->IsComputed() && prop->NeedCompileKey()) {
5003af6ab5fSopenharmony_ci            hasComputedKey_ = true;
5013af6ab5fSopenharmony_ci            scope_->AddClassVariable(prop->Key());
5023af6ab5fSopenharmony_ci        } else if (prop->IsPrivate()) {
5033af6ab5fSopenharmony_ci            privateFieldCnt++;
5043af6ab5fSopenharmony_ci            privateProperties.push_back(stmt);
5053af6ab5fSopenharmony_ci        }
5063af6ab5fSopenharmony_ci    }
5073af6ab5fSopenharmony_ci
5083af6ab5fSopenharmony_ci    if (!privateProperties.empty()) {
5093af6ab5fSopenharmony_ci        hasPrivateElement_ = true;
5103af6ab5fSopenharmony_ci        scope_->AddPrivateName(privateProperties, privateFieldCnt, instancePrivateMethodCnt, staticPrivateMethodCnt);
5113af6ab5fSopenharmony_ci    }
5123af6ab5fSopenharmony_ci
5133af6ab5fSopenharmony_ci    if (instancePrivateMethodCnt > 0) {
5143af6ab5fSopenharmony_ci        needInstanceInitializer_ = true;
5153af6ab5fSopenharmony_ci    }
5163af6ab5fSopenharmony_ci
5173af6ab5fSopenharmony_ci    if (NeedInstanceInitializer()) {
5183af6ab5fSopenharmony_ci        scope_->AddClassVariable(instanceInitializer_->Key());
5193af6ab5fSopenharmony_ci    }
5203af6ab5fSopenharmony_ci}
5213af6ab5fSopenharmony_ci
5223af6ab5fSopenharmony_civoid ClassDefinition::AddFieldType(FieldType &fieldType, const Expression *typeAnnotation,
5233af6ab5fSopenharmony_ci    compiler::PandaGen *pg) const
5243af6ab5fSopenharmony_ci{
5253af6ab5fSopenharmony_ci    switch (typeAnnotation->Type()) {
5263af6ab5fSopenharmony_ci        case AstNodeType::TS_NUMBER_KEYWORD: {
5273af6ab5fSopenharmony_ci            fieldType |= FieldType::NUMBER;
5283af6ab5fSopenharmony_ci            break;
5293af6ab5fSopenharmony_ci        }
5303af6ab5fSopenharmony_ci        case AstNodeType::TS_STRING_KEYWORD: {
5313af6ab5fSopenharmony_ci            fieldType |= FieldType::STRING;
5323af6ab5fSopenharmony_ci            break;
5333af6ab5fSopenharmony_ci        }
5343af6ab5fSopenharmony_ci        case AstNodeType::TS_BOOLEAN_KEYWORD: {
5353af6ab5fSopenharmony_ci            fieldType |= FieldType::BOOLEAN;
5363af6ab5fSopenharmony_ci            break;
5373af6ab5fSopenharmony_ci        }
5383af6ab5fSopenharmony_ci        case AstNodeType::TS_TYPE_REFERENCE: {
5393af6ab5fSopenharmony_ci            AddFieldTypeForTypeReference(typeAnnotation->AsTSTypeReference(), fieldType, pg);
5403af6ab5fSopenharmony_ci            break;
5413af6ab5fSopenharmony_ci        }
5423af6ab5fSopenharmony_ci        case AstNodeType::TS_BIGINT_KEYWORD: {
5433af6ab5fSopenharmony_ci            fieldType |= FieldType::BIGINT;
5443af6ab5fSopenharmony_ci            break;
5453af6ab5fSopenharmony_ci        }
5463af6ab5fSopenharmony_ci        case AstNodeType::TS_NULL_KEYWORD: {
5473af6ab5fSopenharmony_ci            fieldType |= FieldType::TS_NULL;
5483af6ab5fSopenharmony_ci            break;
5493af6ab5fSopenharmony_ci        }
5503af6ab5fSopenharmony_ci        case AstNodeType::TS_UNDEFINED_KEYWORD: {
5513af6ab5fSopenharmony_ci            fieldType |= FieldType::TS_UNDEFINED;
5523af6ab5fSopenharmony_ci            break;
5533af6ab5fSopenharmony_ci        }
5543af6ab5fSopenharmony_ci        default: {
5553af6ab5fSopenharmony_ci            UNREACHABLE();
5563af6ab5fSopenharmony_ci        }
5573af6ab5fSopenharmony_ci    }
5583af6ab5fSopenharmony_ci}
5593af6ab5fSopenharmony_ci
5603af6ab5fSopenharmony_civoid ClassDefinition::AddFieldTypeForTypeReference(const TSTypeReference *typeReference, FieldType &fieldType,
5613af6ab5fSopenharmony_ci    compiler::PandaGen *pg) const
5623af6ab5fSopenharmony_ci{
5633af6ab5fSopenharmony_ci    auto typeName = typeReference->TypeName();
5643af6ab5fSopenharmony_ci    ASSERT(typeName != nullptr);
5653af6ab5fSopenharmony_ci    if (!typeName->IsIdentifier()) {
5663af6ab5fSopenharmony_ci        fieldType |= FieldType::GENERIC;
5673af6ab5fSopenharmony_ci        return;
5683af6ab5fSopenharmony_ci    }
5693af6ab5fSopenharmony_ci
5703af6ab5fSopenharmony_ci    util::StringView propertyName = typeName->AsIdentifier()->Name();
5713af6ab5fSopenharmony_ci    binder::ScopeFindResult result = scope_->Find(propertyName);
5723af6ab5fSopenharmony_ci
5733af6ab5fSopenharmony_ci    // identify import type
5743af6ab5fSopenharmony_ci    parser::SourceTextModuleRecord *moduleRecord = pg->Binder()->Program()->TypeModuleRecord();
5753af6ab5fSopenharmony_ci    const auto &regularImportEntries = moduleRecord->GetRegularImportEntries();
5763af6ab5fSopenharmony_ci    if (regularImportEntries.find(propertyName) != regularImportEntries.end()) {
5773af6ab5fSopenharmony_ci        fieldType |= FieldType::GENERIC;
5783af6ab5fSopenharmony_ci        return;
5793af6ab5fSopenharmony_ci    }
5803af6ab5fSopenharmony_ci
5813af6ab5fSopenharmony_ci    if (IsTypeParam(propertyName) || (result.variable != nullptr && result.variable->IsModuleVariable())) {
5823af6ab5fSopenharmony_ci        fieldType |= FieldType::GENERIC;
5833af6ab5fSopenharmony_ci        return;
5843af6ab5fSopenharmony_ci    }
5853af6ab5fSopenharmony_ci
5863af6ab5fSopenharmony_ci    // ts enum type
5873af6ab5fSopenharmony_ci    const ir::AstNode *declNode = GetDeclNodeFromIdentifier(typeName->AsIdentifier());
5883af6ab5fSopenharmony_ci    if (declNode != nullptr) {
5893af6ab5fSopenharmony_ci        // If the result of "declNode->Original()" is nullptr, it means the declNode is original and not transformed.
5903af6ab5fSopenharmony_ci        auto originalNode = (declNode->Original() != nullptr) ? declNode->Original() : declNode;
5913af6ab5fSopenharmony_ci        if (originalNode->Type() == ir::AstNodeType::TS_ENUM_DECLARATION) {
5923af6ab5fSopenharmony_ci            fieldType |= FieldType::STRING | FieldType::NUMBER;
5933af6ab5fSopenharmony_ci            return;
5943af6ab5fSopenharmony_ci        }
5953af6ab5fSopenharmony_ci    }
5963af6ab5fSopenharmony_ci
5973af6ab5fSopenharmony_ci    // sendable class and sendable fuction
5983af6ab5fSopenharmony_ci    fieldType |= FieldType::TS_TYPE_REF;
5993af6ab5fSopenharmony_ci}
6003af6ab5fSopenharmony_ci
6013af6ab5fSopenharmony_ciconst ir::AstNode *ClassDefinition::GetDeclNodeFromIdentifier(const ir::Identifier *identifier) const
6023af6ab5fSopenharmony_ci{
6033af6ab5fSopenharmony_ci    if (identifier == nullptr) {
6043af6ab5fSopenharmony_ci        return nullptr;
6053af6ab5fSopenharmony_ci    }
6063af6ab5fSopenharmony_ci
6073af6ab5fSopenharmony_ci    std::vector<binder::Variable *> variables;
6083af6ab5fSopenharmony_ci    variables.reserve(identifier->TSVariables().size() + 1U);
6093af6ab5fSopenharmony_ci    variables.emplace_back(identifier->Variable());
6103af6ab5fSopenharmony_ci    for (const auto &v : identifier->TSVariables()) {
6113af6ab5fSopenharmony_ci        variables.emplace_back(v);
6123af6ab5fSopenharmony_ci    }
6133af6ab5fSopenharmony_ci
6143af6ab5fSopenharmony_ci    for (const auto &v : variables) {
6153af6ab5fSopenharmony_ci        if (v == nullptr || v->Declaration() == nullptr || v->Declaration()->Node() == nullptr) {
6163af6ab5fSopenharmony_ci            continue;
6173af6ab5fSopenharmony_ci        }
6183af6ab5fSopenharmony_ci
6193af6ab5fSopenharmony_ci        auto res = v->Declaration()->Node();
6203af6ab5fSopenharmony_ci        return res;
6213af6ab5fSopenharmony_ci    }
6223af6ab5fSopenharmony_ci    return nullptr;
6233af6ab5fSopenharmony_ci}
6243af6ab5fSopenharmony_ci
6253af6ab5fSopenharmony_cibool ClassDefinition::IsTypeParam(const util::StringView &propertyName) const
6263af6ab5fSopenharmony_ci{
6273af6ab5fSopenharmony_ci    if (typeParams_ == nullptr) {
6283af6ab5fSopenharmony_ci        return false;
6293af6ab5fSopenharmony_ci    }
6303af6ab5fSopenharmony_ci
6313af6ab5fSopenharmony_ci    for (auto param : typeParams_->Params()) {
6323af6ab5fSopenharmony_ci        util::StringView paramName = param->Name()->AsIdentifier()->Name();
6333af6ab5fSopenharmony_ci        if (paramName == propertyName) {
6343af6ab5fSopenharmony_ci            return true;
6353af6ab5fSopenharmony_ci        }
6363af6ab5fSopenharmony_ci    }
6373af6ab5fSopenharmony_ci    return false;
6383af6ab5fSopenharmony_ci}
6393af6ab5fSopenharmony_ci
6403af6ab5fSopenharmony_ciint32_t ClassDefinition::CreateFieldTypeBuffer(compiler::PandaGen *pg) const
6413af6ab5fSopenharmony_ci{
6423af6ab5fSopenharmony_ci    ASSERT(IsSendable());
6433af6ab5fSopenharmony_ci    auto *instanceBuf = pg->NewLiteralBuffer();
6443af6ab5fSopenharmony_ci    compiler::LiteralBuffer staticBuf(pg->Allocator());
6453af6ab5fSopenharmony_ci    uint32_t instanceFieldCnt {0};
6463af6ab5fSopenharmony_ci
6473af6ab5fSopenharmony_ci    for (auto *prop : body_) {
6483af6ab5fSopenharmony_ci        if (!prop->IsClassProperty()) {
6493af6ab5fSopenharmony_ci            continue;
6503af6ab5fSopenharmony_ci        }
6513af6ab5fSopenharmony_ci
6523af6ab5fSopenharmony_ci        auto *classProp = prop->AsClassProperty();
6533af6ab5fSopenharmony_ci        auto *buf = classProp->IsStatic() ? &staticBuf : (++instanceFieldCnt, instanceBuf);
6543af6ab5fSopenharmony_ci        auto name = util::Helpers::LiteralToPropName(pg->Allocator(), classProp->Key());
6553af6ab5fSopenharmony_ci        buf->Add(pg->Allocator()->New<StringLiteral>(name));
6563af6ab5fSopenharmony_ci
6573af6ab5fSopenharmony_ci        FieldType fieldType = FieldType::NONE;
6583af6ab5fSopenharmony_ci        const auto *typeAnnotation = classProp->TypeAnnotation();
6593af6ab5fSopenharmony_ci        if (typeAnnotation == nullptr) {
6603af6ab5fSopenharmony_ci            util::Helpers::ThrowError(ErrorType::GENERIC, pg->Binder()->Program(), prop->Start(),
6613af6ab5fSopenharmony_ci                "Field in sendable class must have type annotation");
6623af6ab5fSopenharmony_ci        }
6633af6ab5fSopenharmony_ci        if (typeAnnotation->IsTSUnionType()) {
6643af6ab5fSopenharmony_ci            for (const auto *type : typeAnnotation->AsTSUnionType()->Types()) {
6653af6ab5fSopenharmony_ci                AddFieldType(fieldType, type, pg);
6663af6ab5fSopenharmony_ci            }
6673af6ab5fSopenharmony_ci        } else {
6683af6ab5fSopenharmony_ci            AddFieldType(fieldType, typeAnnotation, pg);
6693af6ab5fSopenharmony_ci        }
6703af6ab5fSopenharmony_ci        buf->Add(pg->Allocator()->New<NumberLiteral>(static_cast<uint8_t>(fieldType)));
6713af6ab5fSopenharmony_ci    }
6723af6ab5fSopenharmony_ci
6733af6ab5fSopenharmony_ci    instanceBuf->Insert(&staticBuf);
6743af6ab5fSopenharmony_ci    instanceBuf->Add(pg->Allocator()->New<NumberLiteral>(instanceFieldCnt));
6753af6ab5fSopenharmony_ci    return pg->AddLiteralBuffer(instanceBuf);
6763af6ab5fSopenharmony_ci}
6773af6ab5fSopenharmony_ci
6783af6ab5fSopenharmony_civoid ClassDefinition::CompileSendableClass(compiler::PandaGen *pg) const
6793af6ab5fSopenharmony_ci{
6803af6ab5fSopenharmony_ci    compiler::RegScope rs(pg);
6813af6ab5fSopenharmony_ci    compiler::VReg classReg = pg->AllocReg();
6823af6ab5fSopenharmony_ci
6833af6ab5fSopenharmony_ci    compiler::LocalRegScope lrs(pg, scope_);
6843af6ab5fSopenharmony_ci
6853af6ab5fSopenharmony_ci    compiler::VReg baseReg = CompileHeritageClause(pg);
6863af6ab5fSopenharmony_ci    util::StringView ctorId = ctor_->Function()->Scope()->InternalName();
6873af6ab5fSopenharmony_ci    util::BitSet compiled(body_.size());
6883af6ab5fSopenharmony_ci
6893af6ab5fSopenharmony_ci    int32_t fieldTypeBufIdx = CreateFieldTypeBuffer(pg);
6903af6ab5fSopenharmony_ci    int32_t bufIdx = CreateClassPublicBuffer(pg, compiled, fieldTypeBufIdx);
6913af6ab5fSopenharmony_ci    pg->DefineSendableClass(this, ctorId, bufIdx, baseReg);
6923af6ab5fSopenharmony_ci
6933af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, classReg);
6943af6ab5fSopenharmony_ci
6953af6ab5fSopenharmony_ci    InitializeClassName(pg);
6963af6ab5fSopenharmony_ci
6973af6ab5fSopenharmony_ci    if (NeedStaticInitializer()) {
6983af6ab5fSopenharmony_ci        StaticInitialize(pg, classReg);
6993af6ab5fSopenharmony_ci    }
7003af6ab5fSopenharmony_ci}
7013af6ab5fSopenharmony_ci
7023af6ab5fSopenharmony_civoid ClassDefinition::CompileGetterOrSetter(compiler::PandaGen *pg, compiler::VReg dest,
7033af6ab5fSopenharmony_ci    const MethodDefinition *prop) const
7043af6ab5fSopenharmony_ci{
7053af6ab5fSopenharmony_ci    compiler::VReg keyReg = prop->Computed() ? prop->KeyReg() : pg->LoadPropertyKey(prop->Key(), false);
7063af6ab5fSopenharmony_ci
7073af6ab5fSopenharmony_ci    compiler::VReg undef = pg->AllocReg();
7083af6ab5fSopenharmony_ci    pg->LoadConst(this, compiler::Constant::JS_UNDEFINED);
7093af6ab5fSopenharmony_ci    pg->StoreAccumulator(this, undef);
7103af6ab5fSopenharmony_ci
7113af6ab5fSopenharmony_ci    compiler::VReg getter = undef;
7123af6ab5fSopenharmony_ci    compiler::VReg setter = undef;
7133af6ab5fSopenharmony_ci
7143af6ab5fSopenharmony_ci    pg->LoadAccumulator(this, dest);
7153af6ab5fSopenharmony_ci
7163af6ab5fSopenharmony_ci    compiler::VReg accessor = pg->AllocReg();
7173af6ab5fSopenharmony_ci    prop->Value()->Compile(pg);
7183af6ab5fSopenharmony_ci    pg->StoreAccumulator(prop->Value(), accessor);
7193af6ab5fSopenharmony_ci
7203af6ab5fSopenharmony_ci    if (prop->Kind() == ir::MethodDefinitionKind::GET) {
7213af6ab5fSopenharmony_ci        getter = accessor;
7223af6ab5fSopenharmony_ci    } else {
7233af6ab5fSopenharmony_ci        setter = accessor;
7243af6ab5fSopenharmony_ci    }
7253af6ab5fSopenharmony_ci
7263af6ab5fSopenharmony_ci    pg->DefineGetterSetterByValue(this, dest, keyReg, getter, setter, prop->Computed());
7273af6ab5fSopenharmony_ci}
7283af6ab5fSopenharmony_ci
7293af6ab5fSopenharmony_ci}  // namespace panda::es2panda::ir
730