summaryrefslogtreecommitdiffstats
path: root/clang/lib
diff options
context:
space:
mode:
authorEli Friedman <eli.friedman@gmail.com>2012-04-16 19:23:57 +0000
committerEli Friedman <eli.friedman@gmail.com>2012-04-16 19:23:57 +0000
commit2f5b7c542e688701037bedb040c2b39d021acf14 (patch)
tree43ddbf9327b9ccb10f201458d01d4e70d731fee9 /clang/lib
parent44d2497138095224591706b245eb10848e111c0e (diff)
downloadbcm5719-llvm-2f5b7c542e688701037bedb040c2b39d021acf14.tar.gz
bcm5719-llvm-2f5b7c542e688701037bedb040c2b39d021acf14.zip
Per Richard's comments on r154794, add the checks necessary to handle constant-folding relational comparisons safely in case the user is using -fwrapv or equivalent.
llvm-svn: 154849
Diffstat (limited to 'clang/lib')
-rw-r--r--clang/lib/AST/ExprConstant.cpp15
1 files changed, 13 insertions, 2 deletions
diff --git a/clang/lib/AST/ExprConstant.cpp b/clang/lib/AST/ExprConstant.cpp
index 4839c299765..66a88b065ce 100644
--- a/clang/lib/AST/ExprConstant.cpp
+++ b/clang/lib/AST/ExprConstant.cpp
@@ -5090,8 +5090,6 @@ bool IntExprEvaluator::VisitBinaryOperator(const BinaryOperator *E) {
// The comparison here must be unsigned, and performed with the same
// width as the pointer.
- // FIXME: Knowing the base is the same for the LHS and RHS isn't enough
- // for relational operators.
unsigned PtrSize = Info.Ctx.getTypeSize(LHSTy);
uint64_t CompareLHS = LHSOffset.getQuantity();
uint64_t CompareRHS = RHSOffset.getQuantity();
@@ -5100,6 +5098,19 @@ bool IntExprEvaluator::VisitBinaryOperator(const BinaryOperator *E) {
CompareLHS &= Mask;
CompareRHS &= Mask;
+ // If there is a base and this is a relational operator, we can only
+ // compare pointers within the object in question; otherwise, the result
+ // depends on where the object is located in memory.
+ if (!LHSValue.Base.isNull() && E->isRelationalOp()) {
+ QualType BaseTy = getType(LHSValue.Base);
+ if (BaseTy->isIncompleteType())
+ return Error(E);
+ CharUnits Size = Info.Ctx.getTypeSizeInChars(BaseTy);
+ uint64_t OffsetLimit = Size.getQuantity();
+ if (CompareLHS > OffsetLimit || CompareRHS > OffsetLimit)
+ return Error(E);
+ }
+
switch (E->getOpcode()) {
default: llvm_unreachable("missing comparison operator");
case BO_LT: return Success(CompareLHS < CompareRHS, E);
OpenPOWER on IntegriCloud