{-# OPTIONS --no-auto-inline #-}
module Haskell.Prim.Int where
open import Haskell.Prim
open import Haskell.Prim.Word
open import Haskell.Prim.Integer
open import Haskell.Prim.Bool
data Int : Set where
int64 : Word64 → Int
intToWord : Int → Word64
intToWord (int64 a) = a
unsafeIntToNat : Int → Nat
unsafeIntToNat a = w2n (intToWord a)
private
2⁶⁴ : Nat
2⁶⁴ = 18446744073709551616
2⁶³ : Nat
2⁶³ = 9223372036854775808
maxInt : Nat
maxInt = monusNat 2⁶³ 1
instance
iNumberInt : Number Int
iNumberInt .Number.Constraint n = IsTrue (ltNat n 2⁶³)
iNumberInt .fromNat n = int64 (n2w n)
iNegativeInt : Negative Int
iNegativeInt .Negative.Constraint n = IsTrue (ltNat n (addNat 1 2⁶³))
iNegativeInt .fromNeg n = int64 (n2w (monusNat 2⁶⁴ n))
isNegativeInt : Int → Bool
isNegativeInt (int64 w) = ltNat maxInt (w2n w)
eqInt : Int → Int → Bool
eqInt (int64 a) (int64 b) = eqNat (w2n a) (w2n b)
negateInt : Int → Int
negateInt (int64 a) = int64 (n2w (monusNat 2⁶⁴ (w2n a)))
intToInteger : Int → Integer
intToInteger a = if isNegativeInt a then negsuc (monusNat (unsafeIntToNat (negateInt a)) 1)
else pos (unsafeIntToNat a)
integerToInt : Integer → Int
integerToInt (pos n) = int64 (n2w n)
integerToInt (negsuc n) = negateInt (int64 (n2w (suc n)))
private
ltPosInt : Int → Int → Bool
ltPosInt (int64 a) (int64 b) = ltWord a b
ltInt : Int → Int → Bool
ltInt a b with isNegativeInt a | isNegativeInt b
... | True | False = True
... | False | True = False
... | True | True = ltPosInt (negateInt b) (negateInt a)
... | False | False = ltPosInt a b
addInt : Int → Int → Int
addInt (int64 a) (int64 b) = int64 (addWord a b)
subInt : Int → Int → Int
subInt a b = addInt a (negateInt b)
mulInt : Int → Int → Int
mulInt (int64 a) (int64 b) = int64 (mulWord a b)
absInt : Int → Int
absInt a = if isNegativeInt a then negateInt a else a
signInt : Int → Int
signInt a = if isNegativeInt a then -1
else if eqInt a 0 then 0 else 1
showInt : Int → List Char
showInt a = showInteger (intToInteger a)
@0 IsNonNegativeInt : Int → Set
IsNonNegativeInt a@(int64 _) =
if isNegativeInt a then TypeError (primStringAppend (primStringFromList (showInt a)) " is negative")
else ⊤
intToNat : (a : Int) → @0 ⦃ IsNonNegativeInt a ⦄ → Nat
intToNat a = unsafeIntToNat a