14514f5e3Sopenharmony_ci/* 24514f5e3Sopenharmony_ci * Copyright (c) 2023 Huawei Device Co., Ltd. 34514f5e3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 44514f5e3Sopenharmony_ci * you may not use this file except in compliance with the License. 54514f5e3Sopenharmony_ci * You may obtain a copy of the License at 64514f5e3Sopenharmony_ci * 74514f5e3Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 84514f5e3Sopenharmony_ci * 94514f5e3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 104514f5e3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 114514f5e3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 124514f5e3Sopenharmony_ci * See the License for the specific language governing permissions and 134514f5e3Sopenharmony_ci * limitations under the License. 144514f5e3Sopenharmony_ci */ 154514f5e3Sopenharmony_ci 164514f5e3Sopenharmony_ci#ifndef ECMASCRIPT_COMPILER_RANGE_GUARD_H 174514f5e3Sopenharmony_ci#define ECMASCRIPT_COMPILER_RANGE_GUARD_H 184514f5e3Sopenharmony_ci 194514f5e3Sopenharmony_ci#include "ecmascript/compiler/circuit_builder.h" 204514f5e3Sopenharmony_ci#include "ecmascript/compiler/gate_accessor.h" 214514f5e3Sopenharmony_ci#include "ecmascript/compiler/pass_manager.h" 224514f5e3Sopenharmony_ci#include "ecmascript/compiler/base/depend_chain_helper.h" 234514f5e3Sopenharmony_ci#include "ecmascript/mem/chunk_containers.h" 244514f5e3Sopenharmony_ci#include "ecmascript/compiler/number_gate_info.h" 254514f5e3Sopenharmony_ci#include "ecmascript/compiler/combined_pass_visitor.h" 264514f5e3Sopenharmony_ci 274514f5e3Sopenharmony_cinamespace panda::ecmascript::kungfu { 284514f5e3Sopenharmony_ciclass RangeGuard : public PassVisitor { 294514f5e3Sopenharmony_cipublic: 304514f5e3Sopenharmony_ci RangeGuard(Circuit* circuit, RPOVisitor* visitor, Chunk* chunk) 314514f5e3Sopenharmony_ci : PassVisitor(circuit, chunk, visitor), circuit_(circuit), 324514f5e3Sopenharmony_ci builder_(circuit), acc_(circuit), dependChains_(chunk) {} 334514f5e3Sopenharmony_ci 344514f5e3Sopenharmony_ci ~RangeGuard() = default; 354514f5e3Sopenharmony_ci 364514f5e3Sopenharmony_ci void Initialize() override; 374514f5e3Sopenharmony_ci GateRef VisitGate(GateRef gate) override; 384514f5e3Sopenharmony_ci bool CheckInputSource(GateRef lhs, GateRef rhs); 394514f5e3Sopenharmony_ci uint32_t CheckIndexCheckLengthInput(GateRef lhs, GateRef rhs) const; 404514f5e3Sopenharmony_ci uint32_t CheckIndexCheckIndexInput(GateRef lhs, GateRef rhs) const; 414514f5e3Sopenharmony_ciprivate: 424514f5e3Sopenharmony_ci 434514f5e3Sopenharmony_ci DependChains* GetDependChain(GateRef dependIn) 444514f5e3Sopenharmony_ci { 454514f5e3Sopenharmony_ci size_t idx = acc_.GetId(dependIn); 464514f5e3Sopenharmony_ci ASSERT(idx <= circuit_->GetMaxGateId()); 474514f5e3Sopenharmony_ci return dependChains_[idx]; 484514f5e3Sopenharmony_ci } 494514f5e3Sopenharmony_ci 504514f5e3Sopenharmony_ci uint32_t FoundIndexCheckedForLength(DependChains* dependChain, GateRef input) const 514514f5e3Sopenharmony_ci { 524514f5e3Sopenharmony_ci for (auto iter = dependChain->begin(); iter != dependChain->end(); ++iter) { 534514f5e3Sopenharmony_ci uint32_t length = CheckIndexCheckLengthInput(iter.GetCurrentGate(), input); 544514f5e3Sopenharmony_ci if (length > 0) { // found !!! 554514f5e3Sopenharmony_ci return length; 564514f5e3Sopenharmony_ci } 574514f5e3Sopenharmony_ci } 584514f5e3Sopenharmony_ci 594514f5e3Sopenharmony_ci return 0; 604514f5e3Sopenharmony_ci } 614514f5e3Sopenharmony_ci 624514f5e3Sopenharmony_ci uint32_t FoundIndexCheckedForIndex(DependChains* dependChain, GateRef input) const 634514f5e3Sopenharmony_ci { 644514f5e3Sopenharmony_ci for (auto iter = dependChain->begin(); iter != dependChain->end(); ++iter) { 654514f5e3Sopenharmony_ci uint32_t length = CheckIndexCheckIndexInput(iter.GetCurrentGate(), input); 664514f5e3Sopenharmony_ci if (length > 0) { // found !!! 674514f5e3Sopenharmony_ci return length; 684514f5e3Sopenharmony_ci } 694514f5e3Sopenharmony_ci } 704514f5e3Sopenharmony_ci 714514f5e3Sopenharmony_ci return 0; 724514f5e3Sopenharmony_ci } 734514f5e3Sopenharmony_ci 744514f5e3Sopenharmony_ci GateRef VisitDependEntry(GateRef gate); 754514f5e3Sopenharmony_ci GateRef UpdateDependChain(GateRef gate, DependChains* dependInfo); 764514f5e3Sopenharmony_ci GateRef TryApplyRangeGuardForLength(DependChains* dependInfo, GateRef gate, GateRef input); 774514f5e3Sopenharmony_ci GateRef TryApplyRangeGuardForIndex(DependChains* dependInfo, GateRef gate, GateRef input); 784514f5e3Sopenharmony_ci GateRef TryApplyRangeGuardGate(GateRef gate); 794514f5e3Sopenharmony_ci GateRef TraverseOthers(GateRef gate); 804514f5e3Sopenharmony_ci GateRef TraverseDependSelector(GateRef gate); 814514f5e3Sopenharmony_ci 824514f5e3Sopenharmony_ci Circuit* circuit_; 834514f5e3Sopenharmony_ci CircuitBuilder builder_; 844514f5e3Sopenharmony_ci GateAccessor acc_; 854514f5e3Sopenharmony_ci ChunkVector<DependChains*> dependChains_; 864514f5e3Sopenharmony_ci 874514f5e3Sopenharmony_ci friend class RangeInfo; 884514f5e3Sopenharmony_ci}; 894514f5e3Sopenharmony_ci} // panda::ecmascript::kungfu 904514f5e3Sopenharmony_ci#endif // ECMASCRIPT_COMPILER_RANGE_GUARD_H