Skip to content

Commit f21f1ee

Browse files
committed
[FPEnv] EarlyCSE support for constrained intrinsics, default FP environment edition
EarlyCSE cannot distinguish between floating point instructions and constrained floating point intrinsics that are marked as running in the default FP environment. Said intrinsics are supposed to behave exactly the same as the regular FP instructions. Teach EarlyCSE to handle them in that case. Differential Revision: https://reviews.llvm.org/D99962
1 parent f4ccbaf commit f21f1ee

File tree

4 files changed

+369
-5
lines changed

4 files changed

+369
-5
lines changed

llvm/include/llvm/IR/IntrinsicInst.h

+1
Original file line numberDiff line numberDiff line change
@@ -432,6 +432,7 @@ class ConstrainedFPIntrinsic : public IntrinsicInst {
432432
bool isTernaryOp() const;
433433
Optional<RoundingMode> getRoundingMode() const;
434434
Optional<fp::ExceptionBehavior> getExceptionBehavior() const;
435+
bool isDefaultFPEnvironment() const;
435436

436437
// Methods for support type inquiry through isa, cast, and dyn_cast:
437438
static bool classof(const IntrinsicInst *I);

llvm/lib/IR/IntrinsicInst.cpp

+24-4
Original file line numberDiff line numberDiff line change
@@ -172,8 +172,10 @@ Value *InstrProfIncrementInst::getStep() const {
172172

173173
Optional<RoundingMode> ConstrainedFPIntrinsic::getRoundingMode() const {
174174
unsigned NumOperands = getNumArgOperands();
175-
Metadata *MD =
176-
cast<MetadataAsValue>(getArgOperand(NumOperands - 2))->getMetadata();
175+
Metadata *MD = nullptr;
176+
auto *MAV = dyn_cast<MetadataAsValue>(getArgOperand(NumOperands - 2));
177+
if (MAV)
178+
MD = MAV->getMetadata();
177179
if (!MD || !isa<MDString>(MD))
178180
return None;
179181
return StrToRoundingMode(cast<MDString>(MD)->getString());
@@ -182,13 +184,31 @@ Optional<RoundingMode> ConstrainedFPIntrinsic::getRoundingMode() const {
182184
Optional<fp::ExceptionBehavior>
183185
ConstrainedFPIntrinsic::getExceptionBehavior() const {
184186
unsigned NumOperands = getNumArgOperands();
185-
Metadata *MD =
186-
cast<MetadataAsValue>(getArgOperand(NumOperands - 1))->getMetadata();
187+
Metadata *MD = nullptr;
188+
auto *MAV = dyn_cast<MetadataAsValue>(getArgOperand(NumOperands - 1));
189+
if (MAV)
190+
MD = MAV->getMetadata();
187191
if (!MD || !isa<MDString>(MD))
188192
return None;
189193
return StrToExceptionBehavior(cast<MDString>(MD)->getString());
190194
}
191195

196+
bool ConstrainedFPIntrinsic::isDefaultFPEnvironment() const {
197+
Optional<fp::ExceptionBehavior> Except = getExceptionBehavior();
198+
if (Except) {
199+
if (Except.getValue() != fp::ebIgnore)
200+
return false;
201+
}
202+
203+
Optional<RoundingMode> Rounding = getRoundingMode();
204+
if (Rounding) {
205+
if (Rounding.getValue() != RoundingMode::NearestTiesToEven)
206+
return false;
207+
}
208+
209+
return true;
210+
}
211+
192212
FCmpInst::Predicate ConstrainedFPCmpIntrinsic::getPredicate() const {
193213
Metadata *MD = cast<MetadataAsValue>(getArgOperand(2))->getMetadata();
194214
if (!MD || !isa<MDString>(MD))

llvm/lib/Transforms/Scalar/EarlyCSE.cpp

+22-1
Original file line numberDiff line numberDiff line change
@@ -108,8 +108,29 @@ struct SimpleValue {
108108

109109
static bool canHandle(Instruction *Inst) {
110110
// This can only handle non-void readnone functions.
111-
if (CallInst *CI = dyn_cast<CallInst>(Inst))
111+
// Also handled are constrained intrinsic that look like the types
112+
// of instruction handled below (UnaryOperator, etc.).
113+
if (CallInst *CI = dyn_cast<CallInst>(Inst)) {
114+
if (Function *F = CI->getCalledFunction()) {
115+
switch ((Intrinsic::ID)F->getIntrinsicID()) {
116+
case Intrinsic::experimental_constrained_fadd:
117+
case Intrinsic::experimental_constrained_fsub:
118+
case Intrinsic::experimental_constrained_fmul:
119+
case Intrinsic::experimental_constrained_fdiv:
120+
case Intrinsic::experimental_constrained_frem:
121+
case Intrinsic::experimental_constrained_fptosi:
122+
case Intrinsic::experimental_constrained_sitofp:
123+
case Intrinsic::experimental_constrained_fptoui:
124+
case Intrinsic::experimental_constrained_uitofp:
125+
case Intrinsic::experimental_constrained_fcmp:
126+
case Intrinsic::experimental_constrained_fcmps: {
127+
auto *CFP = cast<ConstrainedFPIntrinsic>(CI);
128+
return CFP->isDefaultFPEnvironment();
129+
}
130+
}
131+
}
112132
return CI->doesNotAccessMemory() && !CI->getType()->isVoidTy();
133+
}
113134
return isa<CastInst>(Inst) || isa<UnaryOperator>(Inst) ||
114135
isa<BinaryOperator>(Inst) || isa<GetElementPtrInst>(Inst) ||
115136
isa<CmpInst>(Inst) || isa<SelectInst>(Inst) ||

0 commit comments

Comments
 (0)