shithub: MicroHs

Download patch

ref: 83c472492f864dce83e948d6968417bc3538db18
parent: 41034a4ed8653dc474627a0d668eaedff54bb251
author: konsumlamm <konsumlamm@gmail.com>
date: Fri Jan 10 20:23:56 EST 2025

Integer: Optimize `quotRemI`

--- a/lib/Data/Integer.hs
+++ b/lib/Data/Integer.hs
@@ -23,6 +23,7 @@
 import Data.Integer_Type
 import Data.Integral
 import Data.List
+import Data.Maybe_Type
 import Data.Num
 import Data.Ord
 import Data.Ratio_Type
@@ -211,16 +212,20 @@
 quotRemI :: Integer -> Integer -> (Integer, Integer)
 quotRemI _         (I _  [])  = error "Integer: division by 0" -- n / 0
 quotRemI (I _  [])          _ = (I Plus [], I Plus [])         -- 0 / n
-quotRemI (I sx xs) (I sy ys) | all (== (0 :: Word)) ys' =
+quotRemI (I sx xs) (I sy ys) | Just (y, n) <- msd ys =
   -- All but the MSD are 0.  Scale numerator accordingly and divide.
   -- Then add back (the ++) the remainder we scaled off.
-    case quotRemD xs' y of
-      (q, r) -> qrRes sx sy (q, rs ++ r)
-  where ys'       = init ys
-        y         = last ys
-        n         = length ys'
-        (rs, xs') = splitAt n xs  -- xs' is the scaled number
+  let (rs, xs') = splitAt n xs  -- xs' is the scaled number
+  in case quotRemD xs' y of
+    (q, r) -> qrRes sx sy (q, rs ++ r)
 quotRemI (I sx xs) (I sy ys)  = qrRes sx sy (quotRemB xs ys)
+
+msd :: [Digit] -> Maybe (Digit, Int)
+msd = go 0
+  where
+    go _ [] = Nothing
+    go n [d] = Just (d, n)
+    go n (d : ds) = if d == 0 then go (n + 1) ds else Nothing
 
 qrRes :: Sign -> Sign -> ([Digit], [Digit]) -> (Integer, Integer)
 qrRes sx sy (ds, rs) = (sI (mulSign sx sy) ds, sI sx rs)