pow.rs 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. use core::ops::Mul;
  2. use core::num::Wrapping;
  3. use {One, CheckedMul};
  4. /// Binary operator for raising a value to a power.
  5. pub trait Pow<RHS> {
  6. /// The result after applying the operator.
  7. type Output;
  8. /// Returns `self` to the power `rhs`.
  9. ///
  10. /// # Examples
  11. ///
  12. /// ```
  13. /// use num_traits::Pow;
  14. /// assert_eq!(Pow::pow(10u32, 2u32), 100);
  15. /// ```
  16. fn pow(self, rhs: RHS) -> Self::Output;
  17. }
  18. macro_rules! pow_impl {
  19. ($t:ty) => {
  20. pow_impl!($t, u8);
  21. pow_impl!($t, usize);
  22. // FIXME: these should be possible
  23. // pow_impl!($t, u16);
  24. // pow_impl!($t, u32);
  25. // pow_impl!($t, u64);
  26. };
  27. ($t:ty, $rhs:ty) => {
  28. pow_impl!($t, $rhs, usize, pow);
  29. };
  30. ($t:ty, $rhs:ty, $desired_rhs:ty, $method:expr) => {
  31. impl Pow<$rhs> for $t {
  32. type Output = $t;
  33. #[inline]
  34. fn pow(self, rhs: $rhs) -> $t {
  35. ($method)(self, <$desired_rhs>::from(rhs))
  36. }
  37. }
  38. impl<'a> Pow<&'a $rhs> for $t {
  39. type Output = $t;
  40. #[inline]
  41. fn pow(self, rhs: &'a $rhs) -> $t {
  42. ($method)(self, <$desired_rhs>::from(*rhs))
  43. }
  44. }
  45. impl<'a> Pow<$rhs> for &'a $t {
  46. type Output = $t;
  47. #[inline]
  48. fn pow(self, rhs: $rhs) -> $t {
  49. ($method)(*self, <$desired_rhs>::from(rhs))
  50. }
  51. }
  52. impl<'a, 'b> Pow<&'a $rhs> for &'b $t {
  53. type Output = $t;
  54. #[inline]
  55. fn pow(self, rhs: &'a $rhs) -> $t {
  56. ($method)(*self, <$desired_rhs>::from(*rhs))
  57. }
  58. }
  59. };
  60. }
  61. pow_impl!(u8, u8, u32, u8::pow);
  62. pow_impl!(u8, u16, u32, u8::pow);
  63. pow_impl!(u8, u32, u32, u8::pow);
  64. pow_impl!(u8, usize);
  65. pow_impl!(i8, u8, u32, i8::pow);
  66. pow_impl!(i8, u16, u32, i8::pow);
  67. pow_impl!(i8, u32, u32, i8::pow);
  68. pow_impl!(i8, usize);
  69. pow_impl!(u16, u8, u32, u16::pow);
  70. pow_impl!(u16, u16, u32, u16::pow);
  71. pow_impl!(u16, u32, u32, u16::pow);
  72. pow_impl!(u16, usize);
  73. pow_impl!(i16, u8, u32, i16::pow);
  74. pow_impl!(i16, u16, u32, i16::pow);
  75. pow_impl!(i16, u32, u32, i16::pow);
  76. pow_impl!(i16, usize);
  77. pow_impl!(u32, u8, u32, u32::pow);
  78. pow_impl!(u32, u16, u32, u32::pow);
  79. pow_impl!(u32, u32, u32, u32::pow);
  80. pow_impl!(u32, usize);
  81. pow_impl!(i32, u8, u32, i32::pow);
  82. pow_impl!(i32, u16, u32, i32::pow);
  83. pow_impl!(i32, u32, u32, i32::pow);
  84. pow_impl!(i32, usize);
  85. pow_impl!(u64, u8, u32, u64::pow);
  86. pow_impl!(u64, u16, u32, u64::pow);
  87. pow_impl!(u64, u32, u32, u64::pow);
  88. pow_impl!(u64, usize);
  89. pow_impl!(i64, u8, u32, i64::pow);
  90. pow_impl!(i64, u16, u32, i64::pow);
  91. pow_impl!(i64, u32, u32, i64::pow);
  92. pow_impl!(i64, usize);
  93. pow_impl!(usize, u8, u32, usize::pow);
  94. pow_impl!(usize, u16, u32, usize::pow);
  95. pow_impl!(usize, u32, u32, usize::pow);
  96. pow_impl!(usize, usize);
  97. pow_impl!(isize, u8, u32, isize::pow);
  98. pow_impl!(isize, u16, u32, isize::pow);
  99. pow_impl!(isize, u32, u32, isize::pow);
  100. pow_impl!(isize, usize);
  101. pow_impl!(Wrapping<u8>);
  102. pow_impl!(Wrapping<i8>);
  103. pow_impl!(Wrapping<u16>);
  104. pow_impl!(Wrapping<i16>);
  105. pow_impl!(Wrapping<u32>);
  106. pow_impl!(Wrapping<i32>);
  107. pow_impl!(Wrapping<u64>);
  108. pow_impl!(Wrapping<i64>);
  109. pow_impl!(Wrapping<usize>);
  110. pow_impl!(Wrapping<isize>);
  111. // FIXME: these should be possible
  112. // pow_impl!(u8, u64);
  113. // pow_impl!(i16, u64);
  114. // pow_impl!(i8, u64);
  115. // pow_impl!(u16, u64);
  116. // pow_impl!(u32, u64);
  117. // pow_impl!(i32, u64);
  118. // pow_impl!(u64, u64);
  119. // pow_impl!(i64, u64);
  120. // pow_impl!(usize, u64);
  121. // pow_impl!(isize, u64);
  122. #[cfg(feature = "std")]
  123. mod float_impls {
  124. use super::Pow;
  125. pow_impl!(f32, i8, i32, f32::powi);
  126. pow_impl!(f32, u8, i32, f32::powi);
  127. pow_impl!(f32, i16, i32, f32::powi);
  128. pow_impl!(f32, u16, i32, f32::powi);
  129. pow_impl!(f32, i32, i32, f32::powi);
  130. pow_impl!(f64, i8, i32, f64::powi);
  131. pow_impl!(f64, u8, i32, f64::powi);
  132. pow_impl!(f64, i16, i32, f64::powi);
  133. pow_impl!(f64, u16, i32, f64::powi);
  134. pow_impl!(f64, i32, i32, f64::powi);
  135. pow_impl!(f32, f32, f32, f32::powf);
  136. pow_impl!(f64, f32, f64, f64::powf);
  137. pow_impl!(f64, f64, f64, f64::powf);
  138. }
  139. /// Raises a value to the power of exp, using exponentiation by squaring.
  140. ///
  141. /// # Example
  142. ///
  143. /// ```rust
  144. /// use num_traits::pow;
  145. ///
  146. /// assert_eq!(pow(2i8, 4), 16);
  147. /// assert_eq!(pow(6u8, 3), 216);
  148. /// ```
  149. #[inline]
  150. pub fn pow<T: Clone + One + Mul<T, Output = T>>(mut base: T, mut exp: usize) -> T {
  151. if exp == 0 { return T::one() }
  152. while exp & 1 == 0 {
  153. base = base.clone() * base;
  154. exp >>= 1;
  155. }
  156. if exp == 1 { return base }
  157. let mut acc = base.clone();
  158. while exp > 1 {
  159. exp >>= 1;
  160. base = base.clone() * base;
  161. if exp & 1 == 1 {
  162. acc = acc * base.clone();
  163. }
  164. }
  165. acc
  166. }
  167. /// Raises a value to the power of exp, returning `None` if an overflow occurred.
  168. ///
  169. /// Otherwise same as the `pow` function.
  170. ///
  171. /// # Example
  172. ///
  173. /// ```rust
  174. /// use num_traits::checked_pow;
  175. ///
  176. /// assert_eq!(checked_pow(2i8, 4), Some(16));
  177. /// assert_eq!(checked_pow(7i8, 8), None);
  178. /// assert_eq!(checked_pow(7u32, 8), Some(5_764_801));
  179. /// ```
  180. #[inline]
  181. pub fn checked_pow<T: Clone + One + CheckedMul>(mut base: T, mut exp: usize) -> Option<T> {
  182. if exp == 0 { return Some(T::one()) }
  183. macro_rules! optry {
  184. ( $ expr : expr ) => {
  185. if let Some(val) = $expr { val } else { return None }
  186. }
  187. }
  188. while exp & 1 == 0 {
  189. base = optry!(base.checked_mul(&base));
  190. exp >>= 1;
  191. }
  192. if exp == 1 { return Some(base) }
  193. let mut acc = base.clone();
  194. while exp > 1 {
  195. exp >>= 1;
  196. base = optry!(base.checked_mul(&base));
  197. if exp & 1 == 1 {
  198. acc = optry!(acc.checked_mul(&base));
  199. }
  200. }
  201. Some(acc)
  202. }