Skip to content

Commit

Permalink
Fix #1 Enforce that the Word64 values used are little-endian.
Browse files Browse the repository at this point in the history
  • Loading branch information
mpilgrem committed Oct 20, 2024
1 parent 864fd7b commit e435b62
Showing 1 changed file with 25 additions and 9 deletions.
34 changes: 25 additions & 9 deletions src/Data/StaticBytes.hs
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,8 @@ module Data.StaticBytes
, fromStatic
) where

import Data.Bits
import Data.ByteArray
import Data.Bits ( Bits (..), shiftL )
import Data.ByteArray ( ByteArrayAccess (..) )
import qualified Data.ByteString as B
import qualified Data.ByteString.Internal as B
import qualified Data.Primitive.ByteArray as BA
Expand All @@ -35,9 +35,10 @@ import qualified Data.Vector.Primitive as VP
import qualified Data.Vector.Storable as VS
import qualified Data.Vector.Unboxed as VU
import qualified Data.Vector.Unboxed.Base as VU
import Foreign.ForeignPtr
import Foreign.Ptr
import Foreign.Storable
import Foreign.ForeignPtr ( ForeignPtr, withForeignPtr )
import Foreign.Ptr ( Ptr, castPtr )
import Foreign.Storable ( Storable (..) )
import GHC.ByteOrder ( ByteOrder (..), targetByteOrder )
import RIO hiding ( words )
import System.IO.Unsafe ( unsafePerformIO )

Expand Down Expand Up @@ -79,27 +80,31 @@ class DynamicBytes dbytes where
lengthD :: dbytes -> Int
-- Yeah, it looks terrible to use a list here, but fusion should kick in
withPeekD :: dbytes -> ((Int -> IO Word64) -> IO a) -> IO a
-- ^ This assumes that the Word64 values are all little-endian.
-- | May throw a runtime exception if invariants are violated!
fromWordsD :: Int -> [Word64] -> dbytes
-- ^ This assumes that the Word64 values are all little-endian.

fromWordsForeign ::
(ForeignPtr a -> Int -> b)
-> Int
-> [Word64]
-- ^ The Word64 values are assumed to be little-endian.
-> b
fromWordsForeign wrapper len words0 = unsafePerformIO $ do
fptr <- B.mallocByteString len
withForeignPtr fptr $ \ptr -> do
let loop _ [] = pure ()
loop off (w:ws) = do
pokeElemOff (castPtr ptr) off w
pokeElemOff (castPtr ptr) off (fromLE64 w)
loop (off + 1) ws
loop 0 words0
pure $ wrapper fptr len

withPeekForeign ::
(ForeignPtr a, Int, Int)
-> ((Int -> IO Word64) -> IO b)
-- ^ The Word64 values are assumed to be little-endian.
-> IO b
withPeekForeign (fptr, off, len) inner =
withForeignPtr fptr $ \ptr -> do
Expand All @@ -113,7 +118,7 @@ withPeekForeign (fptr, off, len) inner =
let w64' = shiftL (fromIntegral w8) (i * 8) .|. w64
loop w64' (i + 1)
loop 0 0
| otherwise = peekByteOff ptr (off + off')
| otherwise = toLE64 <$> peekByteOff ptr (off + off')
inner f

instance DynamicBytes B.ByteString where
Expand All @@ -133,7 +138,7 @@ instance word8 ~ Word8 => DynamicBytes (VP.Vector word8) where
let loop _ [] =
VP.Vector 0 len <$> BA.unsafeFreezeByteArray ba
loop i (w:ws) = do
BA.writeByteArray ba i w
BA.writeByteArray ba i (fromLE64 w)
loop (i + 1) ws
loop 0 words0
withPeekD (VP.Vector off len ba) inner = do
Expand All @@ -147,7 +152,8 @@ instance word8 ~ Word8 => DynamicBytes (VP.Vector word8) where
let w64' = shiftL (fromIntegral w8) (i * 8) .|. w64
loop w64' (i + 1)
loop 0 0
| otherwise = pure $ BA.indexByteArray ba (off + (off' `div` 8))
| otherwise = pure $
toLE64 $ BA.indexByteArray ba (off + (off' `div` 8))
inner f

instance word8 ~ Word8 => DynamicBytes (VU.Vector word8) where
Expand Down Expand Up @@ -248,3 +254,13 @@ fromStatic ::
=> sbytes
-> dbytes
fromStatic = fromWordsD (lengthS (Nothing :: Maybe sbytes)) . ($ []) . toWordsS

-- | Convert a 64 bit value in CPU endianess to little endian.
toLE64 :: Word64 -> Word64
toLE64 = case targetByteOrder of
BigEndian -> byteSwap64
LittleEndian -> id

-- | Convert a little endian 64 bit value to CPU endianess.
fromLE64 :: Word64 -> Word64
fromLE64 = toLE64

0 comments on commit e435b62

Please sign in to comment.