{-# LANGUAGE CPP #-}
module Machine
( WordM, WordLE, assertMultM, fromLE, toLE, wordBits, wordBytes
) where
#include "MachDeps.h"
#if defined(i386_HOST_ARCH) || defined(x86_64_HOST_ARCH) \
|| ((defined(arm_HOST_ARCH) || defined(aarch64_HOST_ARCH)) \
&& defined(__ARM_FEATURE_UNALIGNED)) \
|| defined(powerpc_HOST_ARCH) || defined(powerpc64_HOST_ARCH) \
|| defined(powerpc64le_HOST_ARCH)
#define MLKEM_ALLOW_UNALIGNED_OP 1
#if (defined(arm_HOST_ARCH) || defined(aarch64_HOST_ARCH)) \
&& !defined(WORDS_BIGENDIAN)
#define MLKEM_FORCE_LITTLE_ENDIAN_ARCH 1
#endif
#endif
import Control.Exception (assert)
#ifdef MLKEM_ALLOW_UNALIGNED_OP
import qualified Data.Memory.Endian as B
#endif
import Data.Bits
import Data.Word
#ifdef MLKEM_ALLOW_UNALIGNED_OP
#if WORD_SIZE_IN_BITS == 64
type WordM = Word64
#else
type WordM = Word32
#endif
type WordLE = B.LE WordM
fromLE :: WordLE -> WordM
#ifdef MLKEM_FORCE_LITTLE_ENDIAN_ARCH
fromLE = B.unLE
#else
fromLE :: WordLE -> WordM
fromLE = WordLE -> WordM
forall a. ByteSwap a => LE a -> a
B.fromLE
#endif
toLE :: WordM -> WordLE
#ifdef MLKEM_FORCE_LITTLE_ENDIAN_ARCH
toLE = B.LE
#else
toLE :: WordM -> WordLE
toLE = WordM -> WordLE
forall a. ByteSwap a => a -> LE a
B.toLE
#endif
#else
type WordM = Word8
type WordLE = WordM
fromLE :: WordLE -> WordM
fromLE = id
toLE :: WordM -> WordLE
toLE = id
#endif
wordBits :: Int
wordBits :: Int
wordBits = WordM -> Int
forall b. FiniteBits b => b -> Int
finiteBitSize (WordM
0 :: WordM)
wordBytes :: Int
wordBytes :: Int
wordBytes = Int -> Int -> Int
forall a. Integral a => a -> a -> a
div Int
wordBits Int
8
assertMultM :: Int -> a -> a
assertMultM :: forall a. Int -> a -> a
assertMultM Int
n = Bool -> a -> a
forall a. (?callStack::CallStack) => Bool -> a -> a
assert (Int
n Int -> Int -> Int
forall a. Bits a => a -> a -> a
.&. Int
mask Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
0)
where mask :: Int
mask = Int
wordBytes Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1