aboutsummaryrefslogtreecommitdiffhomepage
path: root/lib/Solver/STPBuilder.cpp
diff options
context:
space:
mode:
authorDan Liew <delcypher@gmail.com>2014-02-14 18:49:42 +0000
committerDan Liew <delcypher@gmail.com>2014-02-14 18:49:42 +0000
commit66c064d482160e6c31ccf8ecb0aa93a7529d3da1 (patch)
tree1ed4ba4cc639b178f16d8a56bd75799d5033904e /lib/Solver/STPBuilder.cpp
parentd2f1684355e9a99545aa864e3eac81b259712a6f (diff)
parent6e4e74b99a199ead458b0b919286d667bd8e8f28 (diff)
downloadklee-66c064d482160e6c31ccf8ecb0aa93a7529d3da1.tar.gz
Merge pull request #64 from delcypher/overshift-fix
Overshift fixes
Diffstat (limited to 'lib/Solver/STPBuilder.cpp')
-rw-r--r--lib/Solver/STPBuilder.cpp102
1 files changed, 48 insertions, 54 deletions
diff --git a/lib/Solver/STPBuilder.cpp b/lib/Solver/STPBuilder.cpp
index 90252656..ad0df3c2 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),
@@ -190,14 +189,13 @@ ExprHandle STPBuilder::bvRightShift(ExprHandle expr, unsigned amount, unsigned s
}
// logical left shift
-ExprHandle STPBuilder::bvLeftShift(ExprHandle expr, unsigned amount, unsigned shiftBits) {
+ExprHandle STPBuilder::bvLeftShift(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 {
// stp shift does "expr @ [0 x s]" which we then have to extract,
// rolling our own gives slightly smaller exprs
@@ -208,96 +206,97 @@ ExprHandle STPBuilder::bvLeftShift(ExprHandle expr, unsigned amount, unsigned sh
}
// left shift by a variable amount on an expression of the specified width
-ExprHandle STPBuilder::bvVarLeftShift(ExprHandle expr, ExprHandle amount, unsigned width) {
+ExprHandle STPBuilder::bvVarLeftShift(ExprHandle expr, ExprHandle shift) {
+ unsigned width = vc_getBVLength(vc, expr);
ExprHandle res = bvZero(width);
- int shiftBits = getShiftBits( width );
-
- //get the shift amount (looking only at the bits appropriate for the given width)
- ExprHandle shift = vc_bvExtract( vc, amount, shiftBits - 1, 0 );
-
//construct a big if-then-elif-elif-... with one case per possible shift amount
for( int i=width-1; i>=0; i-- ) {
res = vc_iteExpr(vc,
- eqExpr(shift, bvConst32(shiftBits, i)),
- bvLeftShift(expr, i, shiftBits),
+ eqExpr(shift, bvConst32(width, i)),
+ bvLeftShift(expr, i),
res);
}
+
+ // If overshifting, shift to zero
+ res = vc_iteExpr(vc,
+ vc_bvLtExpr(vc, shift, bvConst32(vc_getBVLength(vc,shift), width)),
+ res,
+ bvZero(width));
return res;
}
// logical right shift by a variable amount on an expression of the specified width
-ExprHandle STPBuilder::bvVarRightShift(ExprHandle expr, ExprHandle amount, unsigned width) {
+ExprHandle STPBuilder::bvVarRightShift(ExprHandle expr, ExprHandle shift) {
+ unsigned width = vc_getBVLength(vc, expr);
ExprHandle res = bvZero(width);
- int shiftBits = getShiftBits( width );
-
- //get the shift amount (looking only at the bits appropriate for the given width)
- ExprHandle shift = vc_bvExtract( vc, amount, shiftBits - 1, 0 );
-
//construct a big if-then-elif-elif-... with one case per possible shift amount
for( int i=width-1; i>=0; i-- ) {
res = vc_iteExpr(vc,
- eqExpr(shift, bvConst32(shiftBits, i)),
- bvRightShift(expr, i, shiftBits),
+ eqExpr(shift, bvConst32(width, i)),
+ bvRightShift(expr, i),
res);
}
+ // If overshifting, shift to zero
+ res = vc_iteExpr(vc,
+ vc_bvLtExpr(vc, shift, bvConst32(vc_getBVLength(vc,shift), width)),
+ res,
+ bvZero(width));
return res;
}
// arithmetic right shift by a variable amount on an expression of the specified width
-ExprHandle STPBuilder::bvVarArithRightShift(ExprHandle expr, ExprHandle amount, unsigned width) {
- int shiftBits = getShiftBits( width );
-
- //get the shift amount (looking only at the bits appropriate for the given width)
- ExprHandle shift = vc_bvExtract( vc, amount, shiftBits - 1, 0 );
+ExprHandle STPBuilder::bvVarArithRightShift(ExprHandle expr, ExprHandle shift) {
+ unsigned width = vc_getBVLength(vc, expr);
//get the sign bit to fill with
ExprHandle signedBool = bvBoolExtract(expr, width-1);
//start with the result if shifting by width-1
- ExprHandle res = constructAShrByConstant(expr, width-1, signedBool, shiftBits);
+ ExprHandle res = constructAShrByConstant(expr, width-1, signedBool);
//construct a big if-then-elif-elif-... with one case per possible shift amount
// XXX more efficient to move the ite on the sign outside all exprs?
// XXX more efficient to sign extend, right shift, then extract lower bits?
for( int i=width-2; i>=0; i-- ) {
res = vc_iteExpr(vc,
- eqExpr(shift, bvConst32(shiftBits,i)),
+ eqExpr(shift, bvConst32(width,i)),
constructAShrByConstant(expr,
i,
- signedBool,
- shiftBits),
+ signedBool),
res);
}
+ // If overshifting, shift to zero
+ res = vc_iteExpr(vc,
+ vc_bvLtExpr(vc, shift, bvConst32(vc_getBVLength(vc,shift), width)),
+ res,
+ bvZero(width));
return res;
}
ExprHandle STPBuilder::constructAShrByConstant(ExprHandle expr,
- unsigned amount,
- ExprHandle isSigned,
- unsigned shiftBits) {
+ unsigned shift,
+ ExprHandle isSigned) {
unsigned width = vc_getBVLength(vc, expr);
- unsigned shift = amount & ((1<<shiftBits) - 1);
if (shift==0) {
return expr;
} else if (shift>=width-1) {
- return vc_iteExpr(vc, isSigned, bvMinusOne(width), bvZero(width));
+ return bvZero(width); // Overshift to zero
} else {
return vc_iteExpr(vc,
isSigned,
ExprHandle(vc_bvConcatExpr(vc,
bvMinusOne(shift),
bvExtract(expr, width - 1, shift))),
- bvRightShift(expr, shift, shiftBits));
+ bvRightShift(expr, shift));
}
}
ExprHandle STPBuilder::constructMulByConstant(ExprHandle expr, unsigned width, uint64_t x) {
- unsigned shiftBits = getShiftBits(width);
uint64_t add, sub;
ExprHandle res = 0;
@@ -313,7 +312,7 @@ ExprHandle STPBuilder::constructMulByConstant(ExprHandle expr, unsigned width, u
if ((add&bit) || (sub&bit)) {
assert(!((add&bit) && (sub&bit)) && "invalid mult constants");
- ExprHandle op = bvLeftShift(expr, j, shiftBits);
+ ExprHandle op = bvLeftShift(expr, j);
if (add&bit) {
if (res) {
@@ -367,9 +366,9 @@ ExprHandle STPBuilder::constructUDivByConstant(ExprHandle expr_n, unsigned width
// n/d = (((n - t1) >> sh1) + t1) >> sh2;
ExprHandle n_minus_t1 = vc_bvMinusExpr( vc, width, expr_n, t1 );
- ExprHandle shift_sh1 = bvVarRightShift( n_minus_t1, expr_sh1, 32 );
+ ExprHandle shift_sh1 = bvVarRightShift( n_minus_t1, expr_sh1);
ExprHandle plus_t1 = vc_bvPlusExpr( vc, width, shift_sh1, t1 );
- ExprHandle res = bvVarRightShift( plus_t1, expr_sh2, 32 );
+ ExprHandle res = bvVarRightShift( plus_t1, expr_sh2);
return res;
}
@@ -407,7 +406,7 @@ ExprHandle STPBuilder::constructSDivByConstant(ExprHandle expr_n, unsigned width
// Improved variable arithmetic right shift: sign extend, shift,
// extract.
ExprHandle extend_npm = vc_bvSignExtend( vc, n_plus_mulsh, 64 );
- ExprHandle shift_npm = bvVarRightShift( extend_npm, expr_shpost, 64 );
+ ExprHandle shift_npm = bvVarRightShift( extend_npm, expr_shpost);
ExprHandle shift_shpost = vc_bvExtract( vc, shift_npm, 31, 0 ); //lower 32-bits
// XSIGN(n) is -1 if n is negative, positive one otherwise
@@ -659,8 +658,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,
@@ -810,28 +808,25 @@ ExprHandle STPBuilder::constructActual(ref<Expr> e, int *width_out) {
assert(*width_out!=1 && "uncanonicalized shl");
if (ConstantExpr *CE = dyn_cast<ConstantExpr>(se->right)) {
- return bvLeftShift(left, (unsigned) CE->getLimitedValue(),
- getShiftBits(*width_out));
+ return bvLeftShift(left, (unsigned) CE->getLimitedValue());
} else {
int shiftWidth;
ExprHandle amount = construct(se->right, &shiftWidth);
- return bvVarLeftShift( left, amount, *width_out );
+ return bvVarLeftShift( left, amount);
}
}
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);
- return bvVarRightShift( left, amount, *width_out );
+ return bvVarRightShift( left, amount);
}
}
@@ -843,12 +838,11 @@ ExprHandle STPBuilder::constructActual(ref<Expr> e, int *width_out) {
if (ConstantExpr *CE = dyn_cast<ConstantExpr>(ase->right)) {
unsigned shift = (unsigned) CE->getLimitedValue();
ExprHandle signedBool = bvBoolExtract(left, *width_out-1);
- return constructAShrByConstant(left, shift, signedBool,
- getShiftBits(*width_out));
+ return constructAShrByConstant(left, shift, signedBool);
} else {
int shiftWidth;
ExprHandle amount = construct(ase->right, &shiftWidth);
- return bvVarArithRightShift( left, amount, *width_out );
+ return bvVarArithRightShift( left, amount);
}
}