-
Notifications
You must be signed in to change notification settings - Fork 50
/
Copy pathFloatingPointNumber.java
238 lines (216 loc) · 9.03 KB
/
FloatingPointNumber.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
// This file is part of JavaSMT,
// an API wrapper for a collection of SMT solvers:
// https://github.com/sosy-lab/java-smt
//
// SPDX-FileCopyrightText: 2024 Dirk Beyer <https://www.sosy-lab.org>
//
// SPDX-License-Identifier: Apache-2.0
package org.sosy_lab.java_smt.api;
import com.google.auto.value.AutoValue;
import com.google.common.base.Preconditions;
import com.google.errorprone.annotations.Immutable;
import com.google.errorprone.annotations.InlineMe;
import java.math.BigInteger;
import java.util.BitSet;
/**
* Represents a floating-point number with customizable precision, consisting of sign, exponent, and
* mantissa components.
*/
@Immutable
@AutoValue
public abstract class FloatingPointNumber {
public static final int SINGLE_PRECISION_EXPONENT_SIZE = 8;
public static final int SINGLE_PRECISION_MANTISSA_SIZE = 23;
public static final int DOUBLE_PRECISION_EXPONENT_SIZE = 11;
public static final int DOUBLE_PRECISION_MANTISSA_SIZE = 52;
public enum Sign {
POSITIVE,
NEGATIVE;
/**
* get the Sign for a flag.
*
* @param isNegative whether the sign is negative (TRUE) or positive (FALSE).
*/
public static Sign of(boolean isNegative) {
return isNegative ? NEGATIVE : POSITIVE;
}
public boolean isNegative() {
return this == NEGATIVE;
}
}
/**
* The sign of the floating-point number.
*
* @return whether the number is positive (FALSE) or negative (TRUE).
*/
@Deprecated(
since = "2025.01, because using a boolean flag as signBit is misleading",
forRemoval = true)
@InlineMe(
replacement = "this.getMathSign() == Sign.NEGATIVE",
imports = "org.sosy_lab.java_smt.api.FloatingPointNumber.Sign")
public final boolean getSign() {
return getMathSign() == Sign.NEGATIVE;
}
/** The sign of the floating-point number, i.e. whether it is positive or negative. */
public abstract Sign getMathSign();
/**
* The exponent of the floating-point number, given as numeric value from binary representation.
* The number is unsigned (not negative) and includes a bias of 2^(exponentSize-1)-1 that is used
* in IEEE 754.
*/
public abstract BigInteger getExponent();
/**
* The mantissa (aka significand) of the floating-point number, given as numeric value from binary
* representation. The mantissa does not include the hidden bit that is used to denote normalized
* numbers in IEEE 754.
*/
public abstract BigInteger getMantissa();
public abstract int getExponentSize();
public abstract int getMantissaSize();
/**
* Get a floating-point number with the given sign, exponent, and mantissa.
*
* @param sign the sign-bit of the floating-point number as specified by IEEE 754, aka FALSE for
* positive and TRUE for negative
* @param exponent the exponent of the floating-point number, given as unsigned (not negative)
* number, including a bias of 2^(exponentSize-1)-1
* @param mantissa the mantissa of the floating-point number, given as unsigned (not negative)
* number without hidden bit
* @param exponentSize the (maximum) size of the exponent in bits
* @param mantissaSize the (maximum) size of the mantissa in bits
* @see #of(Sign, BigInteger, BigInteger, int, int)
*/
@Deprecated(
since = "2025.01, because using a boolean flag as signBit is misleading",
forRemoval = true)
@InlineMe(
replacement =
"FloatingPointNumber.of(Sign.of(sign), exponent, mantissa, exponentSize, mantissaSize)",
imports = {
"org.sosy_lab.java_smt.api.FloatingPointNumber",
"org.sosy_lab.java_smt.api.FloatingPointNumber.Sign"
})
public static FloatingPointNumber of(
boolean sign, BigInteger exponent, BigInteger mantissa, int exponentSize, int mantissaSize) {
return of(Sign.of(sign), exponent, mantissa, exponentSize, mantissaSize);
}
/**
* Get a floating-point number with the given sign, exponent, and mantissa.
*
* @param sign the sign of the floating-point number
* @param exponent the exponent of the floating-point number, given as unsigned (not negative)
* number, including a bias of 2^(exponentSize-1)-1
* @param mantissa the mantissa of the floating-point number, given as unsigned (not negative)
* number without hidden bit
* @param exponentSize the (maximum) size of the exponent in bits
* @param mantissaSize the (maximum) size of the mantissa in bits
*/
public static FloatingPointNumber of(
Sign sign, BigInteger exponent, BigInteger mantissa, int exponentSize, int mantissaSize) {
Preconditions.checkArgument(exponent.bitLength() <= exponentSize);
Preconditions.checkArgument(mantissa.bitLength() <= mantissaSize);
Preconditions.checkArgument(exponent.compareTo(BigInteger.ZERO) >= 0);
Preconditions.checkArgument(mantissa.compareTo(BigInteger.ZERO) >= 0);
return new AutoValue_FloatingPointNumber(sign, exponent, mantissa, exponentSize, mantissaSize);
}
/**
* Get a floating-point number encoded as bitvector as defined by IEEE 754.
*
* @param bits the bit-representation of the floating-point number, consisting of sign bit,
* exponent (without bias) and mantissa (without hidden bit) in this exact ordering
* @param exponentSize the size of the exponent in bits
* @param mantissaSize the size of the mantissa in bits
*/
public static FloatingPointNumber of(String bits, int exponentSize, int mantissaSize) {
Preconditions.checkArgument(0 < exponentSize);
Preconditions.checkArgument(0 < mantissaSize);
Preconditions.checkArgument(
bits.length() == 1 + exponentSize + mantissaSize,
"Bitsize (%s) of floating point numeral does not match the size of sign, exponent and "
+ "mantissa (%s + %s + %s).",
bits.length(),
1,
exponentSize,
mantissaSize);
Preconditions.checkArgument(bits.chars().allMatch(c -> c == '0' || c == '1'));
Sign sign = Sign.of(bits.charAt(0) == '1');
BigInteger exponent = new BigInteger(bits.substring(1, 1 + exponentSize), 2);
BigInteger mantissa =
new BigInteger(bits.substring(1 + exponentSize, 1 + exponentSize + mantissaSize), 2);
return of(sign, exponent, mantissa, exponentSize, mantissaSize);
}
/**
* Returns true if this floating-point number is an IEEE-754-2008 single precision type with 32
* bits length consisting of an 8 bit exponent, a 23 bit mantissa and a single sign bit.
*
* @return true for IEEE-754 single precision type, false otherwise.
*/
public boolean isIEEE754SinglePrecision() {
return getExponentSize() == SINGLE_PRECISION_EXPONENT_SIZE
&& getMantissaSize() == SINGLE_PRECISION_MANTISSA_SIZE;
}
/**
* Returns true if this floating-point number is an IEEE-754-2008 double precision type with 64
* bits length consisting of an 11 bit exponent, a 52 bit mantissa and a single sign bit.
*
* @return true for IEEE-754 double precision type, false otherwise.
*/
public boolean isIEEE754DoublePrecision() {
return getExponentSize() == DOUBLE_PRECISION_EXPONENT_SIZE
&& getMantissaSize() == DOUBLE_PRECISION_MANTISSA_SIZE;
}
/** compute a representation as Java-based float value, if possible. */
public float floatValue() {
Preconditions.checkArgument(
isIEEE754SinglePrecision(),
"Can not represent floating point number %s as Java-based float value.",
this);
var bits = getBits();
return Float.intBitsToFloat(bits.isEmpty() ? 0 : (int) bits.toLongArray()[0]);
}
/** compute a representation as Java-based double value, if possible. */
public double doubleValue() {
Preconditions.checkArgument(
isIEEE754SinglePrecision() || isIEEE754DoublePrecision(),
"Can not represent floating point number %s as Java-based double value.",
this);
if (isIEEE754SinglePrecision()) {
// lets be nice to the user and automatically convert from single to double precision
return floatValue();
}
var bits = getBits();
return Double.longBitsToDouble(bits.isEmpty() ? 0 : getBits().toLongArray()[0]);
}
private BitSet getBits() {
var mantissaSize = getMantissaSize();
var exponentSize = getExponentSize();
var mantissa = getMantissa();
var exponent = getExponent();
var bits = new BitSet(1 + exponentSize + mantissaSize);
if (getMathSign().isNegative()) {
bits.set(exponentSize + mantissaSize); // if negative, set first bit to 1
}
for (int i = 0; i < exponentSize; i++) {
bits.set(mantissaSize + i, exponent.testBit(i));
}
for (int i = 0; i < mantissaSize; i++) {
bits.set(i, mantissa.testBit(i));
}
return bits;
}
/**
* Return a bit-representation of sign-bit, exponent, and mantissa, i.e., a concatenation of their
* bit-representations in this exact ordering.
*/
@Override
public final String toString() {
var length = 1 + getExponentSize() + getMantissaSize();
var str = new StringBuilder(length);
var bits = getBits();
for (int i = 0; i < length; i++) {
str.append(bits.get(i) ? '1' : '0');
}
return str.reverse().toString();
}
}