
(FPCore (x) :precision binary64 (atanh x))
double code(double x) {
return atanh(x);
}
def code(x): return math.atanh(x)
function code(x) return atanh(x) end
function tmp = code(x) tmp = atanh(x); end
code[x_] := N[ArcTanh[x], $MachinePrecision]
\begin{array}{l}
\\
\tanh^{-1} x
\end{array}
Sampling outcomes in binary64 precision:
Herbie found 3 alternatives:
| Alternative | Accuracy | Speedup |
|---|
(FPCore (x) :precision binary64 (* 0.5 (log1p (/ (* 2.0 x) (- 1.0 x)))))
double code(double x) {
return 0.5 * log1p(((2.0 * x) / (1.0 - x)));
}
public static double code(double x) {
return 0.5 * Math.log1p(((2.0 * x) / (1.0 - x)));
}
def code(x): return 0.5 * math.log1p(((2.0 * x) / (1.0 - x)))
function code(x) return Float64(0.5 * log1p(Float64(Float64(2.0 * x) / Float64(1.0 - x)))) end
code[x_] := N[(0.5 * N[Log[1 + N[(N[(2.0 * x), $MachinePrecision] / N[(1.0 - x), $MachinePrecision]), $MachinePrecision]], $MachinePrecision]), $MachinePrecision]
\begin{array}{l}
\\
0.5 \cdot \mathsf{log1p}\left(\frac{2 \cdot x}{1 - x}\right)
\end{array}
(FPCore (x)
:precision binary64
(let* ((t_0 (/ x (fma -1.0 x 1.0))) (t_1 (fma (* 4.0 t_0) t_0 (* -2.0 t_0))))
(*
0.5
(-
(log1p (* 8.0 (pow t_0 3.0)))
(- (log1p (pow t_1 3.0)) (log1p (- (* t_1 t_1) t_1)))))))
double code(double x) {
double t_0 = x / fma(-1.0, x, 1.0);
double t_1 = fma((4.0 * t_0), t_0, (-2.0 * t_0));
return 0.5 * (log1p((8.0 * pow(t_0, 3.0))) - (log1p(pow(t_1, 3.0)) - log1p(((t_1 * t_1) - t_1))));
}
function code(x) t_0 = Float64(x / fma(-1.0, x, 1.0)) t_1 = fma(Float64(4.0 * t_0), t_0, Float64(-2.0 * t_0)) return Float64(0.5 * Float64(log1p(Float64(8.0 * (t_0 ^ 3.0))) - Float64(log1p((t_1 ^ 3.0)) - log1p(Float64(Float64(t_1 * t_1) - t_1))))) end
code[x_] := Block[{t$95$0 = N[(x / N[(-1.0 * x + 1.0), $MachinePrecision]), $MachinePrecision]}, Block[{t$95$1 = N[(N[(4.0 * t$95$0), $MachinePrecision] * t$95$0 + N[(-2.0 * t$95$0), $MachinePrecision]), $MachinePrecision]}, N[(0.5 * N[(N[Log[1 + N[(8.0 * N[Power[t$95$0, 3.0], $MachinePrecision]), $MachinePrecision]], $MachinePrecision] - N[(N[Log[1 + N[Power[t$95$1, 3.0], $MachinePrecision]], $MachinePrecision] - N[Log[1 + N[(N[(t$95$1 * t$95$1), $MachinePrecision] - t$95$1), $MachinePrecision]], $MachinePrecision]), $MachinePrecision]), $MachinePrecision]), $MachinePrecision]]]
\begin{array}{l}
\\
\begin{array}{l}
t_0 := \frac{x}{\mathsf{fma}\left(-1, x, 1\right)}\\
t_1 := \mathsf{fma}\left(4 \cdot t\_0, t\_0, -2 \cdot t\_0\right)\\
0.5 \cdot \left(\mathsf{log1p}\left(8 \cdot {t\_0}^{3}\right) - \left(\mathsf{log1p}\left({t\_1}^{3}\right) - \mathsf{log1p}\left(t\_1 \cdot t\_1 - t\_1\right)\right)\right)
\end{array}
\end{array}
Initial program 100.0%
lift-log1p.f64N/A
lift-*.f64N/A
lift--.f64N/A
lift-/.f64N/A
flip3-+N/A
log-divN/A
lower--.f64N/A
Applied rewrites99.9%
lift-log1p.f64N/A
lift--.f64N/A
lift-*.f64N/A
lift-*.f64N/A
lift-/.f64N/A
lift-fma.f64N/A
lift-/.f64N/A
lift-fma.f64N/A
lift-*.f64N/A
lift-/.f64N/A
lift-fma.f64N/A
Applied rewrites100.0%
Final simplification100.0%
(FPCore (x) :precision binary64 (* 0.5 (log1p (/ (* 2.0 x) (- 1.0 x)))))
double code(double x) {
return 0.5 * log1p(((2.0 * x) / (1.0 - x)));
}
public static double code(double x) {
return 0.5 * Math.log1p(((2.0 * x) / (1.0 - x)));
}
def code(x): return 0.5 * math.log1p(((2.0 * x) / (1.0 - x)))
function code(x) return Float64(0.5 * log1p(Float64(Float64(2.0 * x) / Float64(1.0 - x)))) end
code[x_] := N[(0.5 * N[Log[1 + N[(N[(2.0 * x), $MachinePrecision] / N[(1.0 - x), $MachinePrecision]), $MachinePrecision]], $MachinePrecision]), $MachinePrecision]
\begin{array}{l}
\\
0.5 \cdot \mathsf{log1p}\left(\frac{2 \cdot x}{1 - x}\right)
\end{array}
Initial program 100.0%
(FPCore (x)
:precision binary64
(*
0.5
(*
(fma
(fma (fma 0.2857142857142857 (* x x) 0.4) (* x x) 0.6666666666666666)
(* x x)
2.0)
x)))
double code(double x) {
return 0.5 * (fma(fma(fma(0.2857142857142857, (x * x), 0.4), (x * x), 0.6666666666666666), (x * x), 2.0) * x);
}
function code(x) return Float64(0.5 * Float64(fma(fma(fma(0.2857142857142857, Float64(x * x), 0.4), Float64(x * x), 0.6666666666666666), Float64(x * x), 2.0) * x)) end
code[x_] := N[(0.5 * N[(N[(N[(N[(0.2857142857142857 * N[(x * x), $MachinePrecision] + 0.4), $MachinePrecision] * N[(x * x), $MachinePrecision] + 0.6666666666666666), $MachinePrecision] * N[(x * x), $MachinePrecision] + 2.0), $MachinePrecision] * x), $MachinePrecision]), $MachinePrecision]
\begin{array}{l}
\\
0.5 \cdot \left(\mathsf{fma}\left(\mathsf{fma}\left(\mathsf{fma}\left(0.2857142857142857, x \cdot x, 0.4\right), x \cdot x, 0.6666666666666666\right), x \cdot x, 2\right) \cdot x\right)
\end{array}
Initial program 100.0%
Taylor expanded in x around 0
*-commutativeN/A
lower-*.f64N/A
Applied rewrites99.9%
herbie shell --seed 2025066
(FPCore (x)
:name "Rust f64::atanh"
:precision binary64
(* 0.5 (log1p (/ (* 2.0 x) (- 1.0 x)))))