13af6ab5fSopenharmony_ci/*
23af6ab5fSopenharmony_ci * Copyright (c) 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#ifndef ES2PANDA_EVALUATE_VARBINDER_SCOPES_H
173af6ab5fSopenharmony_ci#define ES2PANDA_EVALUATE_VARBINDER_SCOPES_H
183af6ab5fSopenharmony_ci
193af6ab5fSopenharmony_ci#include "parser/program/program.h"
203af6ab5fSopenharmony_ci#include "varbinder/ETSBinder.h"
213af6ab5fSopenharmony_ci#include "varbinder/recordTable.h"
223af6ab5fSopenharmony_ci
233af6ab5fSopenharmony_cinamespace ark::es2panda::evaluate {
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_ci// This scope must be used before running VarBinder or Checker on nodes from another Program.
263af6ab5fSopenharmony_ciclass ProgramScope final {
273af6ab5fSopenharmony_cipublic:
283af6ab5fSopenharmony_ci    explicit ProgramScope(varbinder::ETSBinder *varBinder, parser::Program *program)
293af6ab5fSopenharmony_ci        : varBinder_(varBinder),
303af6ab5fSopenharmony_ci          prevProgram_(varBinder->Program()),
313af6ab5fSopenharmony_ci          prevRecordTable_(varBinder->GetRecordTable()),
323af6ab5fSopenharmony_ci          prevTopScope_(varBinder->TopScope()),
333af6ab5fSopenharmony_ci          prevVarScope_(varBinder->VarScope()),
343af6ab5fSopenharmony_ci          prevScope_(varBinder->GetScope())
353af6ab5fSopenharmony_ci    {
363af6ab5fSopenharmony_ci        ASSERT(varBinder);
373af6ab5fSopenharmony_ci        ASSERT(program);
383af6ab5fSopenharmony_ci
393af6ab5fSopenharmony_ci        varBinder_->SetProgram(program);
403af6ab5fSopenharmony_ci
413af6ab5fSopenharmony_ci        auto &extTables = varBinder_->GetExternalRecordTable();
423af6ab5fSopenharmony_ci        auto iter = extTables.find(program);
433af6ab5fSopenharmony_ci        ASSERT(iter != extTables.end());
443af6ab5fSopenharmony_ci        varBinder_->SetRecordTable(iter->second);
453af6ab5fSopenharmony_ci
463af6ab5fSopenharmony_ci        varBinder_->ResetAllScopes(program->GlobalScope(), program->GlobalScope(), program->GlobalScope());
473af6ab5fSopenharmony_ci    }
483af6ab5fSopenharmony_ci
493af6ab5fSopenharmony_ci    ~ProgramScope() noexcept
503af6ab5fSopenharmony_ci    {
513af6ab5fSopenharmony_ci        varBinder_->SetProgram(prevProgram_);
523af6ab5fSopenharmony_ci        varBinder_->SetRecordTable(prevRecordTable_);
533af6ab5fSopenharmony_ci        varBinder_->ResetAllScopes(prevTopScope_, prevVarScope_, prevScope_);
543af6ab5fSopenharmony_ci    }
553af6ab5fSopenharmony_ci
563af6ab5fSopenharmony_ci    NO_COPY_SEMANTIC(ProgramScope);
573af6ab5fSopenharmony_ci    NO_MOVE_SEMANTIC(ProgramScope);
583af6ab5fSopenharmony_ci
593af6ab5fSopenharmony_ci    void *operator new(size_t) = delete;
603af6ab5fSopenharmony_ci    void *operator new[](size_t) = delete;
613af6ab5fSopenharmony_ci
623af6ab5fSopenharmony_ciprivate:
633af6ab5fSopenharmony_ci    varbinder::ETSBinder *varBinder_ {nullptr};
643af6ab5fSopenharmony_ci    parser::Program *prevProgram_ {nullptr};
653af6ab5fSopenharmony_ci    varbinder::RecordTable *prevRecordTable_ {nullptr};
663af6ab5fSopenharmony_ci    varbinder::GlobalScope *prevTopScope_ {nullptr};
673af6ab5fSopenharmony_ci    varbinder::VariableScope *prevVarScope_ {nullptr};
683af6ab5fSopenharmony_ci    varbinder::Scope *prevScope_ {nullptr};
693af6ab5fSopenharmony_ci};
703af6ab5fSopenharmony_ci
713af6ab5fSopenharmony_ci// The scope is required for running VarBinder or Checker on nodes from another class,
723af6ab5fSopenharmony_ci// so that entities will be registered with correct names in record table.
733af6ab5fSopenharmony_ciclass RecordTableClassScope final {
743af6ab5fSopenharmony_cipublic:
753af6ab5fSopenharmony_ci    explicit RecordTableClassScope(varbinder::ETSBinder *varBinder, ir::AstNode *recordClass) : varBinder_(varBinder)
763af6ab5fSopenharmony_ci    {
773af6ab5fSopenharmony_ci        ASSERT(varBinder_);
783af6ab5fSopenharmony_ci
793af6ab5fSopenharmony_ci        auto *recordTable = varBinder_->GetRecordTable();
803af6ab5fSopenharmony_ci        ASSERT(recordTable);
813af6ab5fSopenharmony_ci
823af6ab5fSopenharmony_ci        prevRecordClass_ = recordTable->ClassDefinition();
833af6ab5fSopenharmony_ci        if (prevRecordClass_ == nullptr) {
843af6ab5fSopenharmony_ci            prevRecordClass_ = recordTable->InterfaceDeclaration();
853af6ab5fSopenharmony_ci        }
863af6ab5fSopenharmony_ci
873af6ab5fSopenharmony_ci        if (recordClass != nullptr) {
883af6ab5fSopenharmony_ci            if (recordClass->IsClassDefinition()) {
893af6ab5fSopenharmony_ci                recordTable->SetClassDefinition(recordClass->AsClassDefinition());
903af6ab5fSopenharmony_ci            } else {
913af6ab5fSopenharmony_ci                recordTable->SetInterfaceDeclaration(recordClass->AsTSInterfaceDeclaration());
923af6ab5fSopenharmony_ci            }
933af6ab5fSopenharmony_ci        } else {
943af6ab5fSopenharmony_ci            ir::ClassDefinition *nullDef = nullptr;
953af6ab5fSopenharmony_ci            recordTable->SetClassDefinition(nullDef);
963af6ab5fSopenharmony_ci        }
973af6ab5fSopenharmony_ci    }
983af6ab5fSopenharmony_ci
993af6ab5fSopenharmony_ci    // NOLINTNEXTLINE(bugprone-exception-escape)
1003af6ab5fSopenharmony_ci    ~RecordTableClassScope() noexcept
1013af6ab5fSopenharmony_ci    {
1023af6ab5fSopenharmony_ci        auto *recordTable = varBinder_->GetRecordTable();
1033af6ab5fSopenharmony_ci        ASSERT(recordTable != nullptr);
1043af6ab5fSopenharmony_ci
1053af6ab5fSopenharmony_ci        if (prevRecordClass_ != nullptr) {
1063af6ab5fSopenharmony_ci            if (prevRecordClass_->IsClassDefinition()) {
1073af6ab5fSopenharmony_ci                recordTable->SetClassDefinition(prevRecordClass_->AsClassDefinition());
1083af6ab5fSopenharmony_ci            } else {
1093af6ab5fSopenharmony_ci                recordTable->SetInterfaceDeclaration(prevRecordClass_->AsTSInterfaceDeclaration());
1103af6ab5fSopenharmony_ci            }
1113af6ab5fSopenharmony_ci        } else {
1123af6ab5fSopenharmony_ci            ir::ClassDefinition *nullDef = nullptr;
1133af6ab5fSopenharmony_ci            recordTable->SetClassDefinition(nullDef);
1143af6ab5fSopenharmony_ci        }
1153af6ab5fSopenharmony_ci    }
1163af6ab5fSopenharmony_ci
1173af6ab5fSopenharmony_ci    NO_COPY_SEMANTIC(RecordTableClassScope);
1183af6ab5fSopenharmony_ci    NO_MOVE_SEMANTIC(RecordTableClassScope);
1193af6ab5fSopenharmony_ci
1203af6ab5fSopenharmony_ci    void *operator new(size_t) = delete;
1213af6ab5fSopenharmony_ci    void *operator new[](size_t) = delete;
1223af6ab5fSopenharmony_ci
1233af6ab5fSopenharmony_ciprivate:
1243af6ab5fSopenharmony_ci    varbinder::ETSBinder *varBinder_ {nullptr};
1253af6ab5fSopenharmony_ci    ir::AstNode *prevRecordClass_ {nullptr};
1263af6ab5fSopenharmony_ci};
1273af6ab5fSopenharmony_ci
1283af6ab5fSopenharmony_ci}  // namespace ark::es2panda::evaluate
1293af6ab5fSopenharmony_ci
1303af6ab5fSopenharmony_ci#endif  // ES2PANDA_EVALUATE_VARBINDER_SCOPES_H
131