Skip to content

Commit

Permalink
Introduce floatbv_round_to_integral_exprt
Browse files Browse the repository at this point in the history
This adds a new expression, floatbv_round_to_integral, which rounds an IEEE
754 floating-point number given as bit-vector to the nearest integer,
considering the explicitly given rounding mode.
  • Loading branch information
kroening committed Dec 21, 2024
1 parent d5cf498 commit e189f1c
Show file tree
Hide file tree
Showing 25 changed files with 587 additions and 316 deletions.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

This file was deleted.

60 changes: 60 additions & 0 deletions regression/smt2_solver/fp/roundToIntegral1.smt2
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
(set-logic FP)

(assert (not (and

; round up
(= (fp.roundToIntegral roundTowardPositive (_ NaN 11 53)) (_ NaN 11 53))
(= (fp.roundToIntegral roundTowardPositive (_ +oo 11 53)) (_ +oo 11 53))
(= (fp.roundToIntegral roundTowardPositive (_ -oo 11 53)) (_ -oo 11 53))
REQUIRE(ieee_floatt::from_double(up, 1).round_to_integral() == 1);
REQUIRE(ieee_floatt::from_double(up, 0.1).round_to_integral() == 1);
REQUIRE(ieee_floatt::from_double(up, -0.1).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(up, 10.1).round_to_integral() == 11);
REQUIRE(ieee_floatt::from_double(up, -10.1).round_to_integral() == -10);
REQUIRE(ieee_floatt::from_double(up, dmax).round_to_integral() == dmax);

; round down
(= (fp.roundToIntegral roundTowardNegative (_ NaN 11 53)) (_ NaN 11 53))
(= (fp.roundToIntegral roundTowardNegative (_ +oo 11 53)) (_ +oo 11 53))
(= (fp.roundToIntegral roundTowardNegative (_ -oo 11 53)) (_ -oo 11 53))
REQUIRE(ieee_floatt::from_double(down, 0).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(down, -0.0).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(down, 1).round_to_integral() == 1);
REQUIRE(ieee_floatt::from_double(down, 0.1).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(down, -0.1).round_to_integral() == -1);
REQUIRE(ieee_floatt::from_double(down, 10.1).round_to_integral() == 10);
REQUIRE(ieee_floatt::from_double(down, -10.1).round_to_integral() == -11);
REQUIRE(ieee_floatt::from_double(down, 0x1.0p+52).round_to_integral() == 0x1.0p+52);
REQUIRE(ieee_floatt::from_double(down, dmax).round_to_integral() == dmax);

; round to nearest ties to even
(= (fp.roundToIntegral roundNearestTiesToEven (_ NaN 11 53)) (_ NaN 11 53))
(= (fp.roundToIntegral roundNearestTiesToEven (_ +oo 11 53)) (_ +oo 11 53))
(= (fp.roundToIntegral roundNearestTiesToEven (_ -oo 11 53)) (_ -oo 11 53))
REQUIRE(ieee_floatt::from_double(even, 0).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(even, -0.0).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(even, 1).round_to_integral() == 1);
REQUIRE(ieee_floatt::from_double(even, 0.1).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(even, -0.1).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(even, 10.1).round_to_integral() == 10);
REQUIRE(ieee_floatt::from_double(even, -10.1).round_to_integral() == -10);
REQUIRE(ieee_floatt::from_double(even, 0x1.0p+52).round_to_integral() == 0x1.0p+52);
REQUIRE(ieee_floatt::from_double(even, dmax).round_to_integral() == dmax);

; round to zero
(= (fp.roundToIntegral roundTowardZero (_ NaN 11 53)) (_ NaN 11 53))
(= (fp.roundToIntegral roundTowardZero (_ +oo 11 53)) (_ +oo 11 53))
(= (fp.roundToIntegral roundTowardZero (_ -oo 11 53)) (_ -oo 11 53))
REQUIRE(ieee_floatt::from_double(zero, 0).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(zero, -0.0).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(zero, 1).round_to_integral() == 1);
REQUIRE(ieee_floatt::from_double(zero, 0.1).round_to_integral() == 0);
REQUIRE(ieee_floatt::from_double(zero, -0.1).round_to_integral() == -0.0);
REQUIRE(ieee_floatt::from_double(zero, 10.1).round_to_integral() == 10);
REQUIRE(ieee_floatt::from_double(zero, -10.1).round_to_integral() == -10);
REQUIRE(ieee_floatt::from_double(zero, 0x1.0p+52).round_to_integral() == 0x1.0p+52);
REQUIRE(ieee_floatt::from_double(zero, dmax).round_to_integral() == dmax);
)))

; should be unsat
(check-sat)
18 changes: 18 additions & 0 deletions src/ansi-c/c_typecheck_expr.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3237,6 +3237,24 @@ exprt c_typecheck_baset::do_special_functions(

return std::move(infl_expr);
}
else if(
identifier == CPROVER_PREFIX "round_to_integralf" ||
identifier == CPROVER_PREFIX "round_to_integrald" ||
identifier == CPROVER_PREFIX "round_to_integralld")
{
if(expr.arguments().size() != 2)
{
error().source_location = f_op.source_location();
error() << identifier << " expects two arguments" << eom;
throw 0;

Check warning on line 3249 in src/ansi-c/c_typecheck_expr.cpp

View check run for this annotation

Codecov / codecov/patch

src/ansi-c/c_typecheck_expr.cpp#L3247-L3249

Added lines #L3247 - L3249 were not covered by tests
}

auto round_to_integral_expr =
floatbv_round_to_integral_exprt{expr.arguments()[0], expr.arguments()[1]};
round_to_integral_expr.add_source_location() = source_location;

return std::move(round_to_integral_expr);
}
else if(
identifier == CPROVER_PREFIX "abs" || identifier == CPROVER_PREFIX "labs" ||
identifier == CPROVER_PREFIX "llabs" ||
Expand Down
3 changes: 3 additions & 0 deletions src/ansi-c/cprover_builtin_headers.h
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,9 @@ int __CPROVER_islessgreaterf(float f, float g);
int __CPROVER_islessgreaterd(double f, double g);
int __CPROVER_isunorderedf(float f, float g);
int __CPROVER_isunorderedd(double f, double g);
float __CPROVER_round_to_integralf(float, int);
double __CPROVER_round_to_integrald(double, int);
long double __CPROVER_round_to_integralld(long double, int);

// absolute value
int __CPROVER_abs(int x);
Expand Down
Loading

0 comments on commit e189f1c

Please sign in to comment.