aboutsummaryrefslogtreecommitdiffhomepage
path: root/lib/Solver
diff options
context:
space:
mode:
authorDan Liew <daniel.liew@imperial.ac.uk>2013-11-07 14:59:37 +0000
committerDan Liew <daniel.liew@imperial.ac.uk>2014-02-14 18:29:35 +0000
commit3be107cbf55f69f063795182690a5f9689e1d805 (patch)
tree1e46e72e496286a5504b5ea08f3803503f361665 /lib/Solver
parent71048bea167794f109b2f0b715b6cc1ba8b6eab9 (diff)
downloadklee-3be107cbf55f69f063795182690a5f9689e1d805.tar.gz
Fixed overshift of logical right shift by constant so that it
overshifts to zero. Test case is included.
Diffstat (limited to 'lib/Solver')
-rw-r--r--lib/Solver/STPBuilder.cpp16
-rw-r--r--lib/Solver/STPBuilder.h2
2 files changed, 7 insertions, 11 deletions
diff --git a/lib/Solver/STPBuilder.cpp b/lib/Solver/STPBuilder.cpp
index 24004c2a..a92b6f2b 100644
--- a/lib/Solver/STPBuilder.cpp
+++ b/lib/Solver/STPBuilder.cpp
@@ -174,14 +174,13 @@ ExprHandle STPBuilder::eqExpr(ExprHandle a, ExprHandle b) {
}
// logical right shift
-ExprHandle STPBuilder::bvRightShift(ExprHandle expr, unsigned amount, unsigned shiftBits) {
+ExprHandle STPBuilder::bvRightShift(ExprHandle expr, unsigned shift) {
unsigned width = vc_getBVLength(vc, expr);
- unsigned shift = amount & ((1<<shiftBits) - 1);
if (shift==0) {
return expr;
} else if (shift>=width) {
- return bvZero(width);
+ return bvZero(width); // Overshift to zero
} else {
return vc_bvConcatExpr(vc,
bvZero(shift),
@@ -236,7 +235,7 @@ ExprHandle STPBuilder::bvVarRightShift(ExprHandle expr, ExprHandle shift) {
for( int i=width-1; i>=0; i-- ) {
res = vc_iteExpr(vc,
eqExpr(shift, bvConst32(width, i)),
- bvRightShift(expr, i, width),
+ bvRightShift(expr, i),
res);
}
@@ -294,7 +293,7 @@ ExprHandle STPBuilder::constructAShrByConstant(ExprHandle expr,
ExprHandle(vc_bvConcatExpr(vc,
bvMinusOne(shift),
bvExtract(expr, width - 1, shift))),
- bvRightShift(expr, shift, shiftBits));
+ bvRightShift(expr, shift));
}
}
@@ -660,8 +659,7 @@ ExprHandle STPBuilder::constructActual(ref<Expr> e, int *width_out) {
if (bits64::isPowerOfTwo(divisor)) {
return bvRightShift(left,
- bits64::indexOfSingleBit(divisor),
- getShiftBits(*width_out));
+ bits64::indexOfSingleBit(divisor));
} else if (optimizeDivides) {
if (*width_out == 32) //only works for 32-bit division
return constructUDivByConstant( left, *width_out,
@@ -822,12 +820,10 @@ ExprHandle STPBuilder::constructActual(ref<Expr> e, int *width_out) {
case Expr::LShr: {
LShrExpr *lse = cast<LShrExpr>(e);
ExprHandle left = construct(lse->left, width_out);
- unsigned shiftBits = getShiftBits(*width_out);
assert(*width_out!=1 && "uncanonicalized lshr");
if (ConstantExpr *CE = dyn_cast<ConstantExpr>(lse->right)) {
- return bvRightShift(left, (unsigned) CE->getLimitedValue(),
- shiftBits);
+ return bvRightShift(left, (unsigned) CE->getLimitedValue());
} else {
int shiftWidth;
ExprHandle amount = construct(lse->right, &shiftWidth);
diff --git a/lib/Solver/STPBuilder.h b/lib/Solver/STPBuilder.h
index 828c267d..48bddeed 100644
--- a/lib/Solver/STPBuilder.h
+++ b/lib/Solver/STPBuilder.h
@@ -101,7 +101,7 @@ private:
//logical left and right shift (not arithmetic)
ExprHandle bvLeftShift(ExprHandle expr, unsigned shift);
- ExprHandle bvRightShift(ExprHandle expr, unsigned amount, unsigned shiftBits);
+ ExprHandle bvRightShift(ExprHandle expr, unsigned shift);
ExprHandle bvVarLeftShift(ExprHandle expr, ExprHandle shift);
ExprHandle bvVarRightShift(ExprHandle expr, ExprHandle shift);
ExprHandle bvVarArithRightShift(ExprHandle expr, ExprHandle amount, unsigned width);