@@ -471,7 +471,7 @@ using RepeatedValue = std::pair<Value*, APInt>;
471
471
static bool LinearizeExprTree (Instruction *I,
472
472
SmallVectorImpl<RepeatedValue> &Ops,
473
473
ReassociatePass::OrderedSet &ToRedo,
474
- bool &HasNUW ) {
474
+ reassociate::OverflowTracking &Flags ) {
475
475
assert ((isa<UnaryOperator>(I) || isa<BinaryOperator>(I)) &&
476
476
" Expected a UnaryOperator or BinaryOperator!" );
477
477
LLVM_DEBUG (dbgs () << " LINEARIZE: " << *I << ' \n ' );
@@ -512,6 +512,7 @@ static bool LinearizeExprTree(Instruction *I,
512
512
using LeafMap = DenseMap<Value *, APInt>;
513
513
LeafMap Leaves; // Leaf -> Total weight so far.
514
514
SmallVector<Value *, 8 > LeafOrder; // Ensure deterministic leaf output order.
515
+ const DataLayout DL = I->getModule ()->getDataLayout ();
515
516
516
517
#ifndef NDEBUG
517
518
SmallPtrSet<Value *, 8 > Visited; // For checking the iteration scheme.
@@ -520,8 +521,10 @@ static bool LinearizeExprTree(Instruction *I,
520
521
std::pair<Instruction*, APInt> P = Worklist.pop_back_val ();
521
522
I = P.first ; // We examine the operands of this binary operator.
522
523
523
- if (isa<OverflowingBinaryOperator>(I))
524
- HasNUW &= I->hasNoUnsignedWrap ();
524
+ if (isa<OverflowingBinaryOperator>(I)) {
525
+ Flags.HasNUW &= I->hasNoUnsignedWrap ();
526
+ Flags.HasNSW &= I->hasNoSignedWrap ();
527
+ }
525
528
526
529
for (unsigned OpIdx = 0 ; OpIdx < I->getNumOperands (); ++OpIdx) { // Visit operands.
527
530
Value *Op = I->getOperand (OpIdx);
@@ -648,6 +651,8 @@ static bool LinearizeExprTree(Instruction *I,
648
651
// Ensure the leaf is only output once.
649
652
It->second = 0 ;
650
653
Ops.push_back (std::make_pair (V, Weight));
654
+ if (Opcode == Instruction::Add && Flags.AllKnownNonNegative && Flags.HasNSW )
655
+ Flags.AllKnownNonNegative &= isKnownNonNegative (V, SimplifyQuery (DL));
651
656
}
652
657
653
658
// For nilpotent operations or addition there may be no operands, for example
@@ -666,7 +671,7 @@ static bool LinearizeExprTree(Instruction *I,
666
671
// / linearized and optimized, emit them in-order.
667
672
void ReassociatePass::RewriteExprTree (BinaryOperator *I,
668
673
SmallVectorImpl<ValueEntry> &Ops,
669
- bool HasNUW ) {
674
+ OverflowTracking Flags ) {
670
675
assert (Ops.size () > 1 && " Single values should be used directly!" );
671
676
672
677
// Since our optimizations should never increase the number of operations, the
@@ -834,8 +839,12 @@ void ReassociatePass::RewriteExprTree(BinaryOperator *I,
834
839
// Note that it doesn't hold for mul if one of the operands is zero.
835
840
// TODO: We can preserve NUW flag if we prove that all mul operands
836
841
// are non-zero.
837
- if (HasNUW && ExpressionChangedStart->getOpcode () == Instruction::Add)
838
- ExpressionChangedStart->setHasNoUnsignedWrap ();
842
+ if (ExpressionChangedStart->getOpcode () == Instruction::Add) {
843
+ if (Flags.HasNUW )
844
+ ExpressionChangedStart->setHasNoUnsignedWrap ();
845
+ if (Flags.HasNSW && (Flags.AllKnownNonNegative || Flags.HasNUW ))
846
+ ExpressionChangedStart->setHasNoSignedWrap ();
847
+ }
839
848
}
840
849
}
841
850
@@ -1192,8 +1201,8 @@ Value *ReassociatePass::RemoveFactorFromExpression(Value *V, Value *Factor) {
1192
1201
return nullptr ;
1193
1202
1194
1203
SmallVector<RepeatedValue, 8 > Tree;
1195
- bool HasNUW = true ;
1196
- MadeChange |= LinearizeExprTree (BO, Tree, RedoInsts, HasNUW );
1204
+ OverflowTracking Flags ;
1205
+ MadeChange |= LinearizeExprTree (BO, Tree, RedoInsts, Flags );
1197
1206
SmallVector<ValueEntry, 8 > Factors;
1198
1207
Factors.reserve (Tree.size ());
1199
1208
for (unsigned i = 0 , e = Tree.size (); i != e; ++i) {
@@ -1235,7 +1244,7 @@ Value *ReassociatePass::RemoveFactorFromExpression(Value *V, Value *Factor) {
1235
1244
1236
1245
if (!FoundFactor) {
1237
1246
// Make sure to restore the operands to the expression tree.
1238
- RewriteExprTree (BO, Factors, HasNUW );
1247
+ RewriteExprTree (BO, Factors, Flags );
1239
1248
return nullptr ;
1240
1249
}
1241
1250
@@ -1247,7 +1256,7 @@ Value *ReassociatePass::RemoveFactorFromExpression(Value *V, Value *Factor) {
1247
1256
RedoInsts.insert (BO);
1248
1257
V = Factors[0 ].Op ;
1249
1258
} else {
1250
- RewriteExprTree (BO, Factors, HasNUW );
1259
+ RewriteExprTree (BO, Factors, Flags );
1251
1260
V = BO;
1252
1261
}
1253
1262
@@ -2373,8 +2382,8 @@ void ReassociatePass::ReassociateExpression(BinaryOperator *I) {
2373
2382
// First, walk the expression tree, linearizing the tree, collecting the
2374
2383
// operand information.
2375
2384
SmallVector<RepeatedValue, 8 > Tree;
2376
- bool HasNUW = true ;
2377
- MadeChange |= LinearizeExprTree (I, Tree, RedoInsts, HasNUW );
2385
+ OverflowTracking Flags ;
2386
+ MadeChange |= LinearizeExprTree (I, Tree, RedoInsts, Flags );
2378
2387
SmallVector<ValueEntry, 8 > Ops;
2379
2388
Ops.reserve (Tree.size ());
2380
2389
for (const RepeatedValue &E : Tree)
@@ -2567,7 +2576,7 @@ void ReassociatePass::ReassociateExpression(BinaryOperator *I) {
2567
2576
dbgs () << ' \n ' );
2568
2577
// Now that we ordered and optimized the expressions, splat them back into
2569
2578
// the expression tree, removing any unneeded nodes.
2570
- RewriteExprTree (I, Ops, HasNUW );
2579
+ RewriteExprTree (I, Ops, Flags );
2571
2580
}
2572
2581
2573
2582
void
0 commit comments