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
|
# Require exact values up to
FPBITS = 256
# Overkill accuracy
F = RealField(400)
def BaseFPBits(bits, capacity):
return bits * capacity - int(ceil(F(log(sum(binomial(2**bits - 1, i) for i in range(capacity+1)), 2))))
def Log2Factorial(capacity):
return int(floor(log(factorial(capacity), 2)))
print("uint64_t BaseFPBits(uint32_t bits, uint32_t capacity) {")
print(" // Correction table for low bits/capacities")
TBLS={}
FARS={}
SKIPS={}
for bits in range(1, 32):
TBL = []
for capacity in range(1, min(2**bits, FPBITS)):
exact = BaseFPBits(bits, capacity)
approx = Log2Factorial(capacity)
TBL.append((exact, approx))
MIN = 10000000000
while len(TBL) and ((TBL[-1][0] == TBL[-1][1]) or (TBL[-1][0] >= FPBITS and TBL[-1][1] >= FPBITS)):
MIN = min(MIN, TBL[-1][0] - TBL[-1][1])
TBL.pop()
while len(TBL) and (TBL[-1][0] - TBL[-1][1] == MIN):
TBL.pop()
SKIP = 0
while SKIP < len(TBL) and TBL[SKIP][0] == TBL[SKIP][1]:
SKIP += 1
DIFFS = [TBL[i][0] - TBL[i][1] for i in range(SKIP, len(TBL))]
if len(DIFFS) > 0 and len(DIFFS) * Integer(max(DIFFS)).nbits() > 64:
print(" static constexpr uint8_t ADD%i[] = {%s};" % (bits, ", ".join(("%i" % (TBL[i][0] - TBL[i][1])) for i in range(SKIP, len(TBL)))))
TBLS[bits] = DIFFS
FARS[bits] = MIN
SKIPS[bits] = SKIP
print("")
print(" if (capacity == 0) return 0;")
print(" uint64_t ret = 0;")
print(" if (bits < 32 && capacity >= (1U << bits)) {")
print(" ret = uint64_t{bits} * (capacity - (1U << bits) + 1);")
print(" capacity = (1U << bits) - 1;")
print(" }")
print(" ret += Log2Factorial(capacity);")
print(" switch (bits) {")
for bits in sorted(TBLS.keys()):
if len(TBLS[bits]) == 0:
continue
width = Integer(max(TBLS[bits])).nbits()
if len(TBLS[bits]) == 1:
add = "%i" % TBLS[bits][0]
elif len(TBLS[bits]) * width <= 64:
code = sum((2**(width*i) * TBLS[bits][i]) for i in range(len(TBLS[bits])))
if width == 1:
add = "(0x%x >> (capacity - %i)) & 1" % (code, 1 + SKIPS[bits])
else:
add = "(0x%x >> %i * (capacity - %i)) & %i" % (code, width, 1 + SKIPS[bits], 2**width - 1)
else:
add = "ADD%i[capacity - %i]" % (bits, 1 + SKIPS[bits])
if len(TBLS[bits]) + SKIPS[bits] == 2**bits - 1:
print(" case %i: return ret + (capacity <= %i ? 0 : %s);" % (bits, SKIPS[bits], add))
else:
print(" case %i: return ret + (capacity <= %i ? 0 : capacity > %i ? %i : %s);" % (bits, SKIPS[bits], len(TBLS[bits]) + SKIPS[bits], FARS[bits], add))
print(" default: return ret;")
print(" }")
print("}")
print("void TestBaseFPBits() {")
print(" static constexpr uint16_t TBL[20][100] = {%s};" % (", ".join("{" + ", ".join(("%i" % BaseFPBits(bits, capacity)) for capacity in range(0, 100)) + "}" for bits in range(1, 21))))
print(" for (int bits = 1; bits <= 20; ++bits) {")
print(" for (int capacity = 0; capacity < 100; ++capacity) {")
print(" uint64_t computed = BaseFPBits(bits, capacity), exact = TBL[bits - 1][capacity];")
print(" CHECK(exact == computed || (exact >= 256 && computed >= 256));")
print(" }")
print(" }")
print("}")
|