From 775ec8d9d5d94d8cb56ae462a6d3841847afef74 Mon Sep 17 00:00:00 2001 From: Alyssa Rosenzweig Date: Tue, 27 Aug 2024 12:31:17 -0400 Subject: [PATCH] DeadStoreElimination: drop now that it's redundant it's only really load bearing for pf/af, which is handled as a global flag opt now. this mitigates some of the compile time hit from globalizing flag opts. Signed-off-by: Alyssa Rosenzweig --- FEXCore/Source/CMakeLists.txt | 1 - FEXCore/Source/Interface/IR/PassManager.cpp | 1 - FEXCore/Source/Interface/IR/Passes.h | 1 - .../IR/Passes/DeadStoreElimination.cpp | 154 ------------------ 4 files changed, 157 deletions(-) delete mode 100644 FEXCore/Source/Interface/IR/Passes/DeadStoreElimination.cpp diff --git a/FEXCore/Source/CMakeLists.txt b/FEXCore/Source/CMakeLists.txt index 85b23af6b8..971271a789 100644 --- a/FEXCore/Source/CMakeLists.txt +++ b/FEXCore/Source/CMakeLists.txt @@ -140,7 +140,6 @@ set (SRCS Interface/IR/Passes/IRValidation.cpp Interface/IR/Passes/RAValidation.cpp Interface/IR/Passes/RedundantFlagCalculationElimination.cpp - Interface/IR/Passes/DeadStoreElimination.cpp Interface/IR/Passes/RegisterAllocationPass.cpp Interface/IR/Passes/x87StackOptimizationPass.cpp Utils/Telemetry.cpp diff --git a/FEXCore/Source/Interface/IR/PassManager.cpp b/FEXCore/Source/Interface/IR/PassManager.cpp index 6f5170e6b5..5072364a68 100644 --- a/FEXCore/Source/Interface/IR/PassManager.cpp +++ b/FEXCore/Source/Interface/IR/PassManager.cpp @@ -71,7 +71,6 @@ void PassManager::AddDefaultPasses(FEXCore::Context::ContextImpl* ctx) { if (!DisablePasses()) { InsertPass(CreateX87StackOptimizationPass()); - InsertPass(CreateDeadStoreElimination()); InsertPass(CreateConstProp(ctx->HostFeatures.SupportsTSOImm9, &ctx->CPUID)); InsertPass(CreateDeadFlagCalculationEliminination()); } diff --git a/FEXCore/Source/Interface/IR/Passes.h b/FEXCore/Source/Interface/IR/Passes.h index 3b377dc6fa..a8a3d1cb64 100644 --- a/FEXCore/Source/Interface/IR/Passes.h +++ b/FEXCore/Source/Interface/IR/Passes.h @@ -18,7 +18,6 @@ class RegisterAllocationData; fextl::unique_ptr CreateConstProp(bool SupportsTSOImm9, const FEXCore::CPUIDEmu* CPUID); fextl::unique_ptr CreateDeadFlagCalculationEliminination(); -fextl::unique_ptr CreateDeadStoreElimination(); fextl::unique_ptr CreateRegisterAllocationPass(); fextl::unique_ptr CreateX87StackOptimizationPass(); diff --git a/FEXCore/Source/Interface/IR/Passes/DeadStoreElimination.cpp b/FEXCore/Source/Interface/IR/Passes/DeadStoreElimination.cpp deleted file mode 100644 index 5a3471324f..0000000000 --- a/FEXCore/Source/Interface/IR/Passes/DeadStoreElimination.cpp +++ /dev/null @@ -1,154 +0,0 @@ -// SPDX-License-Identifier: MIT -/* -$info$ -tags: ir|opts -desc: Cross block store-after-store elimination -$end_info$ -*/ - -#include "Interface/IR/IREmitter.h" -#include "Interface/IR/PassManager.h" - -#include -#include -#include -#include -#include - -#include -#include -#include - -namespace FEXCore::IR { - -constexpr int PropagationRounds = 5; - -// Return a bit representing a single GPR or FPR. -static inline uint64_t RegBit(RegisterClassType Class, uint32_t Reg) { - uint32_t AdjustedReg = (Class == FPRClass) ? (32 + Reg) : Reg; - - return 1ULL << AdjustedReg; -} - -class DeadStoreElimination final : public FEXCore::IR::Pass { -public: - void Run(IREmitter* IREmit) override; -}; - -struct ReadWriteKill { - uint64_t reads {0}; - uint64_t writes {0}; - uint64_t kill {0}; -}; - -struct Info { - ReadWriteKill reg; -}; - -/** - * @brief This is a temporary pass to detect simple multiblock dead reg stores - * - * First pass computes which regs are read and written per block - * - * Second pass computes which regs are stored, but overwritten by the next block(s). - * It also propagates this information a few times to catch dead regs across multiple blocks. - * - * Third pass removes the dead stores. - * - */ -void DeadStoreElimination::Run(IREmitter* IREmit) { - FEXCORE_PROFILE_SCOPED("PassManager::DSE"); - - auto CurrentIR = IREmit->ViewIR(); - fextl::vector InfoMap(CurrentIR.GetSSACount()); - - // Pass 1 - // Compute regs read/writes per block - // This is conservative and doesn't try to be smart about loads after writes - { - for (auto [BlockNode, BlockIROp] : CurrentIR.GetBlocks()) { - auto& BlockInfo = InfoMap[CurrentIR.GetID(BlockNode).Value]; - - for (auto [CodeNode, IROp] : CurrentIR.GetCode(BlockNode)) { - if (IROp->Op == OP_STOREREGISTER) { - auto Op = IROp->C(); - BlockInfo.reg.writes |= RegBit(Op->Class, Op->Reg); - } else if (IROp->Op == OP_LOADREGISTER) { - auto Op = IROp->C(); - BlockInfo.reg.reads |= RegBit(Op->Class, Op->Reg); - } else if (IROp->Op == OP_INVALIDATEFLAGS) { - auto Op = IROp->C(); - - if (Op->Flags & (1u << X86State::RFLAG_PF_RAW_LOC)) { - BlockInfo.reg.writes |= RegBit(GPRClass, Core::CPUState::PF_AS_GREG); - } - - if (Op->Flags & (1u << X86State::RFLAG_AF_RAW_LOC)) { - BlockInfo.reg.writes |= RegBit(GPRClass, Core::CPUState::AF_AS_GREG); - } - } - } - } - } - - // Pass 2 - // Compute flags/registers that are stored, but always ovewritten in the next blocks - // Propagate the information a few times to eliminate more - for (int i = 0; i < PropagationRounds; i++) { - for (auto [BlockNode, BlockIROp] : CurrentIR.GetBlocks()) { - auto CodeBlock = BlockIROp->C(); - - auto IROp = CurrentIR.GetNode(CurrentIR.GetNode(CodeBlock->Last)->Header.Previous)->Op(CurrentIR.GetData()); - - if (IROp->Op == OP_JUMP) { - auto Op = IROp->C(); - auto& BlockInfo = InfoMap[CurrentIR.GetID(BlockNode).Value]; - auto& TargetInfo = InfoMap[Op->Header.Args[0].ID().Value]; - - // stores to remove are written by the next block but not read - BlockInfo.reg.kill = TargetInfo.reg.writes & ~(TargetInfo.reg.reads) & ~BlockInfo.reg.reads; - - // If written by the next block can be considered as written by this block, if not read - BlockInfo.reg.writes |= BlockInfo.reg.kill & ~BlockInfo.reg.reads; - } else if (IROp->Op == OP_CONDJUMP) { - auto Op = IROp->C(); - - auto& BlockInfo = InfoMap[CurrentIR.GetID(BlockNode).Value]; - auto& TrueTargetInfo = InfoMap[Op->TrueBlock.ID().Value]; - auto& FalseTargetInfo = InfoMap[Op->FalseBlock.ID().Value]; - - // stores to remove are written by the next blocks but not read - BlockInfo.reg.kill = TrueTargetInfo.reg.writes & ~(TrueTargetInfo.reg.reads) & ~BlockInfo.reg.reads; - BlockInfo.reg.kill &= FalseTargetInfo.reg.writes & ~(FalseTargetInfo.reg.reads) & ~BlockInfo.reg.reads; - - // if written by the next blocks can be considered as written by this block, if not read - BlockInfo.reg.writes |= BlockInfo.reg.kill & ~BlockInfo.reg.reads; - } - } - } - - // Pass 3 - // Remove the dead stores - { - for (auto [BlockNode, BlockIROp] : CurrentIR.GetBlocks()) { - auto& BlockInfo = InfoMap[CurrentIR.GetID(BlockNode).Value]; - - for (auto [CodeNode, IROp] : CurrentIR.GetCode(BlockNode)) { - if (IROp->Op == OP_STOREREGISTER) { - auto Op = IROp->C(); - - // If this OP_STOREREGISTER is never read, remove it - if (BlockInfo.reg.kill & RegBit(Op->Class, Op->Reg)) { - IREmit->Remove(CodeNode); - } - } - } - } - } -} - -fextl::unique_ptr CreateDeadStoreElimination() { - return fextl::make_unique(); -} - -} // namespace FEXCore::IR