mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-25 20:23:11 +01:00
20998194c9
Recently added convertConstantExprsToInstructions() does not handle a case when a same ConstantExpr used multiple times in the same instruction. A first use is replaced and the rest of the uses in the instruction are replaced as well with the replaceUsesOfWith(). Then function attempts to replace a constant already destroyed. So far this interface is only used by the AMDGPU BE. Differential Revision: https://reviews.llvm.org/D104425
163 lines
5.4 KiB
C++
163 lines
5.4 KiB
C++
//===- ReplaceConstant.cpp - Replace LLVM constant expression--------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements a utility function for replacing LLVM constant
|
|
// expressions by instructions.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/IR/ReplaceConstant.h"
|
|
#include "llvm/IR/IRBuilder.h"
|
|
#include "llvm/IR/Instructions.h"
|
|
#include "llvm/IR/NoFolder.h"
|
|
|
|
namespace llvm {
|
|
// Replace a constant expression by instructions with equivalent operations at
|
|
// a specified location.
|
|
Instruction *createReplacementInstr(ConstantExpr *CE, Instruction *Instr) {
|
|
IRBuilder<NoFolder> Builder(Instr);
|
|
unsigned OpCode = CE->getOpcode();
|
|
switch (OpCode) {
|
|
case Instruction::GetElementPtr: {
|
|
SmallVector<Value *, 4> CEOpVec(CE->operands());
|
|
ArrayRef<Value *> CEOps(CEOpVec);
|
|
return dyn_cast<Instruction>(
|
|
Builder.CreateInBoundsGEP(cast<GEPOperator>(CE)->getSourceElementType(),
|
|
CEOps[0], CEOps.slice(1)));
|
|
}
|
|
case Instruction::Add:
|
|
case Instruction::Sub:
|
|
case Instruction::Mul:
|
|
case Instruction::UDiv:
|
|
case Instruction::SDiv:
|
|
case Instruction::FDiv:
|
|
case Instruction::URem:
|
|
case Instruction::SRem:
|
|
case Instruction::FRem:
|
|
case Instruction::Shl:
|
|
case Instruction::LShr:
|
|
case Instruction::AShr:
|
|
case Instruction::And:
|
|
case Instruction::Or:
|
|
case Instruction::Xor:
|
|
return dyn_cast<Instruction>(
|
|
Builder.CreateBinOp((Instruction::BinaryOps)OpCode, CE->getOperand(0),
|
|
CE->getOperand(1), CE->getName()));
|
|
case Instruction::Trunc:
|
|
case Instruction::ZExt:
|
|
case Instruction::SExt:
|
|
case Instruction::FPToUI:
|
|
case Instruction::FPToSI:
|
|
case Instruction::UIToFP:
|
|
case Instruction::SIToFP:
|
|
case Instruction::FPTrunc:
|
|
case Instruction::FPExt:
|
|
case Instruction::PtrToInt:
|
|
case Instruction::IntToPtr:
|
|
case Instruction::BitCast:
|
|
case Instruction::AddrSpaceCast:
|
|
return dyn_cast<Instruction>(
|
|
Builder.CreateCast((Instruction::CastOps)OpCode, CE->getOperand(0),
|
|
CE->getType(), CE->getName()));
|
|
default:
|
|
llvm_unreachable("Unhandled constant expression!\n");
|
|
}
|
|
}
|
|
|
|
void convertConstantExprsToInstructions(Instruction *I, ConstantExpr *CE,
|
|
SmallPtrSetImpl<Instruction *> *Insts) {
|
|
// Collect all reachable paths to CE from constant exprssion operands of I.
|
|
std::map<Use *, std::vector<std::vector<ConstantExpr *>>> CEPaths;
|
|
collectConstantExprPaths(I, CE, CEPaths);
|
|
|
|
// Convert all constant expressions to instructions which are collected at
|
|
// CEPaths.
|
|
convertConstantExprsToInstructions(I, CEPaths, Insts);
|
|
}
|
|
|
|
void convertConstantExprsToInstructions(
|
|
Instruction *I,
|
|
std::map<Use *, std::vector<std::vector<ConstantExpr *>>> &CEPaths,
|
|
SmallPtrSetImpl<Instruction *> *Insts) {
|
|
SmallPtrSet<ConstantExpr *, 8> Visited;
|
|
for (Use &U : I->operands()) {
|
|
// The operand U is either not a constant expression operand or the
|
|
// constant expression paths do not belong to U, ignore U.
|
|
if (!CEPaths.count(&U))
|
|
continue;
|
|
|
|
// If the instruction I is a PHI instruction, then fix the instruction
|
|
// insertion point to the entry of the incoming basic block for operand U.
|
|
auto *BI = I;
|
|
if (auto *Phi = dyn_cast<PHINode>(I)) {
|
|
BasicBlock *BB = Phi->getIncomingBlock(U);
|
|
BI = &(*(BB->getFirstInsertionPt()));
|
|
}
|
|
|
|
// Go through the paths associated with operand U, and convert all the
|
|
// constant expressions along all paths to corresponding instructions.
|
|
auto *II = I;
|
|
auto &Paths = CEPaths[&U];
|
|
for (auto &Path : Paths) {
|
|
for (auto *CE : Path) {
|
|
if (!Visited.insert(CE).second)
|
|
continue;
|
|
auto *NI = CE->getAsInstruction();
|
|
NI->insertBefore(BI);
|
|
II->replaceUsesOfWith(CE, NI);
|
|
CE->removeDeadConstantUsers();
|
|
BI = II = NI;
|
|
if (Insts)
|
|
Insts->insert(NI);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void collectConstantExprPaths(
|
|
Instruction *I, ConstantExpr *CE,
|
|
std::map<Use *, std::vector<std::vector<ConstantExpr *>>> &CEPaths) {
|
|
for (Use &U : I->operands()) {
|
|
// If the operand U is not a constant expression operand, then ignore it.
|
|
auto *CE2 = dyn_cast<ConstantExpr>(U.get());
|
|
if (!CE2)
|
|
continue;
|
|
|
|
// Holds all reachable paths from CE2 to CE.
|
|
std::vector<std::vector<ConstantExpr *>> Paths;
|
|
|
|
// Collect all reachable paths from CE2 to CE.
|
|
std::vector<ConstantExpr *> Path{CE2};
|
|
std::vector<std::vector<ConstantExpr *>> Stack{Path};
|
|
while (!Stack.empty()) {
|
|
std::vector<ConstantExpr *> TPath = Stack.back();
|
|
Stack.pop_back();
|
|
auto *CE3 = TPath.back();
|
|
|
|
if (CE3 == CE) {
|
|
Paths.push_back(TPath);
|
|
continue;
|
|
}
|
|
|
|
for (auto &UU : CE3->operands()) {
|
|
if (auto *CE4 = dyn_cast<ConstantExpr>(UU.get())) {
|
|
std::vector<ConstantExpr *> NPath(TPath.begin(), TPath.end());
|
|
NPath.push_back(CE4);
|
|
Stack.push_back(NPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Associate all the collected paths with U, and save it.
|
|
if (!Paths.empty())
|
|
CEPaths[&U] = Paths;
|
|
}
|
|
}
|
|
|
|
} // namespace llvm
|