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#ifndef ES2PANDA_COMPILER_SCOPES_DECLARATION_H
173af6ab5fSopenharmony_ci#define ES2PANDA_COMPILER_SCOPES_DECLARATION_H
183af6ab5fSopenharmony_ci
193af6ab5fSopenharmony_ci#include <binder/variableFlags.h>
203af6ab5fSopenharmony_ci#include <macros.h>
213af6ab5fSopenharmony_ci#include <util/ustring.h>
223af6ab5fSopenharmony_ci
233af6ab5fSopenharmony_cinamespace panda::es2panda::ir {
243af6ab5fSopenharmony_ciclass AstNode;
253af6ab5fSopenharmony_ciclass ScriptFunction;
263af6ab5fSopenharmony_ciclass TSInterfaceDeclaration;
273af6ab5fSopenharmony_ciclass TSModuleDeclaration;
283af6ab5fSopenharmony_ciclass TSEnumDeclaration;
293af6ab5fSopenharmony_ciclass ImportDeclaration;
303af6ab5fSopenharmony_ci}  // namespace panda::es2panda::ir
313af6ab5fSopenharmony_ci
323af6ab5fSopenharmony_cinamespace panda::es2panda::binder {
333af6ab5fSopenharmony_ci
343af6ab5fSopenharmony_ciclass Scope;
353af6ab5fSopenharmony_ciclass LocalScope;
363af6ab5fSopenharmony_ciclass TSEnumScope;
373af6ab5fSopenharmony_ci
383af6ab5fSopenharmony_ci#define DECLARE_CLASSES(decl_kind, className) class className;
393af6ab5fSopenharmony_ciDECLARATION_KINDS(DECLARE_CLASSES)
403af6ab5fSopenharmony_ci#undef DECLARE_CLASSES
413af6ab5fSopenharmony_ci
423af6ab5fSopenharmony_ciclass Decl {
433af6ab5fSopenharmony_cipublic:
443af6ab5fSopenharmony_ci    virtual ~Decl() = default;
453af6ab5fSopenharmony_ci    NO_COPY_SEMANTIC(Decl);
463af6ab5fSopenharmony_ci    NO_MOVE_SEMANTIC(Decl);
473af6ab5fSopenharmony_ci
483af6ab5fSopenharmony_ci    virtual DeclType Type() const = 0;
493af6ab5fSopenharmony_ci
503af6ab5fSopenharmony_ci    const util::StringView &Name() const
513af6ab5fSopenharmony_ci    {
523af6ab5fSopenharmony_ci        return name_;
533af6ab5fSopenharmony_ci    }
543af6ab5fSopenharmony_ci
553af6ab5fSopenharmony_ci    const ir::AstNode *Node() const
563af6ab5fSopenharmony_ci    {
573af6ab5fSopenharmony_ci        return node_;
583af6ab5fSopenharmony_ci    }
593af6ab5fSopenharmony_ci
603af6ab5fSopenharmony_ci#define DECLARE_CHECKS_CASTS(declKind, className)         \
613af6ab5fSopenharmony_ci    bool Is##className() const                            \
623af6ab5fSopenharmony_ci    {                                                     \
633af6ab5fSopenharmony_ci        return Type() == DeclType::declKind;              \
643af6ab5fSopenharmony_ci    }                                                     \
653af6ab5fSopenharmony_ci    className *As##className()                            \
663af6ab5fSopenharmony_ci    {                                                     \
673af6ab5fSopenharmony_ci        ASSERT(Is##className());                          \
683af6ab5fSopenharmony_ci        return reinterpret_cast<className *>(this);       \
693af6ab5fSopenharmony_ci    }                                                     \
703af6ab5fSopenharmony_ci    const className *As##className() const                \
713af6ab5fSopenharmony_ci    {                                                     \
723af6ab5fSopenharmony_ci        ASSERT(Is##className());                          \
733af6ab5fSopenharmony_ci        return reinterpret_cast<const className *>(this); \
743af6ab5fSopenharmony_ci    }
753af6ab5fSopenharmony_ci    DECLARATION_KINDS(DECLARE_CHECKS_CASTS)
763af6ab5fSopenharmony_ci#undef DECLARE_CHECKS_CASTS
773af6ab5fSopenharmony_ci
783af6ab5fSopenharmony_ci    void BindNode(const ir::AstNode *node)
793af6ab5fSopenharmony_ci    {
803af6ab5fSopenharmony_ci        node_ = node;
813af6ab5fSopenharmony_ci    }
823af6ab5fSopenharmony_ci
833af6ab5fSopenharmony_ci    bool IsLetOrConstOrClassDecl() const
843af6ab5fSopenharmony_ci    {
853af6ab5fSopenharmony_ci        return IsLetDecl() || IsConstDecl() || IsClassDecl();
863af6ab5fSopenharmony_ci    }
873af6ab5fSopenharmony_ci
883af6ab5fSopenharmony_ci    DeclarationFlags Flags() const
893af6ab5fSopenharmony_ci    {
903af6ab5fSopenharmony_ci        return flags_;
913af6ab5fSopenharmony_ci    }
923af6ab5fSopenharmony_ci
933af6ab5fSopenharmony_ci    void AddFlag(DeclarationFlags flag)
943af6ab5fSopenharmony_ci    {
953af6ab5fSopenharmony_ci        flags_ |= flag;
963af6ab5fSopenharmony_ci    }
973af6ab5fSopenharmony_ci
983af6ab5fSopenharmony_ci    bool HasFlag(DeclarationFlags flag) const
993af6ab5fSopenharmony_ci    {
1003af6ab5fSopenharmony_ci        return (flags_ & flag) != 0;
1013af6ab5fSopenharmony_ci    }
1023af6ab5fSopenharmony_ci
1033af6ab5fSopenharmony_ci    bool IsImportDecl() const
1043af6ab5fSopenharmony_ci    {
1053af6ab5fSopenharmony_ci        return HasFlag(DeclarationFlags::IMPORT);
1063af6ab5fSopenharmony_ci    }
1073af6ab5fSopenharmony_ci
1083af6ab5fSopenharmony_ci    bool IsImportOrExportDecl() const
1093af6ab5fSopenharmony_ci    {
1103af6ab5fSopenharmony_ci        return HasFlag(DeclarationFlags::IMPORT | DeclarationFlags::EXPORT);
1113af6ab5fSopenharmony_ci    }
1123af6ab5fSopenharmony_ci
1133af6ab5fSopenharmony_ci    bool IsExportDeclInTsModule() const
1143af6ab5fSopenharmony_ci    {
1153af6ab5fSopenharmony_ci        return HasFlag(DeclarationFlags::EXPORT_IN_TSMODULE);
1163af6ab5fSopenharmony_ci    }
1173af6ab5fSopenharmony_ci
1183af6ab5fSopenharmony_ci    void SetDeclare(bool isDeclare)
1193af6ab5fSopenharmony_ci    {
1203af6ab5fSopenharmony_ci        isDeclare_ = isDeclare;
1213af6ab5fSopenharmony_ci    }
1223af6ab5fSopenharmony_ci
1233af6ab5fSopenharmony_ci    bool IsDeclare() const
1243af6ab5fSopenharmony_ci    {
1253af6ab5fSopenharmony_ci        return isDeclare_;
1263af6ab5fSopenharmony_ci    }
1273af6ab5fSopenharmony_ci
1283af6ab5fSopenharmony_ci    bool IsSendableClassDecl() const;
1293af6ab5fSopenharmony_ci
1303af6ab5fSopenharmony_ci    bool IsSendableFunctionDecl() const;
1313af6ab5fSopenharmony_ci
1323af6ab5fSopenharmony_ci    bool NeedSetInSendableEnv(Scope *scope) const;
1333af6ab5fSopenharmony_ci
1343af6ab5fSopenharmony_ciprotected:
1353af6ab5fSopenharmony_ci    explicit Decl(util::StringView name) : name_(name) {}
1363af6ab5fSopenharmony_ci
1373af6ab5fSopenharmony_ci    util::StringView name_;
1383af6ab5fSopenharmony_ci    DeclarationFlags flags_ {};
1393af6ab5fSopenharmony_ci    const ir::AstNode *node_ {};
1403af6ab5fSopenharmony_ci    bool isDeclare_ {false};
1413af6ab5fSopenharmony_ci};
1423af6ab5fSopenharmony_ci
1433af6ab5fSopenharmony_citemplate <typename T>
1443af6ab5fSopenharmony_ciclass MultiDecl : public Decl {
1453af6ab5fSopenharmony_cipublic:
1463af6ab5fSopenharmony_ci    explicit MultiDecl(ArenaAllocator *allocator, util::StringView name)
1473af6ab5fSopenharmony_ci        : Decl(name), declarations_(allocator->Adapter())
1483af6ab5fSopenharmony_ci    {
1493af6ab5fSopenharmony_ci    }
1503af6ab5fSopenharmony_ci
1513af6ab5fSopenharmony_ci    const ArenaVector<T *> &Decls() const
1523af6ab5fSopenharmony_ci    {
1533af6ab5fSopenharmony_ci        return declarations_;
1543af6ab5fSopenharmony_ci    }
1553af6ab5fSopenharmony_ci
1563af6ab5fSopenharmony_ci    void Add(T *decl)
1573af6ab5fSopenharmony_ci    {
1583af6ab5fSopenharmony_ci        declarations_.push_back(decl);
1593af6ab5fSopenharmony_ci    }
1603af6ab5fSopenharmony_ci
1613af6ab5fSopenharmony_ciprivate:
1623af6ab5fSopenharmony_ci    ArenaVector<T *> declarations_;
1633af6ab5fSopenharmony_ci};
1643af6ab5fSopenharmony_ci
1653af6ab5fSopenharmony_ciclass EnumLiteralDecl : public MultiDecl<ir::TSEnumDeclaration> {
1663af6ab5fSopenharmony_cipublic:
1673af6ab5fSopenharmony_ci    explicit EnumLiteralDecl(ArenaAllocator *allocator, util::StringView name, bool isExport, bool isConst)
1683af6ab5fSopenharmony_ci        : MultiDecl(allocator, name), isExport_(isExport), isConst_(isConst) {}
1693af6ab5fSopenharmony_ci
1703af6ab5fSopenharmony_ci    DeclType Type() const override
1713af6ab5fSopenharmony_ci    {
1723af6ab5fSopenharmony_ci        return DeclType::ENUM_LITERAL;
1733af6ab5fSopenharmony_ci    }
1743af6ab5fSopenharmony_ci
1753af6ab5fSopenharmony_ci    bool IsExport() const
1763af6ab5fSopenharmony_ci    {
1773af6ab5fSopenharmony_ci        return isExport_;
1783af6ab5fSopenharmony_ci    }
1793af6ab5fSopenharmony_ci
1803af6ab5fSopenharmony_ci    bool IsConst() const
1813af6ab5fSopenharmony_ci    {
1823af6ab5fSopenharmony_ci        return isConst_;
1833af6ab5fSopenharmony_ci    }
1843af6ab5fSopenharmony_ci
1853af6ab5fSopenharmony_ci    void BindScope(TSEnumScope *scope)
1863af6ab5fSopenharmony_ci    {
1873af6ab5fSopenharmony_ci        scope_ = scope;
1883af6ab5fSopenharmony_ci    }
1893af6ab5fSopenharmony_ci
1903af6ab5fSopenharmony_ci    TSEnumScope *Scope()
1913af6ab5fSopenharmony_ci    {
1923af6ab5fSopenharmony_ci        return scope_;
1933af6ab5fSopenharmony_ci    }
1943af6ab5fSopenharmony_ci
1953af6ab5fSopenharmony_ciprivate:
1963af6ab5fSopenharmony_ci    TSEnumScope *scope_ {nullptr};
1973af6ab5fSopenharmony_ci    bool isExport_ {};
1983af6ab5fSopenharmony_ci    bool isConst_ {};
1993af6ab5fSopenharmony_ci};
2003af6ab5fSopenharmony_ci
2013af6ab5fSopenharmony_ciclass InterfaceDecl : public MultiDecl<ir::TSInterfaceDeclaration> {
2023af6ab5fSopenharmony_cipublic:
2033af6ab5fSopenharmony_ci    explicit InterfaceDecl(ArenaAllocator *allocator, util::StringView name) : MultiDecl(allocator, name) {}
2043af6ab5fSopenharmony_ci
2053af6ab5fSopenharmony_ci    DeclType Type() const override
2063af6ab5fSopenharmony_ci    {
2073af6ab5fSopenharmony_ci        return DeclType::INTERFACE;
2083af6ab5fSopenharmony_ci    }
2093af6ab5fSopenharmony_ci};
2103af6ab5fSopenharmony_ci
2113af6ab5fSopenharmony_ciclass TypeParameterDecl : public Decl {
2123af6ab5fSopenharmony_cipublic:
2133af6ab5fSopenharmony_ci    explicit TypeParameterDecl(util::StringView name, const ir::AstNode *node);
2143af6ab5fSopenharmony_ci
2153af6ab5fSopenharmony_ci    DeclType Type() const override
2163af6ab5fSopenharmony_ci    {
2173af6ab5fSopenharmony_ci        return DeclType::TYPE_PARAMETER;
2183af6ab5fSopenharmony_ci    }
2193af6ab5fSopenharmony_ci};
2203af6ab5fSopenharmony_ci
2213af6ab5fSopenharmony_ciclass PropertyDecl : public Decl {
2223af6ab5fSopenharmony_cipublic:
2233af6ab5fSopenharmony_ci    explicit PropertyDecl(util::StringView name) : Decl(name) {}
2243af6ab5fSopenharmony_ci
2253af6ab5fSopenharmony_ci    DeclType Type() const override
2263af6ab5fSopenharmony_ci    {
2273af6ab5fSopenharmony_ci        return DeclType::PROPERTY;
2283af6ab5fSopenharmony_ci    }
2293af6ab5fSopenharmony_ci};
2303af6ab5fSopenharmony_ci
2313af6ab5fSopenharmony_ciclass MethodDecl : public Decl {
2323af6ab5fSopenharmony_cipublic:
2333af6ab5fSopenharmony_ci    explicit MethodDecl(util::StringView name) : Decl(name) {}
2343af6ab5fSopenharmony_ci
2353af6ab5fSopenharmony_ci    DeclType Type() const override
2363af6ab5fSopenharmony_ci    {
2373af6ab5fSopenharmony_ci        return DeclType::METHOD;
2383af6ab5fSopenharmony_ci    }
2393af6ab5fSopenharmony_ci};
2403af6ab5fSopenharmony_ci
2413af6ab5fSopenharmony_ciclass EnumDecl : public Decl {
2423af6ab5fSopenharmony_cipublic:
2433af6ab5fSopenharmony_ci    explicit EnumDecl(util::StringView name) : Decl(name) {}
2443af6ab5fSopenharmony_ci
2453af6ab5fSopenharmony_ci    DeclType Type() const override
2463af6ab5fSopenharmony_ci    {
2473af6ab5fSopenharmony_ci        return DeclType::ENUM;
2483af6ab5fSopenharmony_ci    }
2493af6ab5fSopenharmony_ci};
2503af6ab5fSopenharmony_ci
2513af6ab5fSopenharmony_ciclass TypeAliasDecl : public Decl {
2523af6ab5fSopenharmony_cipublic:
2533af6ab5fSopenharmony_ci    explicit TypeAliasDecl(util::StringView name) : Decl(name) {}
2543af6ab5fSopenharmony_ci
2553af6ab5fSopenharmony_ci    DeclType Type() const override
2563af6ab5fSopenharmony_ci    {
2573af6ab5fSopenharmony_ci        return DeclType::TYPE_ALIAS;
2583af6ab5fSopenharmony_ci    }
2593af6ab5fSopenharmony_ci};
2603af6ab5fSopenharmony_ci
2613af6ab5fSopenharmony_ciclass NamespaceDecl : public MultiDecl<ir::TSModuleDeclaration> {
2623af6ab5fSopenharmony_cipublic:
2633af6ab5fSopenharmony_ci    explicit NamespaceDecl(ArenaAllocator *allocator, util::StringView name)
2643af6ab5fSopenharmony_ci        : MultiDecl(allocator, name)
2653af6ab5fSopenharmony_ci    {
2663af6ab5fSopenharmony_ci    }
2673af6ab5fSopenharmony_ci
2683af6ab5fSopenharmony_ci    DeclType Type() const override
2693af6ab5fSopenharmony_ci    {
2703af6ab5fSopenharmony_ci        return DeclType::NAMESPACE;
2713af6ab5fSopenharmony_ci    }
2723af6ab5fSopenharmony_ci
2733af6ab5fSopenharmony_ci    bool IsInstantiated() const;
2743af6ab5fSopenharmony_ci};
2753af6ab5fSopenharmony_ci
2763af6ab5fSopenharmony_ciclass VarDecl : public Decl {
2773af6ab5fSopenharmony_cipublic:
2783af6ab5fSopenharmony_ci    explicit VarDecl(util::StringView name) : Decl(name) {}
2793af6ab5fSopenharmony_ci
2803af6ab5fSopenharmony_ci    DeclType Type() const override
2813af6ab5fSopenharmony_ci    {
2823af6ab5fSopenharmony_ci        return DeclType::VAR;
2833af6ab5fSopenharmony_ci    }
2843af6ab5fSopenharmony_ci};
2853af6ab5fSopenharmony_ci
2863af6ab5fSopenharmony_ciclass LetDecl : public Decl {
2873af6ab5fSopenharmony_cipublic:
2883af6ab5fSopenharmony_ci    explicit LetDecl(util::StringView name) : Decl(name) {}
2893af6ab5fSopenharmony_ci
2903af6ab5fSopenharmony_ci    DeclType Type() const override
2913af6ab5fSopenharmony_ci    {
2923af6ab5fSopenharmony_ci        return DeclType::LET;
2933af6ab5fSopenharmony_ci    }
2943af6ab5fSopenharmony_ci};
2953af6ab5fSopenharmony_ci
2963af6ab5fSopenharmony_ciclass ConstDecl : public Decl {
2973af6ab5fSopenharmony_cipublic:
2983af6ab5fSopenharmony_ci    explicit ConstDecl(util::StringView name) : Decl(name) {}
2993af6ab5fSopenharmony_ci
3003af6ab5fSopenharmony_ci    DeclType Type() const override
3013af6ab5fSopenharmony_ci    {
3023af6ab5fSopenharmony_ci        return DeclType::CONST;
3033af6ab5fSopenharmony_ci    }
3043af6ab5fSopenharmony_ci};
3053af6ab5fSopenharmony_ci
3063af6ab5fSopenharmony_ciclass ClassDecl : public Decl {
3073af6ab5fSopenharmony_cipublic:
3083af6ab5fSopenharmony_ci    explicit ClassDecl(util::StringView name) : Decl(name) {}
3093af6ab5fSopenharmony_ci
3103af6ab5fSopenharmony_ci    DeclType Type() const override
3113af6ab5fSopenharmony_ci    {
3123af6ab5fSopenharmony_ci        return DeclType::CLASS;
3133af6ab5fSopenharmony_ci    }
3143af6ab5fSopenharmony_ci};
3153af6ab5fSopenharmony_ci
3163af6ab5fSopenharmony_ciclass FunctionDecl : public MultiDecl<ir::ScriptFunction> {
3173af6ab5fSopenharmony_cipublic:
3183af6ab5fSopenharmony_ci    explicit FunctionDecl(ArenaAllocator *allocator, util::StringView name, const ir::AstNode *node)
3193af6ab5fSopenharmony_ci        : MultiDecl(allocator, name)
3203af6ab5fSopenharmony_ci    {
3213af6ab5fSopenharmony_ci        node_ = node;
3223af6ab5fSopenharmony_ci    }
3233af6ab5fSopenharmony_ci
3243af6ab5fSopenharmony_ci    DeclType Type() const override
3253af6ab5fSopenharmony_ci    {
3263af6ab5fSopenharmony_ci        return DeclType::FUNC;
3273af6ab5fSopenharmony_ci    }
3283af6ab5fSopenharmony_ci
3293af6ab5fSopenharmony_ci    void SetDeclClass(ClassDecl *declClass)
3303af6ab5fSopenharmony_ci    {
3313af6ab5fSopenharmony_ci        declClass_ = declClass;
3323af6ab5fSopenharmony_ci    }
3333af6ab5fSopenharmony_ci
3343af6ab5fSopenharmony_ci    ClassDecl *GetDeclClass()
3353af6ab5fSopenharmony_ci    {
3363af6ab5fSopenharmony_ci        return declClass_;
3373af6ab5fSopenharmony_ci    }
3383af6ab5fSopenharmony_ci
3393af6ab5fSopenharmony_ciprivate:
3403af6ab5fSopenharmony_ci    ClassDecl *declClass_ {nullptr};
3413af6ab5fSopenharmony_ci};
3423af6ab5fSopenharmony_ci
3433af6ab5fSopenharmony_ciclass ParameterDecl : public Decl {
3443af6ab5fSopenharmony_cipublic:
3453af6ab5fSopenharmony_ci    explicit ParameterDecl(util::StringView name) : Decl(name) {}
3463af6ab5fSopenharmony_ci
3473af6ab5fSopenharmony_ci    DeclType Type() const override
3483af6ab5fSopenharmony_ci    {
3493af6ab5fSopenharmony_ci        return DeclType::PARAM;
3503af6ab5fSopenharmony_ci    }
3513af6ab5fSopenharmony_ci};
3523af6ab5fSopenharmony_ci
3533af6ab5fSopenharmony_ciclass ImportEqualsDecl : public Decl {
3543af6ab5fSopenharmony_cipublic:
3553af6ab5fSopenharmony_ci    explicit ImportEqualsDecl(util::StringView name) : Decl(name) {}
3563af6ab5fSopenharmony_ci
3573af6ab5fSopenharmony_ci    DeclType Type() const override
3583af6ab5fSopenharmony_ci    {
3593af6ab5fSopenharmony_ci        return DeclType::IMPORT_EQUALS;
3603af6ab5fSopenharmony_ci    }
3613af6ab5fSopenharmony_ci};
3623af6ab5fSopenharmony_ci
3633af6ab5fSopenharmony_ci}  // namespace panda::es2panda::binder
3643af6ab5fSopenharmony_ci
3653af6ab5fSopenharmony_ci#endif
366