| // Copyright 2009 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package math |
| |
| |
| // Pow returns x**y, the base-x exponential of y. |
| func Pow(x, y float64) float64 { |
| // TODO: x or y NaN, ±Inf, maybe ±0. |
| switch { |
| case y == 0: |
| return 1 |
| case y == 1: |
| return x |
| case x == 0 && y > 0: |
| return 0 |
| case x == 0 && y < 0: |
| return Inf(1) |
| case y == 0.5: |
| return Sqrt(x) |
| case y == -0.5: |
| return 1 / Sqrt(x) |
| } |
| |
| absy := y; |
| flip := false; |
| if absy < 0 { |
| absy = -absy; |
| flip = true; |
| } |
| yi, yf := Modf(absy); |
| if yf != 0 && x < 0 { |
| return NaN() |
| } |
| if yi >= 1<<63 { |
| return Exp(y * Log(x)) |
| } |
| |
| // ans = a1 * 2^ae (= 1 for now). |
| a1 := float64(1); |
| ae := 0; |
| |
| // ans *= x^yf |
| if yf != 0 { |
| if yf > 0.5 { |
| yf--; |
| yi++; |
| } |
| a1 = Exp(yf * Log(x)); |
| } |
| |
| // ans *= x^yi |
| // by multiplying in successive squarings |
| // of x according to bits of yi. |
| // accumulate powers of two into exp. |
| x1, xe := Frexp(x); |
| for i := int64(yi); i != 0; i >>= 1 { |
| if i&1 == 1 { |
| a1 *= x1; |
| ae += xe; |
| } |
| x1 *= x1; |
| xe <<= 1; |
| if x1 < .5 { |
| x1 += x1; |
| xe--; |
| } |
| } |
| |
| // ans = a1*2^ae |
| // if flip { ans = 1 / ans } |
| // but in the opposite order |
| if flip { |
| a1 = 1 / a1; |
| ae = -ae; |
| } |
| return Ldexp(a1, ae); |
| } |