14514f5e3Sopenharmony_ci/* 24514f5e3Sopenharmony_ci * Copyright (c) 2024 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_STRING_BUILDER_OPTIMIZER_H 174514f5e3Sopenharmony_ci#define ECMASCRIPT_COMPILER_STRING_BUILDER_OPTIMIZER_H 184514f5e3Sopenharmony_ci 194514f5e3Sopenharmony_ci#include "ecmascript/compiler/argument_accessor.h" 204514f5e3Sopenharmony_ci#include "ecmascript/compiler/bytecode_circuit_builder.h" 214514f5e3Sopenharmony_ci#include "ecmascript/compiler/circuit_builder-inl.h" 224514f5e3Sopenharmony_ci#include "ecmascript/compiler/combined_pass_visitor.h" 234514f5e3Sopenharmony_ci#include "ecmascript/compiler/graph_linearizer.h" 244514f5e3Sopenharmony_ci 254514f5e3Sopenharmony_cinamespace panda::ecmascript::kungfu { 264514f5e3Sopenharmony_ciclass StringBuilderOptimizer { 274514f5e3Sopenharmony_cipublic: 284514f5e3Sopenharmony_ci StringBuilderOptimizer(Circuit* circuit, 294514f5e3Sopenharmony_ci bool enableLog, 304514f5e3Sopenharmony_ci const std::string& name, 314514f5e3Sopenharmony_ci CompilationConfig* cmpCfg, 324514f5e3Sopenharmony_ci Chunk* chunk) 334514f5e3Sopenharmony_ci : circuit_(circuit), 344514f5e3Sopenharmony_ci acc_(circuit), 354514f5e3Sopenharmony_ci builder_(circuit, cmpCfg), 364514f5e3Sopenharmony_ci graphLinearizer_(circuit, enableLog, name, chunk, true, true), 374514f5e3Sopenharmony_ci dependEntry_(circuit->GetDependRoot()), 384514f5e3Sopenharmony_ci status_(chunk), 394514f5e3Sopenharmony_ci stringBuilders_(chunk), 404514f5e3Sopenharmony_ci toVisit_(chunk), 414514f5e3Sopenharmony_ci ends_(chunk), 424514f5e3Sopenharmony_ci concatGates_(chunk) {} 434514f5e3Sopenharmony_ci 444514f5e3Sopenharmony_ci ~StringBuilderOptimizer() = default; 454514f5e3Sopenharmony_ci 464514f5e3Sopenharmony_ci void Run(); 474514f5e3Sopenharmony_ciprivate: 484514f5e3Sopenharmony_ci static constexpr int INVALID_ID = -1; 494514f5e3Sopenharmony_ci enum State { 504514f5e3Sopenharmony_ci UNVISITED = 0, 514514f5e3Sopenharmony_ci BEGIN_STRING_BUILDER, 524514f5e3Sopenharmony_ci IN_STRING_BUILDER, 534514f5e3Sopenharmony_ci PENDING_PHI, 544514f5e3Sopenharmony_ci CONFIRMED_IN_STRING_BUILDER, 554514f5e3Sopenharmony_ci END_STRING_BUILDER, 564514f5e3Sopenharmony_ci END_STRING_BUILDER_LOOP_PHI, 574514f5e3Sopenharmony_ci INVALID_OPT, 584514f5e3Sopenharmony_ci NUM_OF_STATE, 594514f5e3Sopenharmony_ci }; 604514f5e3Sopenharmony_ci 614514f5e3Sopenharmony_ci struct Status { 624514f5e3Sopenharmony_ci int id {INVALID_ID}; 634514f5e3Sopenharmony_ci State state {State::INVALID_OPT}; 644514f5e3Sopenharmony_ci }; 654514f5e3Sopenharmony_ci 664514f5e3Sopenharmony_ci Status& GetStatus(GateRef gate) 674514f5e3Sopenharmony_ci { 684514f5e3Sopenharmony_ci ASSERT(acc_.GetId(gate) < status_.size()); 694514f5e3Sopenharmony_ci return status_[acc_.GetId(gate)]; 704514f5e3Sopenharmony_ci } 714514f5e3Sopenharmony_ci const Status& GetStatus(GateRef gate) const 724514f5e3Sopenharmony_ci { 734514f5e3Sopenharmony_ci ASSERT(acc_.GetId(gate) < status_.size()); 744514f5e3Sopenharmony_ci return status_[acc_.GetId(gate)]; 754514f5e3Sopenharmony_ci } 764514f5e3Sopenharmony_ci void SetStatus(GateRef gate, State state, int id = INVALID_ID) 774514f5e3Sopenharmony_ci { 784514f5e3Sopenharmony_ci status_[acc_.GetId(gate)] = Status{id, state}; 794514f5e3Sopenharmony_ci } 804514f5e3Sopenharmony_ci void UpdateStatus(GateRef gate, State state) 814514f5e3Sopenharmony_ci { 824514f5e3Sopenharmony_ci int id = state == State::INVALID_OPT ? INVALID_ID : GetStatus(gate).id; 834514f5e3Sopenharmony_ci status_[acc_.GetId(gate)] = Status{id, state}; 844514f5e3Sopenharmony_ci } 854514f5e3Sopenharmony_ci bool IsInvalidGate(GateRef gate) const 864514f5e3Sopenharmony_ci { 874514f5e3Sopenharmony_ci return GetStatus(gate).state == State::INVALID_OPT; 884514f5e3Sopenharmony_ci } 894514f5e3Sopenharmony_ci 904514f5e3Sopenharmony_ci struct StringBuilder { 914514f5e3Sopenharmony_ci GateRef start {Circuit::NullGate()}; 924514f5e3Sopenharmony_ci int id {INVALID_ID}; 934514f5e3Sopenharmony_ci bool has_loop_phi {false}; 944514f5e3Sopenharmony_ci }; 954514f5e3Sopenharmony_ci 964514f5e3Sopenharmony_ci void VisitGraph(); 974514f5e3Sopenharmony_ci void FindBuilderBegin(GateRef gate); 984514f5e3Sopenharmony_ci void FindInBuilder(GateRef gate); 994514f5e3Sopenharmony_ci void FinalizeStringBuilders(); 1004514f5e3Sopenharmony_ci void VisitGateUse(GateRef use); 1014514f5e3Sopenharmony_ci void StatusTransfer(GateRef gate); 1024514f5e3Sopenharmony_ci bool HasConcatOrPhiUse(GateRef gate); 1034514f5e3Sopenharmony_ci bool CheckStringAddUses(GateRef gate); 1044514f5e3Sopenharmony_ci bool IsLiteralString(GateRef gate); 1054514f5e3Sopenharmony_ci bool IsLoopHeader(GateRef gate); 1064514f5e3Sopenharmony_ci bool LoopContains(GateRef loopPhi, GateRef gate); 1074514f5e3Sopenharmony_ci bool PhiInputsAreConcatsOrPhi(GateRef phi); 1084514f5e3Sopenharmony_ci 1094514f5e3Sopenharmony_ci Circuit *circuit_; 1104514f5e3Sopenharmony_ci GateAccessor acc_; 1114514f5e3Sopenharmony_ci CircuitBuilder builder_; 1124514f5e3Sopenharmony_ci GraphLinearizer graphLinearizer_; 1134514f5e3Sopenharmony_ci GateRef dependEntry_; 1144514f5e3Sopenharmony_ci 1154514f5e3Sopenharmony_ci uint32_t stringBuilderCount_ {0}; 1164514f5e3Sopenharmony_ci size_t currentIndex_ {0}; 1174514f5e3Sopenharmony_ci GateId curStringAddId_ {0}; 1184514f5e3Sopenharmony_ci 1194514f5e3Sopenharmony_ci ChunkVector<Status> status_; 1204514f5e3Sopenharmony_ci ChunkVector<StringBuilder> stringBuilders_; 1214514f5e3Sopenharmony_ci ChunkVector<GateRef> toVisit_; 1224514f5e3Sopenharmony_ci ChunkVector<GateRef> ends_; 1234514f5e3Sopenharmony_ci ChunkVector<GateRef> concatGates_; 1244514f5e3Sopenharmony_ci}; 1254514f5e3Sopenharmony_ci} // panda::ecmascript::kungfu 1264514f5e3Sopenharmony_ci#endif // ECMASCRIPT_COMPILER_STRING_BUILDER_OPTIMIZER_H