use crate::common::f_fmla;
use crate::dekker::Dekker;
use crate::exp2::ldexp;
const LN2H: f64 = f64::from_bits(0x3fe62e42fefa39ef);
const LN2L: f64 = f64::from_bits(0x3c7abc9e3b39803f);
struct Exp2m1 {
exp: Dekker,
err: f64,
}
pub(crate) static EXP_M1_2_TABLE1: [(u64, u64); 64] = [
(0x0000000000000000, 0x3ff0000000000000),
(0xbc719083535b085d, 0x3ff02c9a3e778061),
(0x3c8d73e2a475b465, 0x3ff059b0d3158574),
(0x3c6186be4bb284ff, 0x3ff0874518759bc8),
(0x3c98a62e4adc610b, 0x3ff0b5586cf9890f),
(0x3c403a1727c57b53, 0x3ff0e3ec32d3d1a2),
(0xbc96c51039449b3a, 0x3ff11301d0125b51),
(0xbc932fbf9af1369e, 0x3ff1429aaea92de0),
(0xbc819041b9d78a76, 0x3ff172b83c7d517b),
(0x3c8e5b4c7b4968e4, 0x3ff1a35beb6fcb75),
(0x3c9e016e00a2643c, 0x3ff1d4873168b9aa),
(0x3c8dc775814a8495, 0x3ff2063b88628cd6),
(0x3c99b07eb6c70573, 0x3ff2387a6e756238),
(0x3c82bd339940e9d9, 0x3ff26b4565e27cdd),
(0x3c8612e8afad1255, 0x3ff29e9df51fdee1),
(0x3c90024754db41d5, 0x3ff2d285a6e4030b),
(0x3c86f46ad23182e4, 0x3ff306fe0a31b715),
(0x3c932721843659a6, 0x3ff33c08b26416ff),
(0xbc963aeabf42eae2, 0x3ff371a7373aa9cb),
(0xbc75e436d661f5e3, 0x3ff3a7db34e59ff7),
(0x3c8ada0911f09ebc, 0x3ff3dea64c123422),
(0xbc5ef3691c309278, 0x3ff4160a21f72e2a),
(0x3c489b7a04ef80d0, 0x3ff44e086061892d),
(0x3c73c1a3b69062f0, 0x3ff486a2b5c13cd0),
(0x3c7d4397afec42e2, 0x3ff4bfdad5362a27),
(0xbc94b309d25957e3, 0x3ff4f9b2769d2ca7),
(0xbc807abe1db13cad, 0x3ff5342b569d4f82),
(0x3c99bb2c011d93ad, 0x3ff56f4736b527da),
(0x3c96324c054647ad, 0x3ff5ab07dd485429),
(0x3c9ba6f93080e65e, 0x3ff5e76f15ad2148),
(0xbc9383c17e40b497, 0x3ff6247eb03a5585),
(0xbc9bb60987591c34, 0x3ff6623882552225),
(0xbc9bdd3413b26456, 0x3ff6a09e667f3bcd),
(0xbc6bbe3a683c88ab, 0x3ff6dfb23c651a2f),
(0xbc816e4786887a99, 0x3ff71f75e8ec5f74),
(0xbc90245957316dd3, 0x3ff75feb564267c9),
(0xbc841577ee04992f, 0x3ff7a11473eb0187),
(0x3c705d02ba15797e, 0x3ff7e2f336cf4e62),
(0xbc9d4c1dd41532d8, 0x3ff82589994cce13),
(0xbc9fc6f89bd4f6ba, 0x3ff868d99b4492ed),
(0x3c96e9f156864b27, 0x3ff8ace5422aa0db),
(0x3c85cc13a2e3976c, 0x3ff8f1ae99157736),
(0xbc675fc781b57ebc, 0x3ff93737b0cdc5e5),
(0xbc9d185b7c1b85d1, 0x3ff97d829fde4e50),
(0x3c7c7c46b071f2be, 0x3ff9c49182a3f090),
(0xbc9359495d1cd533, 0x3ffa0c667b5de565),
(0xbc9d2f6edb8d41e1, 0x3ffa5503b23e255d),
(0x3c90fac90ef7fd31, 0x3ffa9e6b5579fdbf),
(0x3c97a1cd345dcc81, 0x3ffae89f995ad3ad),
(0xbc62805e3084d708, 0x3ffb33a2b84f15fb),
(0xbc75584f7e54ac3b, 0x3ffb7f76f2fb5e47),
(0x3c823dd07a2d9e84, 0x3ffbcc1e904bc1d2),
(0x3c811065895048dd, 0x3ffc199bdd85529c),
(0x3c92884dff483cad, 0x3ffc67f12e57d14b),
(0x3c7503cbd1e949db, 0x3ffcb720dcef9069),
(0xbc9cbc3743797a9c, 0x3ffd072d4a07897c),
(0x3c82ed02d75b3707, 0x3ffd5818dcfba487),
(0x3c9c2300696db532, 0x3ffda9e603db3285),
(0xbc91a5cd4f184b5c, 0x3ffdfc97337b9b5f),
(0x3c839e8980a9cc8f, 0x3ffe502ee78b3ff6),
(0xbc9e9c23179c2893, 0x3ffea4afa2a490da),
(0x3c9dc7f486a4b6b0, 0x3ffefa1bee615a27),
(0x3c99d3e12dd8a18b, 0x3fff50765b6e4540),
(0x3c874853f3a5931e, 0x3fffa7c1819e90d8),
];
pub(crate) static EXP_M1_2_TABLE2: [(u64, u64); 64] = [
(0x0000000000000000, 0x3ff0000000000000),
(0x3c9ae8e38c59c72a, 0x3ff000b175effdc7),
(0xbc57b5d0d58ea8f4, 0x3ff00162f3904052),
(0x3c94115cb6b16a8e, 0x3ff0021478e11ce6),
(0xbc8d7c96f201bb2f, 0x3ff002c605e2e8cf),
(0x3c984711d4c35e9f, 0x3ff003779a95f959),
(0xbc80484245243777, 0x3ff0042936faa3d8),
(0xbc94b237da2025f9, 0x3ff004dadb113da0),
(0xbc75e00e62d6b30d, 0x3ff0058c86da1c0a),
(0x3c9a1d6cedbb9481, 0x3ff0063e3a559473),
(0xbc94acf197a00142, 0x3ff006eff583fc3d),
(0xbc6eaf2ea42391a5, 0x3ff007a1b865a8ca),
(0x3c7da93f90835f75, 0x3ff0085382faef83),
(0xbc86a79084ab093c, 0x3ff00905554425d4),
(0x3c986364f8fbe8f8, 0x3ff009b72f41a12b),
(0xbc882e8e14e3110e, 0x3ff00a6910f3b6fd),
(0xbc84f6b2a7609f71, 0x3ff00b1afa5abcbf),
(0xbc7e1a258ea8f71b, 0x3ff00bcceb7707ec),
(0x3c74362ca5bc26f1, 0x3ff00c7ee448ee02),
(0x3c9095a56c919d02, 0x3ff00d30e4d0c483),
(0xbc6406ac4e81a645, 0x3ff00de2ed0ee0f5),
(0x3c9b5a6902767e09, 0x3ff00e94fd0398e0),
(0xbc991b2060859321, 0x3ff00f4714af41d3),
(0x3c8427068ab22306, 0x3ff00ff93412315c),
(0x3c9c1d0660524e08, 0x3ff010ab5b2cbd11),
(0xbc9e7bdfb3204be8, 0x3ff0115d89ff3a8b),
(0x3c8843aa8b9cbbc6, 0x3ff0120fc089ff63),
(0xbc734104ee7edae9, 0x3ff012c1fecd613b),
(0xbc72b6aeb6176892, 0x3ff0137444c9b5b5),
(0x3c7a8cd33b8a1bb3, 0x3ff01426927f5278),
(0x3c72edc08e5da99a, 0x3ff014d8e7ee8d2f),
(0x3c857ba2dc7e0c73, 0x3ff0158b4517bb88),
(0x3c9b61299ab8cdb7, 0x3ff0163da9fb3335),
(0xbc990565902c5f44, 0x3ff016f0169949ed),
(0x3c870fc41c5c2d53, 0x3ff017a28af25567),
(0x3c94b9a6e145d76c, 0x3ff018550706ab62),
(0xbc7008eff5142bf9, 0x3ff019078ad6a19f),
(0xbc977669f033c7de, 0x3ff019ba16628de2),
(0xbc909bb78eeead0a, 0x3ff01a6ca9aac5f3),
(0x3c9371231477ece5, 0x3ff01b1f44af9f9e),
(0x3c75e7626621eb5b, 0x3ff01bd1e77170b4),
(0xbc9bc72b100828a5, 0x3ff01c8491f08f08),
(0xbc6ce39cbbab8bbe, 0x3ff01d37442d5070),
(0x3c816996709da2e2, 0x3ff01de9fe280ac8),
(0xbc8c11f5239bf535, 0x3ff01e9cbfe113ef),
(0x3c8e1d4eb5edc6b3, 0x3ff01f4f8958c1c6),
(0xbc9afb99946ee3f0, 0x3ff020025a8f6a35),
(0xbc98f06d8a148a32, 0x3ff020b533856324),
(0xbc82bf310fc54eb6, 0x3ff02168143b0281),
(0xbc9c95a035eb4175, 0x3ff0221afcb09e3e),
(0xbc9491793e46834d, 0x3ff022cdece68c4f),
(0xbc73e8d0d9c49091, 0x3ff02380e4dd22ad),
(0xbc9314aa16278aa3, 0x3ff02433e494b755),
(0x3c848daf888e9651, 0x3ff024e6ec0da046),
(0x3c856dc8046821f4, 0x3ff02599fb483385),
(0x3c945b42356b9d47, 0x3ff0264d1244c719),
(0xbc7082ef51b61d7e, 0x3ff027003103b10e),
(0x3c72106ed0920a34, 0x3ff027b357854772),
(0xbc9fd4cf26ea5d0f, 0x3ff0286685c9e059),
(0xbc909f8775e78084, 0x3ff02919bbd1d1d8),
(0x3c564cbba902ca27, 0x3ff029ccf99d720a),
(0x3c94383ef231d207, 0x3ff02a803f2d170d),
(0x3c94a47a505b3a47, 0x3ff02b338c811703),
(0x3c9e47120223467f, 0x3ff02be6e199c811),
];
#[inline]
fn q_1(dz: Dekker) -> Dekker {
const Q_1: [u64; 5] = [
0x3ff0000000000000,
0x3ff0000000000000,
0x3fe0000000000000,
0x3fc5555555995d37,
0x3fa55555558489dc,
];
let z = dz.to_f64();
let mut q = f_fmla(f64::from_bits(Q_1[4]), dz.hi, f64::from_bits(Q_1[3]));
q = f_fmla(q, z, f64::from_bits(Q_1[2]));
let mut p0 = Dekker::from_exact_add(f64::from_bits(Q_1[1]), q * z);
p0 = Dekker::quick_mult(dz, p0);
p0 = Dekker::f64_add(f64::from_bits(Q_1[0]), p0);
p0
}
#[inline]
fn exp1(x: Dekker) -> Dekker {
const INVLOG2: f64 = f64::from_bits(0x40b71547652b82fe);
let k = (x.hi * INVLOG2).round_ties_even();
const LOG2H: f64 = f64::from_bits(0x3f262e42fefa39ef);
const LOG2L: f64 = f64::from_bits(0x3bbabc9e3b39803f);
let mut zk = Dekker::from_exact_mult(LOG2H, k);
zk.lo = f_fmla(k, LOG2L, zk.lo);
let mut yz = Dekker::from_exact_add(x.hi - zk.hi, x.lo);
yz.lo -= zk.lo;
let ik: i64 = k as i64;
let im: i64 = (ik >> 12).wrapping_add(0x3ff);
let i2: i64 = (ik >> 6) & 0x3f;
let i1: i64 = ik & 0x3f;
let t1 = Dekker::from_bit_pair(EXP_M1_2_TABLE1[i2 as usize]);
let t2 = Dekker::from_bit_pair(EXP_M1_2_TABLE2[i1 as usize]);
let p0 = Dekker::quick_mult(t2, t1);
let mut q = q_1(yz);
q = Dekker::quick_mult(p0, q);
let mut du = (im as u64).wrapping_shl(52);
if im == 0x7ff {
q.hi *= 2.0;
q.lo *= 2.0;
du = (im.wrapping_sub(1) as u64).wrapping_shl(52);
}
q.hi *= f64::from_bits(du);
q.lo *= f64::from_bits(du);
q
}
#[inline]
fn exp2m1_fast(x: f64, tiny: bool) -> Exp2m1 {
if tiny {
return exp2m1_fast_tiny(x);
}
let mut v = Dekker::from_exact_mult(LN2H, x);
v.lo = f_fmla(x, LN2L, v.lo);
let mut p = exp1(v);
let zf: Dekker = if x >= 0. {
Dekker::from_exact_add(p.hi, -1.0)
} else {
Dekker::from_exact_add(-1.0, p.hi)
};
p.lo += zf.lo;
p.hi = zf.hi;
Exp2m1 {
exp: p,
err: f64::from_bits(0x3b84200000000000) * p.hi,
}
}
#[inline]
fn q_2(dz: Dekker) -> Dekker {
const Q_2: [u64; 9] = [
0x3ff0000000000000,
0x3ff0000000000000,
0x3fe0000000000000,
0x3fc5555555555555,
0x3c655555555c4d26,
0x3fa5555555555555,
0x3f81111111111111,
0x3f56c16c3fbb4213,
0x3f2a01a023ede0d7,
];
let z = dz.to_f64();
let mut q = f_fmla(f64::from_bits(Q_2[8]), dz.hi, f64::from_bits(Q_2[7]));
q = f_fmla(q, z, f64::from_bits(Q_2[6]));
q = f_fmla(q, z, f64::from_bits(Q_2[5]));
let mut p = Dekker::from_exact_mult(q, z);
let r0 = Dekker::from_exact_add(f64::from_bits(Q_2[3]), p.hi);
p.hi = r0.hi;
p.lo += r0.lo + f64::from_bits(Q_2[4]);
p = Dekker::quick_mult(p, dz);
let r1 = Dekker::from_exact_add(f64::from_bits(Q_2[2]), p.hi);
p.hi = r1.hi;
p.lo += r1.lo;
p = Dekker::quick_mult(p, dz);
let r1 = Dekker::from_exact_add(f64::from_bits(Q_2[1]), p.hi);
p.hi = r1.hi;
p.lo += r1.lo;
p = Dekker::quick_mult(p, dz);
let r1 = Dekker::from_exact_add(f64::from_bits(Q_2[0]), p.hi);
p.hi = r1.hi;
p.lo += r1.lo;
p
}
#[inline]
fn exp_2(x: f64) -> Dekker {
let k = (x * f64::from_bits(0x40b0000000000000)).round_ties_even();
let yhh = f_fmla(-k, f64::from_bits(0x3f30000000000000), x);
let ky = Dekker::f64_mult(yhh, Dekker::new(LN2L, LN2H));
let ik = k as i64;
let im = (ik >> 12).wrapping_add(0x3ff);
let i2 = (ik >> 6) & 0x3f;
let i1 = ik & 0x3f;
let t1 = Dekker::from_bit_pair(EXP_M1_2_TABLE1[i2 as usize]);
let t2 = Dekker::from_bit_pair(EXP_M1_2_TABLE2[i1 as usize]);
let p = Dekker::quick_mult(t2, t1);
let mut q = q_2(ky);
q = Dekker::quick_mult(p, q);
let mut ud: u64 = (im as u64).wrapping_shl(52);
if im == 0x7ff {
q.hi *= 2.0;
q.lo *= 2.0;
ud = (im.wrapping_sub(1) as u64).wrapping_shl(52);
}
q.hi *= f64::from_bits(ud);
q.lo *= f64::from_bits(ud);
q
}
#[inline]
fn exp2m1_accurate_tiny(x: f64) -> f64 {
let x2 = x * x;
let x4 = x2 * x2;
const Q: [u64; 22] = [
0x3fe62e42fefa39ef,
0x3c7abc9e3b398040,
0x3fcebfbdff82c58f,
0xbc65e43a53e44dcf,
0x3fac6b08d704a0c0,
0xbc4d331627517168,
0x3f83b2ab6fba4e77,
0x3c14e65df0779f8c,
0x3f55d87fe78a6731,
0x3bd0717fbf4bd050,
0x3f2430912f86c787,
0x3bcbd2bdec9bcd42,
0x3eeffcbfc588b0c7,
0xbb8e60aa6d5e4aa9,
0x3eb62c0223a5c824,
0x3e7b5253d395e7d4,
0x3e3e4cf5158b9160,
0x3dfe8cac734c6058,
0x3dbc3bd64f17199d,
0x3d78161a17e05651,
0x3d33150b3d792231,
0x3cec184260bfad7e,
];
let mut c13 = f_fmla(f64::from_bits(Q[20]), x, f64::from_bits(Q[19])); let c11 = f_fmla(f64::from_bits(Q[18]), x, f64::from_bits(Q[17])); c13 = f_fmla(f64::from_bits(Q[21]), x2, c13); let mut p = Dekker::from_exact_add(
f64::from_bits(Q[15]),
f_fmla(f64::from_bits(Q[16]), x, f_fmla(c11, x2, c13 * x4)),
);
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[14]), p.hi);
p.lo += p0.lo;
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[12]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[13]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[10]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[11]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[8]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[9]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[6]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[7]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[4]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[5]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[2]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[3]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p0 = Dekker::from_exact_add(f64::from_bits(Q[0]), p.hi);
p.lo += p0.lo + f64::from_bits(Q[1]);
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
p.to_f64()
}
fn exp2m1_accurate(x: f64) -> f64 {
let t = x.to_bits();
let ux = t;
let ax = ux & 0x7fffffffffffffffu64;
if ax <= 0x3fc0000000000000u64 {
return exp2m1_accurate_tiny(x);
}
let mut p = exp_2(x);
let zf: Dekker = if x >= 0. {
Dekker::from_exact_add(p.hi, -1.0)
} else {
Dekker::from_exact_add(-1.0, p.hi)
};
p.lo += zf.lo;
p.hi = zf.hi;
p.to_f64()
}
#[inline]
fn exp2m1_fast_tiny(x: f64) -> Exp2m1 {
const P: [u64; 12] = [
0x3fe62e42fefa39ef,
0x3c7abd1697afcaf8,
0x3fcebfbdff82c58f,
0xbc65e5a1d09e1599,
0x3fac6b08d704a0bf,
0x3f83b2ab6fba4e78,
0x3f55d87fe78a84e6,
0x3f2430912f86a480,
0x3eeffcbfbc1f2b36,
0x3eb62c0226c7f6d1,
0x3e7b539529819e63,
0x3e3e4d552bed5b9c,
];
let x2 = x * x;
let x4 = x2 * x2;
let mut c8 = f_fmla(f64::from_bits(P[10]), x, f64::from_bits(P[9])); let c6 = f_fmla(f64::from_bits(P[8]), x, f64::from_bits(P[7])); let mut c4 = f_fmla(f64::from_bits(P[6]), x, f64::from_bits(P[5])); c8 = f_fmla(f64::from_bits(P[11]), x2, c8); c4 = f_fmla(c6, x2, c4); c4 = f_fmla(c8, x4, c4);
let mut p = Dekker::from_exact_mult(c4, x);
let p0 = Dekker::from_exact_add(f64::from_bits(P[4]), p.hi);
p.lo += p0.lo;
p.hi = p0.hi;
p = Dekker::f64_mult(x, p);
let p1 = Dekker::from_exact_add(f64::from_bits(P[2]), p.hi);
p.lo += p1.lo + f64::from_bits(P[3]);
p.hi = p1.hi;
p = Dekker::f64_mult(x, p);
let p2 = Dekker::from_exact_add(f64::from_bits(P[0]), p.hi);
p.lo += p2.lo + f64::from_bits(P[1]);
p.hi = p2.hi;
p = Dekker::f64_mult(x, p);
Exp2m1 {
exp: p,
err: f64::from_bits(0x3bd4e00000000000) * p.hi, }
}
#[inline]
pub fn f_exp2m1(d: f64) -> f64 {
let mut x = d;
let t = x.to_bits();
let ux = t;
let ax = ux & 0x7fffffffffffffffu64;
if ux >= 0xc04b000000000000u64 {
if (ux >> 52) == 0xfff {
return if ux > 0xfff0000000000000u64 {
x + x
} else {
-1.0
};
}
return -1.0 + f64::from_bits(0x3c90000000000000);
} else if ax >= 0x4090000000000000u64 {
if (ux >> 52) == 0x7ff {
return x + x;
}
return f_fmla(
x,
f64::from_bits(0x7bffffffffffffff),
f64::from_bits(0x7fefffffffffffff),
);
} else if ax <= 0x3cc0527dbd87e24du64
{
if ax <= 0x3970000000000000u64
{
if x == 0. {
return x;
}
x *= f64::from_bits(0x4690000000000000);
let mut z = Dekker::from_exact_mult(LN2H, x);
z.lo = f_fmla(LN2L, x, z.lo);
let mut h2 = z.to_f64(); h2 *= f64::from_bits(0x3950000000000000);
let mut h = f_fmla(-h2, f64::from_bits(0x4690000000000000), z.hi);
h += z.lo;
let res = f_fmla(h, f64::from_bits(0x3950000000000000), h2);
return res;
} else {
const C2: f64 = f64::from_bits(0x3fcebfbdff82c58f); let mut z = Dekker::from_exact_mult(LN2H, x);
z.lo = f_fmla(LN2L, x, z.lo);
z.lo += C2 * x * x;
return z.to_f64();
}
}
if ux.wrapping_shl(17) == 0 {
let i = x.floor() as i32;
if x == i as f64 && -53 <= i && i <= 53 {
return if i >= 0 {
((1u64 << i) - 1) as f64
} else {
-1.0 + ldexp(1.0, i as u64)
};
}
}
let result = exp2m1_fast(x, ax <= 0x3fc0000000000000u64);
let left = result.exp.hi + (result.exp.lo - result.err);
let right = result.exp.hi + (result.exp.lo + result.err);
if left != right {
return exp2m1_accurate(x);
}
left
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_exp2m1() {
assert_eq!(3., f_exp2m1(2.0));
assert_eq!(4.656854249492381, f_exp2m1(2.5));
assert_eq!(-0.30801352040368324, f_exp2m1(-0.5311842449009418));
}
}