shithub: MicroHs

Download patch

ref: b9daf68bd8c951e69d2a317c1440594793686133
parent: 62044b52839f697afc20b623eb296488c02eaead
author: konsumlamm <konsumlamm@gmail.com>
date: Wed Jan 8 08:12:22 EST 2025

Integer: Use bit operations

--- a/lib/Data/Integer.hs
+++ b/lib/Data/Integer.hs
@@ -14,6 +14,7 @@
 import Prelude()              -- do not import Prelude
 import Primitives
 import Control.Error
+import Data.Bits
 import Data.Bool
 import Data.Char
 import Data.Enum
@@ -148,7 +149,7 @@
 
 -- Add 3 digits with carry
 addD :: Digit -> Digit -> Digit -> (Digit, Digit)
-addD x y z = (quot s maxD, rem s maxD)  where s = x + y + z
+addD x y z = (quotMaxD s, remMaxD s)  where s = x + y + z
 
 -- Invariant: xs >= ys, so result is always >= 0
 sub :: [Digit] -> [Digit] -> [Digit]
@@ -164,7 +165,7 @@
 subW :: Digit -> Digit -> Digit -> (Digit, Digit)
 subW b x y =
   let d = maxD + x - y - b
-  in (1 - quot d maxD, rem d maxD)
+  in (1 - quotMaxD d, remMaxD d)
 
 -- Remove trailing 0s
 trim0 :: [Digit] -> [Digit]
@@ -196,8 +197,8 @@
 mulD ci (x:xs) y = r : mulD q xs y
   where
     xy = x * y + ci
-    q = quot xy maxD
-    r = rem  xy maxD
+    q = quotMaxD xy
+    r = remMaxD xy
 
 mulM :: [Digit] -> [Digit] -> [Digit]
 mulM xs ys =
@@ -240,7 +241,7 @@
     qr ci []     res = (res, [ci])
     qr ci (x:xs) res = qr r xs (q:res)
       where
-        cx = ci * maxD + x
+        cx = ci `shiftL` shiftD + x
         q = quot cx y
         r = rem cx y
 
--- a/lib/Data/Integer_Type.hs
+++ b/lib/Data/Integer_Type.hs
@@ -22,6 +22,21 @@
   else
     error "Integer: unsupported word size"
 
+shiftD :: Int
+shiftD =
+  if _wordSize `primIntEQ` 64 then
+    (32::Int)
+  else if _wordSize `primIntEQ` 32 then
+    (16::Int)
+  else
+    error "Integer: unsupported word size"
+
+quotMaxD :: Digit -> Digit
+quotMaxD d = d `primWordShr` shiftD
+
+remMaxD :: Digit -> Digit
+remMaxD d = d `primWordAnd` (maxD `primWordSub` 1)
+
 -- Sadly, we also need a bunch of functions.
 
 _intToInteger :: Int -> Integer