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 "hoisting.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include <binder/binder.h>
193af6ab5fSopenharmony_ci#include <binder/scope.h>
203af6ab5fSopenharmony_ci#include <compiler/core/pandagen.h>
213af6ab5fSopenharmony_ci#include <ir/base/scriptFunction.h>
223af6ab5fSopenharmony_ci#include <ir/statements/variableDeclaration.h>
233af6ab5fSopenharmony_ci#include <parser/module/sourceTextModuleRecord.h>
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_cinamespace panda::es2panda::compiler {
263af6ab5fSopenharmony_ci
273af6ab5fSopenharmony_cistatic void StoreModuleVarOrLocalVar(PandaGen *pg, binder::ScopeFindResult &result, const binder::Decl *decl)
283af6ab5fSopenharmony_ci{
293af6ab5fSopenharmony_ci    if (decl->IsImportOrExportDecl()) {
303af6ab5fSopenharmony_ci        ASSERT(pg->Scope()->IsModuleScope());
313af6ab5fSopenharmony_ci        auto *var = pg->Scope()->FindLocal(decl->Name());
323af6ab5fSopenharmony_ci        CHECK_NOT_NULL(var);
333af6ab5fSopenharmony_ci        ASSERT(var->IsModuleVariable());
343af6ab5fSopenharmony_ci        pg->StoreModuleVariable(decl->Node(), var->AsModuleVariable());
353af6ab5fSopenharmony_ci    } else {
363af6ab5fSopenharmony_ci        pg->StoreAccToLexEnv(decl->Node(), result, true);
373af6ab5fSopenharmony_ci    }
383af6ab5fSopenharmony_ci}
393af6ab5fSopenharmony_ci
403af6ab5fSopenharmony_cistatic void HoistVar(PandaGen *pg, binder::Variable *var, const binder::VarDecl *decl)
413af6ab5fSopenharmony_ci{
423af6ab5fSopenharmony_ci    if (decl->IsDeclare()) {
433af6ab5fSopenharmony_ci        return;
443af6ab5fSopenharmony_ci    }
453af6ab5fSopenharmony_ci
463af6ab5fSopenharmony_ci    auto *scope = pg->Scope();
473af6ab5fSopenharmony_ci    if (scope->IsGlobalScope()) {
483af6ab5fSopenharmony_ci        pg->LoadConst(decl->Node(), Constant::JS_UNDEFINED);
493af6ab5fSopenharmony_ci        pg->StoreGlobalVar(decl->Node(), decl->Name());
503af6ab5fSopenharmony_ci        return;
513af6ab5fSopenharmony_ci    }
523af6ab5fSopenharmony_ci
533af6ab5fSopenharmony_ci    auto *funcScope = scope->EnclosingFunctionVariableScope();
543af6ab5fSopenharmony_ci    CHECK_NOT_NULL(funcScope);
553af6ab5fSopenharmony_ci    if (scope->HasParamScope() && funcScope->ParamScope()->HasParam(decl->Name())) {
563af6ab5fSopenharmony_ci        return;
573af6ab5fSopenharmony_ci    }
583af6ab5fSopenharmony_ci
593af6ab5fSopenharmony_ci    binder::ScopeFindResult result(decl->Name(), scope, 0, var);
603af6ab5fSopenharmony_ci
613af6ab5fSopenharmony_ci    pg->LoadConst(decl->Node(), Constant::JS_UNDEFINED);
623af6ab5fSopenharmony_ci    StoreModuleVarOrLocalVar(pg, result, decl);
633af6ab5fSopenharmony_ci}
643af6ab5fSopenharmony_ci
653af6ab5fSopenharmony_cistatic void HoistFunction(PandaGen *pg, binder::Variable *var, const binder::FunctionDecl *decl)
663af6ab5fSopenharmony_ci{
673af6ab5fSopenharmony_ci    const ir::ScriptFunction *scriptFunction = decl->Node()->AsScriptFunction();
683af6ab5fSopenharmony_ci    if (scriptFunction->Declare()) {
693af6ab5fSopenharmony_ci        return;
703af6ab5fSopenharmony_ci    }
713af6ab5fSopenharmony_ci
723af6ab5fSopenharmony_ci    const auto &internalName = scriptFunction->Scope()->InternalName();
733af6ab5fSopenharmony_ci    auto *scope = pg->Scope();
743af6ab5fSopenharmony_ci    if (scope->IsGlobalScope()) {
753af6ab5fSopenharmony_ci        pg->DefineFunction(decl->Node(), scriptFunction, internalName);
763af6ab5fSopenharmony_ci        pg->StoreGlobalVar(decl->Node(), var->Declaration()->Name());
773af6ab5fSopenharmony_ci        return;
783af6ab5fSopenharmony_ci    }
793af6ab5fSopenharmony_ci
803af6ab5fSopenharmony_ci    ASSERT(scope->IsFunctionScope() || scope->IsCatchScope() || scope->IsLocalScope() ||
813af6ab5fSopenharmony_ci           scope->IsModuleScope() || scope->IsTSModuleScope() || scope->IsTSEnumScope() ||
823af6ab5fSopenharmony_ci           scope->IsStaticBlockScope());
833af6ab5fSopenharmony_ci    binder::ScopeFindResult result(decl->Name(), scope, 0, var);
843af6ab5fSopenharmony_ci
853af6ab5fSopenharmony_ci    pg->DefineFunction(decl->Node(), scriptFunction, internalName);
863af6ab5fSopenharmony_ci    StoreModuleVarOrLocalVar(pg, result, decl);
873af6ab5fSopenharmony_ci}
883af6ab5fSopenharmony_ci
893af6ab5fSopenharmony_cistatic void HoistNameSpaceImports(PandaGen *pg)
903af6ab5fSopenharmony_ci{
913af6ab5fSopenharmony_ci    if (pg->Scope()->IsModuleScope()) {
923af6ab5fSopenharmony_ci        parser::SourceTextModuleRecord *moduleRecord = pg->Binder()->Program()->ModuleRecord();
933af6ab5fSopenharmony_ci        ASSERT(moduleRecord != nullptr);
943af6ab5fSopenharmony_ci        for (auto nameSpaceEntry : moduleRecord->GetNamespaceImportEntries()) {
953af6ab5fSopenharmony_ci            auto *var = pg->TopScope()->FindLocal(nameSpaceEntry->localName_);
963af6ab5fSopenharmony_ci            CHECK_NOT_NULL(var);
973af6ab5fSopenharmony_ci            auto *node = var->Declaration()->Node();
983af6ab5fSopenharmony_ci            CHECK_NOT_NULL(node);
993af6ab5fSopenharmony_ci            pg->GetModuleNamespace(node, nameSpaceEntry->moduleRequestIdx_);
1003af6ab5fSopenharmony_ci            pg->StoreVar(node, {nameSpaceEntry->localName_, pg->TopScope(), 0, var}, true);
1013af6ab5fSopenharmony_ci        }
1023af6ab5fSopenharmony_ci    }
1033af6ab5fSopenharmony_ci}
1043af6ab5fSopenharmony_ci
1053af6ab5fSopenharmony_civoid Hoisting::Hoist(PandaGen *pg)
1063af6ab5fSopenharmony_ci{
1073af6ab5fSopenharmony_ci    const auto *scope = pg->Scope();
1083af6ab5fSopenharmony_ci
1093af6ab5fSopenharmony_ci    for (const auto &[_, var] : scope->Bindings()) {
1103af6ab5fSopenharmony_ci        (void)_;
1113af6ab5fSopenharmony_ci        if (!var->HasFlag(binder::VariableFlags::HOIST)) {
1123af6ab5fSopenharmony_ci            continue;
1133af6ab5fSopenharmony_ci        }
1143af6ab5fSopenharmony_ci
1153af6ab5fSopenharmony_ci        const auto *decl = var->Declaration();
1163af6ab5fSopenharmony_ci
1173af6ab5fSopenharmony_ci        if (decl->IsVarDecl()) {
1183af6ab5fSopenharmony_ci            HoistVar(pg, var, decl->AsVarDecl());
1193af6ab5fSopenharmony_ci        } else {
1203af6ab5fSopenharmony_ci            ASSERT(decl->IsFunctionDecl());
1213af6ab5fSopenharmony_ci            HoistFunction(pg, var, decl->AsFunctionDecl());
1223af6ab5fSopenharmony_ci        }
1233af6ab5fSopenharmony_ci    }
1243af6ab5fSopenharmony_ci
1253af6ab5fSopenharmony_ci    HoistNameSpaceImports(pg);
1263af6ab5fSopenharmony_ci}
1273af6ab5fSopenharmony_ci
1283af6ab5fSopenharmony_ci}  // namespace panda::es2panda::compiler
129