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#include "ecmascript/compiler/later_elimination.h" 164514f5e3Sopenharmony_ci 174514f5e3Sopenharmony_cinamespace panda::ecmascript::kungfu { 184514f5e3Sopenharmony_ci 194514f5e3Sopenharmony_civoid LaterElimination::Initialize() 204514f5e3Sopenharmony_ci{ 214514f5e3Sopenharmony_ci dependChains_.resize(circuit_->GetMaxGateId() + 1, nullptr); // 1: +1 for size 224514f5e3Sopenharmony_ci GateRef entry = acc_.GetDependRoot(); 234514f5e3Sopenharmony_ci VisitDependEntry(entry); 244514f5e3Sopenharmony_ci} 254514f5e3Sopenharmony_ci 264514f5e3Sopenharmony_ciGateRef LaterElimination::VisitDependEntry(GateRef gate) 274514f5e3Sopenharmony_ci{ 284514f5e3Sopenharmony_ci auto empty = new (chunk_) DependChains(chunk_); 294514f5e3Sopenharmony_ci return UpdateDependChain(gate, empty); 304514f5e3Sopenharmony_ci} 314514f5e3Sopenharmony_ci 324514f5e3Sopenharmony_ciGateRef LaterElimination::VisitGate(GateRef gate) 334514f5e3Sopenharmony_ci{ 344514f5e3Sopenharmony_ci auto opcode = acc_.GetOpCode(gate); 354514f5e3Sopenharmony_ci switch (opcode) { 364514f5e3Sopenharmony_ci case OpCode::GET_ENV: 374514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_ENV: 384514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_ENV_OBJ: 394514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_ENV_OBJ_HCLASS: 404514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_CONSTANT_VALUE: 414514f5e3Sopenharmony_ci case OpCode::ARRAY_GUARDIAN_CHECK: 424514f5e3Sopenharmony_ci case OpCode::HCLASS_STABLE_ARRAY_CHECK: 434514f5e3Sopenharmony_ci case OpCode::HEAP_OBJECT_CHECK: 444514f5e3Sopenharmony_ci case OpCode::ECMA_OBJECT_CHECK: 454514f5e3Sopenharmony_ci case OpCode::INT32_UNSIGNED_UPPER_BOUND_CHECK: 464514f5e3Sopenharmony_ci case OpCode::OVERFLOW_CHECK: 474514f5e3Sopenharmony_ci case OpCode::VALUE_CHECK_NEG_OVERFLOW: 484514f5e3Sopenharmony_ci case OpCode::FLOAT64_CHECK_RIGHT_IS_ZERO: 494514f5e3Sopenharmony_ci case OpCode::INT32_CHECK_RIGHT_IS_ZERO: 504514f5e3Sopenharmony_ci case OpCode::INT32_DIV_WITH_CHECK: 514514f5e3Sopenharmony_ci case OpCode::LEX_VAR_IS_HOLE_CHECK: 524514f5e3Sopenharmony_ci case OpCode::COW_ARRAY_CHECK: 534514f5e3Sopenharmony_ci case OpCode::FLATTEN_TREE_STRING_CHECK: 544514f5e3Sopenharmony_ci case OpCode::CHECK_AND_CONVERT: 554514f5e3Sopenharmony_ci case OpCode::TAGGED_IS_HEAP_OBJECT: 564514f5e3Sopenharmony_ci case OpCode::IS_MARKER_CELL_VALID: 574514f5e3Sopenharmony_ci case OpCode::IS_SPECIFIC_OBJECT_TYPE: 584514f5e3Sopenharmony_ci return TryEliminateGate(gate); 594514f5e3Sopenharmony_ci case OpCode::DEPEND_SELECTOR: 604514f5e3Sopenharmony_ci return TryEliminateDependSelector(gate); 614514f5e3Sopenharmony_ci default: 624514f5e3Sopenharmony_ci if (acc_.GetDependCount(gate) == 1) { // 1: depend in is 1 634514f5e3Sopenharmony_ci return TryEliminateOther(gate); 644514f5e3Sopenharmony_ci } 654514f5e3Sopenharmony_ci } 664514f5e3Sopenharmony_ci return Circuit::NullGate(); 674514f5e3Sopenharmony_ci} 684514f5e3Sopenharmony_ci 694514f5e3Sopenharmony_ciGateRef LaterElimination::TryEliminateOther(GateRef gate) 704514f5e3Sopenharmony_ci{ 714514f5e3Sopenharmony_ci ASSERT(acc_.GetDependCount(gate) >= 1); 724514f5e3Sopenharmony_ci auto depIn = acc_.GetDep(gate); 734514f5e3Sopenharmony_ci auto dependChain = GetDependChain(depIn); 744514f5e3Sopenharmony_ci if (dependChain == nullptr) { 754514f5e3Sopenharmony_ci return Circuit::NullGate(); 764514f5e3Sopenharmony_ci } 774514f5e3Sopenharmony_ci return UpdateDependChain(gate, dependChain); 784514f5e3Sopenharmony_ci} 794514f5e3Sopenharmony_ci 804514f5e3Sopenharmony_ciGateRef LaterElimination::TryEliminateGate(GateRef gate) 814514f5e3Sopenharmony_ci{ 824514f5e3Sopenharmony_ci ASSERT(acc_.GetDependCount(gate) == 1); 834514f5e3Sopenharmony_ci auto depIn = acc_.GetDep(gate); 844514f5e3Sopenharmony_ci auto dependChain = GetDependChain(depIn); 854514f5e3Sopenharmony_ci // dependChain is null 864514f5e3Sopenharmony_ci if (dependChain == nullptr) { 874514f5e3Sopenharmony_ci return Circuit::NullGate(); 884514f5e3Sopenharmony_ci } 894514f5e3Sopenharmony_ci // lookup gate, replace 904514f5e3Sopenharmony_ci auto preGate = LookupNode(dependChain, gate); 914514f5e3Sopenharmony_ci if (preGate != Circuit::NullGate()) { 924514f5e3Sopenharmony_ci return preGate; 934514f5e3Sopenharmony_ci } 944514f5e3Sopenharmony_ci // update gate, for others elimination 954514f5e3Sopenharmony_ci dependChain = dependChain->UpdateNode(gate); 964514f5e3Sopenharmony_ci return UpdateDependChain(gate, dependChain); 974514f5e3Sopenharmony_ci} 984514f5e3Sopenharmony_ci 994514f5e3Sopenharmony_ciGateRef LaterElimination::TryEliminateDependSelector(GateRef gate) 1004514f5e3Sopenharmony_ci{ 1014514f5e3Sopenharmony_ci auto state = acc_.GetState(gate); 1024514f5e3Sopenharmony_ci if (acc_.IsLoopHead(state)) { 1034514f5e3Sopenharmony_ci // use loop head as depend chain 1044514f5e3Sopenharmony_ci return TryEliminateOther(gate); 1054514f5e3Sopenharmony_ci } 1064514f5e3Sopenharmony_ci 1074514f5e3Sopenharmony_ci auto dependCount = acc_.GetDependCount(gate); 1084514f5e3Sopenharmony_ci for (size_t i = 0; i < dependCount; ++i) { 1094514f5e3Sopenharmony_ci auto depend = acc_.GetDep(gate, i); 1104514f5e3Sopenharmony_ci auto dependChain = GetDependChain(depend); 1114514f5e3Sopenharmony_ci if (dependChain == nullptr) { 1124514f5e3Sopenharmony_ci return Circuit::NullGate(); 1134514f5e3Sopenharmony_ci } 1144514f5e3Sopenharmony_ci } 1154514f5e3Sopenharmony_ci 1164514f5e3Sopenharmony_ci // all depend done. 1174514f5e3Sopenharmony_ci auto depend = acc_.GetDep(gate); 1184514f5e3Sopenharmony_ci auto dependChain = GetDependChain(depend); 1194514f5e3Sopenharmony_ci DependChains* copy = new (chunk_) DependChains(chunk_); 1204514f5e3Sopenharmony_ci copy->CopyFrom(dependChain); 1214514f5e3Sopenharmony_ci for (size_t i = 1; i < dependCount; ++i) { // 1: second in 1224514f5e3Sopenharmony_ci auto dependIn = acc_.GetDep(gate, i); 1234514f5e3Sopenharmony_ci auto tempChain = GetDependChain(dependIn); 1244514f5e3Sopenharmony_ci copy->Merge(tempChain); 1254514f5e3Sopenharmony_ci } 1264514f5e3Sopenharmony_ci return UpdateDependChain(gate, copy); 1274514f5e3Sopenharmony_ci} 1284514f5e3Sopenharmony_ci 1294514f5e3Sopenharmony_ciGateRef LaterElimination::UpdateDependChain(GateRef gate, DependChains* dependChain) 1304514f5e3Sopenharmony_ci{ 1314514f5e3Sopenharmony_ci ASSERT(dependChain != nullptr); 1324514f5e3Sopenharmony_ci auto oldDependChain = GetDependChain(gate); 1334514f5e3Sopenharmony_ci if (dependChain->Equals(oldDependChain)) { 1344514f5e3Sopenharmony_ci return Circuit::NullGate(); 1354514f5e3Sopenharmony_ci } 1364514f5e3Sopenharmony_ci dependChains_[acc_.GetId(gate)] = dependChain; 1374514f5e3Sopenharmony_ci return gate; 1384514f5e3Sopenharmony_ci} 1394514f5e3Sopenharmony_ci 1404514f5e3Sopenharmony_cibool LaterElimination::CheckReplacement(GateRef lhs, GateRef rhs) 1414514f5e3Sopenharmony_ci{ 1424514f5e3Sopenharmony_ci if (!acc_.MetaDataEqu(lhs, rhs)) { 1434514f5e3Sopenharmony_ci if (acc_.GetOpCode(lhs) != acc_.GetOpCode(rhs)) { 1444514f5e3Sopenharmony_ci return false; 1454514f5e3Sopenharmony_ci } 1464514f5e3Sopenharmony_ci } 1474514f5e3Sopenharmony_ci size_t valueCount = acc_.GetNumValueIn(lhs); 1484514f5e3Sopenharmony_ci for (size_t i = 0; i < valueCount; i++) { 1494514f5e3Sopenharmony_ci if (acc_.GetValueIn(lhs, i) != acc_.GetValueIn(rhs, i)) { 1504514f5e3Sopenharmony_ci return false; 1514514f5e3Sopenharmony_ci } 1524514f5e3Sopenharmony_ci } 1534514f5e3Sopenharmony_ci auto opcode = acc_.GetOpCode(lhs); 1544514f5e3Sopenharmony_ci switch (opcode) { 1554514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_ENV_OBJ: 1564514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_ENV_OBJ_HCLASS: 1574514f5e3Sopenharmony_ci case OpCode::GET_GLOBAL_CONSTANT_VALUE: { 1584514f5e3Sopenharmony_ci if (acc_.GetIndex(lhs) != acc_.GetIndex(rhs)) { 1594514f5e3Sopenharmony_ci return false; 1604514f5e3Sopenharmony_ci } 1614514f5e3Sopenharmony_ci break; 1624514f5e3Sopenharmony_ci } 1634514f5e3Sopenharmony_ci case OpCode::IS_SPECIFIC_OBJECT_TYPE: { 1644514f5e3Sopenharmony_ci if (acc_.GetJSType(lhs) != acc_.GetJSType(rhs)) { 1654514f5e3Sopenharmony_ci return false; 1664514f5e3Sopenharmony_ci } 1674514f5e3Sopenharmony_ci break; 1684514f5e3Sopenharmony_ci } 1694514f5e3Sopenharmony_ci case OpCode::CHECK_AND_CONVERT: { 1704514f5e3Sopenharmony_ci if (acc_.GetSrcType(lhs) != acc_.GetSrcType(rhs)) { 1714514f5e3Sopenharmony_ci return false; 1724514f5e3Sopenharmony_ci } 1734514f5e3Sopenharmony_ci if (acc_.GetDstType(lhs) != acc_.GetDstType(rhs)) { 1744514f5e3Sopenharmony_ci return false; 1754514f5e3Sopenharmony_ci } 1764514f5e3Sopenharmony_ci break; 1774514f5e3Sopenharmony_ci } 1784514f5e3Sopenharmony_ci default: 1794514f5e3Sopenharmony_ci break; 1804514f5e3Sopenharmony_ci } 1814514f5e3Sopenharmony_ci return true; 1824514f5e3Sopenharmony_ci} 1834514f5e3Sopenharmony_ci} // namespace panda::ecmascript::kungfu 184