blob: 9d395f4a788d11117da33beda2a33c8ff4146828 [file] [log] [blame]
// Copyright ©2018 The Gonum Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Derived from code by Jeffrey A. Fike at http://adl.stanford.edu/hyperdual/
// The MIT License (MIT)
//
// Copyright (c) 2006 Jeffrey A. Fike
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
package dual
import "math"
// PowReal returns x**p, the base-x exponential of p.
//
// Special cases are (in order):
// PowReal(NaN+xϵ, ±0) = 1+NaNϵ for any x
// PowReal(x, ±0) = 1 for any x
// PowReal(1+xϵ, y) = 1+xyϵ for any y
// PowReal(x, 1) = x for any x
// PowReal(NaN+xϵ, y) = NaN+NaNϵ
// PowReal(x, NaN) = NaN+NaNϵ
// PowReal(±0, y) = ±Inf for y an odd integer < 0
// PowReal(±0, -Inf) = +Inf
// PowReal(±0, +Inf) = +0
// PowReal(±0, y) = +Inf for finite y < 0 and not an odd integer
// PowReal(±0, y) = ±0 for y an odd integer > 0
// PowReal(±0, y) = +0 for finite y > 0 and not an odd integer
// PowReal(-1, ±Inf) = 1
// PowReal(x+0ϵ, +Inf) = +Inf+NaNϵ for |x| > 1
// PowReal(x+yϵ, +Inf) = +Inf for |x| > 1
// PowReal(x, -Inf) = +0+NaNϵ for |x| > 1
// PowReal(x, +Inf) = +0+NaNϵ for |x| < 1
// PowReal(x+0ϵ, -Inf) = +Inf+NaNϵ for |x| < 1
// PowReal(x, -Inf) = +Inf-Infϵ for |x| < 1
// PowReal(+Inf, y) = +Inf for y > 0
// PowReal(+Inf, y) = +0 for y < 0
// PowReal(-Inf, y) = Pow(-0, -y)
// PowReal(x, y) = NaN+NaNϵ for finite x < 0 and finite non-integer y
func PowReal(d Number, p float64) Number {
const tol = 1e-15
r := d.Real
if math.Abs(r) < tol {
if r >= 0 {
r = tol
}
if r < 0 {
r = -tol
}
}
deriv := p * math.Pow(r, p-1)
return Number{
Real: math.Pow(d.Real, p),
Emag: d.Emag * deriv,
}
}
// Pow returns d**r, the base-d exponential of r.
func Pow(d, p Number) Number {
return Exp(Mul(p, Log(d)))
}
// Sqrt returns the square root of d.
//
// Special cases are:
// Sqrt(+Inf) = +Inf
// Sqrt(±0) = (±0+Infϵ)
// Sqrt(x < 0) = NaN
// Sqrt(NaN) = NaN
func Sqrt(d Number) Number {
if d.Real <= 0 {
if d.Real == 0 {
return Number{
Real: d.Real,
Emag: math.Inf(1),
}
}
return Number{
Real: math.NaN(),
Emag: math.NaN(),
}
}
return PowReal(d, 0.5)
}
// Exp returns e**q, the base-e exponential of d.
//
// Special cases are:
// Exp(+Inf) = +Inf
// Exp(NaN) = NaN
// Very large values overflow to 0 or +Inf.
// Very small values underflow to 1.
func Exp(d Number) Number {
fnDeriv := math.Exp(d.Real)
return Number{
Real: fnDeriv,
Emag: fnDeriv * d.Emag,
}
}
// Log returns the natural logarithm of d.
//
// Special cases are:
// Log(+Inf) = (+Inf+0ϵ)
// Log(0) = (-Inf±Infϵ)
// Log(x < 0) = NaN
// Log(NaN) = NaN
func Log(d Number) Number {
switch d.Real {
case 0:
return Number{
Real: math.Log(d.Real),
Emag: math.Copysign(math.Inf(1), d.Real),
}
case math.Inf(1):
return Number{
Real: math.Log(d.Real),
Emag: 0,
}
}
if d.Real < 0 {
return Number{
Real: math.NaN(),
Emag: math.NaN(),
}
}
return Number{
Real: math.Log(d.Real),
Emag: d.Emag / d.Real,
}
}
// Sin returns the sine of d.
//
// Special cases are:
// Sin(±0) = (±0+Nϵ)
// Sin(±Inf) = NaN
// Sin(NaN) = NaN
func Sin(d Number) Number {
if d.Real == 0 {
return Number{
Real: d.Real,
Emag: d.Emag,
}
}
fn := math.Sin(d.Real)
deriv := math.Cos(d.Real)
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}
// Cos returns the cosine of d.
//
// Special cases are:
// Cos(±Inf) = NaN
// Cos(NaN) = NaN
func Cos(d Number) Number {
fn := math.Cos(d.Real)
deriv := -math.Sin(d.Real)
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}
// Tan returns the tangent of d.
//
// Special cases are:
// Tan(±0) = (±0+Nϵ)
// Tan(±Inf) = NaN
// Tan(NaN) = NaN
func Tan(d Number) Number {
if d.Real == 0 {
return Number{
Real: d.Real,
Emag: d.Emag,
}
}
fn := math.Tan(d.Real)
deriv := 1 + fn*fn
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}
// Asin returns the inverse sine of d.
//
// Special cases are:
// Asin(±0) = (±0+Nϵ)
// Asin(±1) = (±Inf+Infϵ)
// Asin(x) = NaN if x < -1 or x > 1
func Asin(d Number) Number {
if d.Real == 0 {
return Number{
Real: d.Real,
Emag: d.Emag,
}
} else if m := math.Abs(d.Real); m >= 1 {
if m == 1 {
return Number{
Real: math.Asin(d.Real),
Emag: math.Inf(1),
}
}
return Number{
Real: math.NaN(),
Emag: math.NaN(),
}
}
fn := math.Asin(d.Real)
deriv := 1 / math.Sqrt(1-d.Real*d.Real)
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}
// Acos returns the inverse cosine of d.
//
// Special cases are:
// Acos(-1) = (Pi-Infϵ)
// Acos(1) = (0-Infϵ)
// Acos(x) = NaN if x < -1 or x > 1
func Acos(d Number) Number {
if m := math.Abs(d.Real); m >= 1 {
if m == 1 {
return Number{
Real: math.Acos(d.Real),
Emag: math.Inf(-1),
}
}
return Number{
Real: math.NaN(),
Emag: math.NaN(),
}
}
fn := math.Acos(d.Real)
deriv := -1 / math.Sqrt(1-d.Real*d.Real)
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}
// Atan returns the inverse tangent of d.
//
// Special cases are:
// Atan(±0) = (±0+Nϵ)
// Atan(±Inf) = (±Pi/2+0ϵ)
func Atan(d Number) Number {
if d.Real == 0 {
return Number{
Real: d.Real,
Emag: d.Emag,
}
}
fn := math.Atan(d.Real)
deriv := 1 / (1 + d.Real*d.Real)
return Number{
Real: fn,
Emag: deriv * d.Emag,
}
}