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#include "evaluate/helpers.h"
173af6ab5fSopenharmony_ci#include "evaluate/irCheckHelper.h"
183af6ab5fSopenharmony_ci#include "compiler/lowering/util.h"
193af6ab5fSopenharmony_ci#include "compiler/lowering/scopesInit/scopesInitPhase.h"
203af6ab5fSopenharmony_ci#include "checker/ETSchecker.h"
213af6ab5fSopenharmony_ci
223af6ab5fSopenharmony_cinamespace ark::es2panda::evaluate {
233af6ab5fSopenharmony_ci
243af6ab5fSopenharmony_ciIrCheckHelper::IrCheckHelper(checker::ETSChecker *checker, varbinder::ETSBinder *varBinder)
253af6ab5fSopenharmony_ci    : checker_(checker), varBinder_(varBinder), recursiveDecls_(checker->Allocator()->Adapter())
263af6ab5fSopenharmony_ci{
273af6ab5fSopenharmony_ci    ASSERT(checker_);
283af6ab5fSopenharmony_ci    ASSERT(varBinder_);
293af6ab5fSopenharmony_ci}
303af6ab5fSopenharmony_ci
313af6ab5fSopenharmony_cibool IrCheckHelper::CheckNewNode(ir::AstNode *node, varbinder::Scope *scope, ir::AstNode *parentClass,
323af6ab5fSopenharmony_ci                                 parser::Program *program)
333af6ab5fSopenharmony_ci{
343af6ab5fSopenharmony_ci    ASSERT(node);
353af6ab5fSopenharmony_ci
363af6ab5fSopenharmony_ci    if (program == nullptr) {
373af6ab5fSopenharmony_ci        program = varBinder_->Program();
383af6ab5fSopenharmony_ci    }
393af6ab5fSopenharmony_ci    if (scope == nullptr) {
403af6ab5fSopenharmony_ci        scope = checker_->Scope();
413af6ab5fSopenharmony_ci    }
423af6ab5fSopenharmony_ci
433af6ab5fSopenharmony_ci    recursiveDecls_.emplace_back(program, scope, parentClass, node);
443af6ab5fSopenharmony_ci
453af6ab5fSopenharmony_ci    if (isRecursive_) {
463af6ab5fSopenharmony_ci        return false;
473af6ab5fSopenharmony_ci    }
483af6ab5fSopenharmony_ci    isRecursive_ = true;
493af6ab5fSopenharmony_ci
503af6ab5fSopenharmony_ci    if (isPrecheckPassed_) {
513af6ab5fSopenharmony_ci        HandleCustomNodes();
523af6ab5fSopenharmony_ci        CheckDecls();
533af6ab5fSopenharmony_ci    }
543af6ab5fSopenharmony_ci
553af6ab5fSopenharmony_ci    isRecursive_ = false;
563af6ab5fSopenharmony_ci
573af6ab5fSopenharmony_ci    return true;
583af6ab5fSopenharmony_ci}
593af6ab5fSopenharmony_ci
603af6ab5fSopenharmony_civoid IrCheckHelper::PreCheck()
613af6ab5fSopenharmony_ci{
623af6ab5fSopenharmony_ci    HandleCustomNodes();
633af6ab5fSopenharmony_ci    CheckDecls();
643af6ab5fSopenharmony_ci
653af6ab5fSopenharmony_ci    isPrecheckPassed_ = true;
663af6ab5fSopenharmony_ci}
673af6ab5fSopenharmony_ci
683af6ab5fSopenharmony_civoid IrCheckHelper::CheckDecls()
693af6ab5fSopenharmony_ci{
703af6ab5fSopenharmony_ci    // All dependent user-classes must be created at this point, so we can run checker.
713af6ab5fSopenharmony_ci    while (!recursiveDecls_.empty()) {
723af6ab5fSopenharmony_ci        auto [program, scope, parent, node] = recursiveDecls_.front();
733af6ab5fSopenharmony_ci        recursiveDecls_.pop_front();
743af6ab5fSopenharmony_ci        helpers::DoScopedAction(checker_, varBinder_, program, scope, parent, [this, node = node, scope = scope]() {
753af6ab5fSopenharmony_ci            varBinder_->ResolveReferencesForScope(node, scope);
763af6ab5fSopenharmony_ci            node->Check(checker_);
773af6ab5fSopenharmony_ci        });
783af6ab5fSopenharmony_ci    }
793af6ab5fSopenharmony_ci}
803af6ab5fSopenharmony_ci
813af6ab5fSopenharmony_civoid IrCheckHelper::HandleCustomNodes()
823af6ab5fSopenharmony_ci{
833af6ab5fSopenharmony_ci    auto iter = recursiveDecls_.begin();
843af6ab5fSopenharmony_ci    while (iter != recursiveDecls_.end()) {
853af6ab5fSopenharmony_ci        // Can trigger `ETSBinder::BuildClassDefinition`,
863af6ab5fSopenharmony_ci        // which can eventually call debug-info plugin to create another class.
873af6ab5fSopenharmony_ci        // Hence we delay `ETSChecker::Check` until all required classes are built and initialized in varbinder.
883af6ab5fSopenharmony_ci        auto [program, scope, parent, node] = *iter;
893af6ab5fSopenharmony_ci        helpers::DoScopedAction(checker_, varBinder_, program, scope, parent,
903af6ab5fSopenharmony_ci                                [varBinder = varBinder_, node = node]() { varBinder->HandleCustomNodes(node); });
913af6ab5fSopenharmony_ci        ++iter;
923af6ab5fSopenharmony_ci    }
933af6ab5fSopenharmony_ci}
943af6ab5fSopenharmony_ci
953af6ab5fSopenharmony_civoid IrCheckHelper::CheckGlobalEntity(parser::Program *program, ir::AstNode *node, bool mustCheck)
963af6ab5fSopenharmony_ci{
973af6ab5fSopenharmony_ci    ASSERT(program);
983af6ab5fSopenharmony_ci
993af6ab5fSopenharmony_ci    auto *globalClass = program->GlobalClass();
1003af6ab5fSopenharmony_ci    auto *globalClassScope = program->GlobalClassScope();
1013af6ab5fSopenharmony_ci
1023af6ab5fSopenharmony_ci    helpers::DoScopedAction(checker_, varBinder_, program, globalClassScope, globalClass,
1033af6ab5fSopenharmony_ci                            [this, globalClassScope, node]() {
1043af6ab5fSopenharmony_ci                                compiler::InitScopesPhaseETS::RunExternalNode(node, varBinder_);
1053af6ab5fSopenharmony_ci                                varBinder_->ResolveReferencesForScope(node, globalClassScope);
1063af6ab5fSopenharmony_ci                            });
1073af6ab5fSopenharmony_ci    if (mustCheck) {
1083af6ab5fSopenharmony_ci        CheckNewNode(node, globalClassScope, globalClass, program);
1093af6ab5fSopenharmony_ci    }
1103af6ab5fSopenharmony_ci}
1113af6ab5fSopenharmony_ci
1123af6ab5fSopenharmony_civoid IrCheckHelper::CheckLocalEntity(ir::AstNode *node)
1133af6ab5fSopenharmony_ci{
1143af6ab5fSopenharmony_ci    compiler::InitScopesPhaseETS::RunExternalNode(node, varBinder_);
1153af6ab5fSopenharmony_ci    CheckNewNode(node, nullptr, nullptr, nullptr);
1163af6ab5fSopenharmony_ci}
1173af6ab5fSopenharmony_ci
1183af6ab5fSopenharmony_ci}  // namespace ark::es2panda::evaluate
119