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