* Copyright (c) 2021-2026 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "forUpdateStatement.h"
#include <compiler/base/condition.h>
#include <compiler/core/pandagen.h>
#include <ir/astDump.h>
namespace panda::es2panda::ir {
void ForUpdateStatement::Iterate(const NodeTraverser &cb) const
{
if (init_) {
cb(init_);
}
if (test_) {
cb(test_);
}
if (update_) {
cb(update_);
}
cb(body_);
}
void ForUpdateStatement::Dump(ir::AstDumper *dumper) const
{
dumper->Add({{"type", "ForUpdateStatement"},
{"init", AstDumper::Nullable(init_)},
{"test", AstDumper::Nullable(test_)},
{"update", AstDumper::Nullable(update_)},
{"body", body_}});
}
void ForUpdateStatement::Compile(compiler::PandaGen *pg) const
{
compiler::LocalRegScope loopRegScope(pg, scope_);
compiler::LabelTarget labelTarget(pg);
compiler::LoopEnvScope envScope(pg, labelTarget, scope_);
if (init_) {
ASSERT(init_->IsVariableDeclaration() || init_->IsExpression());
init_->Compile(pg);
}
auto *startLabel = pg->AllocLabel();
pg->SetLabel(this, startLabel);
{
if (test_) {
compiler::Condition::Compile(pg, test_, labelTarget.BreakTarget());
}
body_->Compile(pg);
pg->SetLabel(this, labelTarget.ContinueTarget());
envScope.CopyPerIterationCtx();
}
if (update_) {
update_->Compile(pg);
}
pg->Branch(this, startLabel);
pg->SetLabel(this, labelTarget.BreakTarget());
}
void ForUpdateStatement::UpdateSelf(const NodeUpdater &cb, binder::Binder *binder)
{
auto *loopScope = Scope();
auto loopCtx = binder::LexicalScope<binder::LoopScope>::Enter(binder, loopScope);
if (init_) {
init_ = std::get<ir::AstNode *>(cb(init_));
}
if (test_) {
test_ = std::get<ir::AstNode *>(cb(test_))->AsExpression();
}
if (update_) {
update_ = std::get<ir::AstNode *>(cb(update_))->AsExpression();
}
body_ = UpdateChildStatement(cb, binder, body_);
}
}