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 "ifStatement.h"
173af6ab5fSopenharmony_ci
183af6ab5fSopenharmony_ci#include "compiler/base/condition.h"
193af6ab5fSopenharmony_ci#include "compiler/core/pandagen.h"
203af6ab5fSopenharmony_ci#include "ir/astDump.h"
213af6ab5fSopenharmony_ci#include "ir/statements/blockStatement.h"
223af6ab5fSopenharmony_ci#include "ir/expression.h"
233af6ab5fSopenharmony_ci#include "typescript/checker.h"
243af6ab5fSopenharmony_ci
253af6ab5fSopenharmony_cinamespace panda::es2panda::ir {
263af6ab5fSopenharmony_ci
273af6ab5fSopenharmony_civoid IfStatement::Iterate(const NodeTraverser &cb) const
283af6ab5fSopenharmony_ci{
293af6ab5fSopenharmony_ci    cb(test_);
303af6ab5fSopenharmony_ci    cb(consequent_);
313af6ab5fSopenharmony_ci
323af6ab5fSopenharmony_ci    if (alternate_) {
333af6ab5fSopenharmony_ci        cb(alternate_);
343af6ab5fSopenharmony_ci    }
353af6ab5fSopenharmony_ci}
363af6ab5fSopenharmony_ci
373af6ab5fSopenharmony_civoid IfStatement::Dump(ir::AstDumper *dumper) const
383af6ab5fSopenharmony_ci{
393af6ab5fSopenharmony_ci    dumper->Add({{"type", "IfStatement"},
403af6ab5fSopenharmony_ci                 {"test", test_},
413af6ab5fSopenharmony_ci                 {"consequent", consequent_},
423af6ab5fSopenharmony_ci                 {"alternate", AstDumper::Nullable(alternate_)}});
433af6ab5fSopenharmony_ci}
443af6ab5fSopenharmony_ci
453af6ab5fSopenharmony_civoid IfStatement::Compile(compiler::PandaGen *pg) const
463af6ab5fSopenharmony_ci{
473af6ab5fSopenharmony_ci    auto *consequentEnd = pg->AllocLabel();
483af6ab5fSopenharmony_ci    compiler::Label *statementEnd = consequentEnd;
493af6ab5fSopenharmony_ci
503af6ab5fSopenharmony_ci    compiler::Condition::Compile(pg, test_, consequentEnd);
513af6ab5fSopenharmony_ci    consequent_->Compile(pg);
523af6ab5fSopenharmony_ci
533af6ab5fSopenharmony_ci    if (alternate_) {
543af6ab5fSopenharmony_ci        statementEnd = pg->AllocLabel();
553af6ab5fSopenharmony_ci        pg->Branch(pg->Insns().back()->Node(), statementEnd);
563af6ab5fSopenharmony_ci
573af6ab5fSopenharmony_ci        pg->SetLabel(this, consequentEnd);
583af6ab5fSopenharmony_ci        alternate_->Compile(pg);
593af6ab5fSopenharmony_ci    }
603af6ab5fSopenharmony_ci
613af6ab5fSopenharmony_ci    pg->SetLabel(this, statementEnd);
623af6ab5fSopenharmony_ci}
633af6ab5fSopenharmony_ci
643af6ab5fSopenharmony_cichecker::Type *IfStatement::Check(checker::Checker *checker) const
653af6ab5fSopenharmony_ci{
663af6ab5fSopenharmony_ci    checker::Type *testType = test_->Check(checker);
673af6ab5fSopenharmony_ci    checker->CheckTruthinessOfType(testType, Start());
683af6ab5fSopenharmony_ci    checker->CheckTestingKnownTruthyCallableOrAwaitableType(test_, testType, consequent_);
693af6ab5fSopenharmony_ci
703af6ab5fSopenharmony_ci    consequent_->Check(checker);
713af6ab5fSopenharmony_ci
723af6ab5fSopenharmony_ci    if (alternate_) {
733af6ab5fSopenharmony_ci        alternate_->Check(checker);
743af6ab5fSopenharmony_ci    }
753af6ab5fSopenharmony_ci
763af6ab5fSopenharmony_ci    return nullptr;
773af6ab5fSopenharmony_ci}
783af6ab5fSopenharmony_ci
793af6ab5fSopenharmony_civoid IfStatement::UpdateSelf(const NodeUpdater &cb, binder::Binder *binder)
803af6ab5fSopenharmony_ci{
813af6ab5fSopenharmony_ci    test_ = std::get<ir::AstNode *>(cb(test_))->AsExpression();
823af6ab5fSopenharmony_ci    consequent_ = UpdateIfStatementChildStatement(cb, binder, consequent_);
833af6ab5fSopenharmony_ci
843af6ab5fSopenharmony_ci    if (alternate_) {
853af6ab5fSopenharmony_ci        alternate_ = UpdateIfStatementChildStatement(cb, binder, alternate_);
863af6ab5fSopenharmony_ci    }
873af6ab5fSopenharmony_ci}
883af6ab5fSopenharmony_ci
893af6ab5fSopenharmony_ciStatement *IfStatement::UpdateIfStatementChildStatement(const NodeUpdater &cb,
903af6ab5fSopenharmony_ci                                                        const binder::Binder *binder,
913af6ab5fSopenharmony_ci                                                        Statement *statement) const
923af6ab5fSopenharmony_ci{
933af6ab5fSopenharmony_ci    auto newStatement = cb(statement);
943af6ab5fSopenharmony_ci    if (std::holds_alternative<ir::AstNode *>(newStatement)) {
953af6ab5fSopenharmony_ci        return std::get<ir::AstNode *>(newStatement)->AsStatement();
963af6ab5fSopenharmony_ci    }
973af6ab5fSopenharmony_ci
983af6ab5fSopenharmony_ci    ASSERT(std::holds_alternative<std::vector<ir::AstNode *>>(newStatement));
993af6ab5fSopenharmony_ci    ArenaVector<ir::Statement *> statements(binder->Allocator()->Adapter());
1003af6ab5fSopenharmony_ci    auto newStatements = std::get<std::vector<ir::AstNode *>>(newStatement);
1013af6ab5fSopenharmony_ci    for (auto *it : newStatements) {
1023af6ab5fSopenharmony_ci        statements.push_back(it->AsStatement());
1033af6ab5fSopenharmony_ci    }
1043af6ab5fSopenharmony_ci    return binder->Allocator()->New<ir::BlockStatement>(nullptr, std::move(statements));
1053af6ab5fSopenharmony_ci}
1063af6ab5fSopenharmony_ci
1073af6ab5fSopenharmony_ci}  // namespace panda::es2panda::ir
108