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