summaryrefslogtreecommitdiff
path: root/testsuite/tests/numeric/should_run/CarryOverflow.hs
blob: d188ca9e38678a103650742afef17c1f9ce1e352 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
{-# LANGUAGE MagicHash, UnboxedTuples #-}

import GHC.Word
import GHC.Exts

import Control.Monad
import Data.Bits
import Data.List (sort)
import qualified Data.List.NonEmpty as NE
import System.Exit

allEqual :: Eq a => [a] -> Bool
allEqual [] = error "allEqual: nothing to compare"
allEqual (x:xs) = all (== x) xs

testWords :: [Word]
testWords = map NE.head . NE.group . sort $
            concatMap (\w -> [w - 1, w, w + 1]) $
            concatMap (\w -> [w, maxBound - w]) $
            trailingOnes ++ randoms
  where trailingOnes = takeWhile (/= 0) $ iterate (`div` 2) $ maxBound
        -- What would a Haskell program be without some Fibonacci numbers?
        randoms = take 40 $ drop 100 fibs
        fibs = 0 : 1 : zipWith (+) fibs (tail fibs)


wordSizeInBits :: Int
wordSizeInBits = length $ takeWhile (/= 0) $ iterate (`div` 2) (maxBound :: Word)


-- plusWord2# (Word# carry)

ways_plusWord2# :: [Word -> Word -> Bool]
ways_plusWord2# = [ltTest, integerTest, primopTest]
  where ltTest x y =
          let r = x + y in r < x
        integerTest x y =
          let r = fromIntegral x + fromIntegral y :: Integer
          in r > fromIntegral (maxBound :: Word)
        primopTest (W# x) (W# y) = case plusWord2# x y of
          (# 0##, _ #) -> False
          (# 1##, _ #) -> True
          _            -> error "unexpected result from plusWord2#"

-- addIntC# (Int# addition overflow)

ways_addIntC# :: [Int -> Int -> Bool]
ways_addIntC# = [ltTest, integerTest, highBitTest, primopTest]
  where ltTest x y =
          let r = x + y in (y > 0 && r < x) || (y < 0 && r > x)
        integerTest x y =
          let r = fromIntegral x + fromIntegral y :: Integer
          in r < fromIntegral (minBound :: Int) || r > fromIntegral (maxBound :: Int)
        highBitTest x y =
          let r = x + y in testBit ((x `xor` r) .&. (y `xor` r)) (wordSizeInBits - 1)
        primopTest (I# x) (I# y) = case addIntC# x y of
          (# _, 0# #) -> False
          _ -> True

-- subIntC# (Int# subtraction overflow)

ways_subIntC# :: [Int -> Int -> Bool]
ways_subIntC# = [ltTest, integerTest, highBitTest, primopTest]
  where ltTest x y =
          let r = x - y in (y > 0 && r > x) || (y < 0 && r < x)
        integerTest x y =
          let r = fromIntegral x - fromIntegral y :: Integer
          in r < fromIntegral (minBound :: Int) || r > fromIntegral (maxBound :: Int)
        highBitTest x y =
          let r = x - y in testBit ((x `xor` r) .&. complement (y `xor` r)) (wordSizeInBits - 1)
        primopTest (I# x) (I# y) = case subIntC# x y of
          (# _, 0# #) -> False
          _ -> True

runTest :: Show a => String -> [a -> a -> Bool] -> a -> a -> IO ()
runTest label ways x y = do
  let results = map (\f -> f x y) ways
  unless (allEqual results) $ do
    putStrLn $ "Failed (" ++ label ++ "): " ++ show (x,y) ++ " " ++ show results
    exitWith (ExitFailure 1)

main :: IO ()
main = do
  forM_ testWords $ \x ->
    forM_ testWords $ \y -> do
      runTest "ways_plusWord2#" ways_plusWord2# x y
      runTest "ways_addIntC#" ways_addIntC# (fromIntegral x) (fromIntegral y)
      runTest "ways_subIntC#" ways_subIntC# (fromIntegral x) (fromIntegral y)
  putStrLn "Passed"