module Codec.Beam
( encode, Metadata, export, insertModuleInfo
, Op, X(..), Y(..), F(..), Nil(..), Label(..), Literal(..), Lambda(..), Import(..)
, Register, IsRegister(toRegister), Source, IsSource(toSource)
, RegisterF, IsRegisterF(toRegisterF), SourceF, IsSourceF(toSourceF)
, importBif0, importBif1, importBif2, importBif3, importBif4
) where
import Data.Bits ((.|.), (.&.))
import Data.ByteString.Builder (Builder)
import Data.ByteString.Lazy (ByteString)
import Data.Int (Int64)
import Data.Monoid ((<>))
import Data.Text (Text)
import Data.Text.Encoding (encodeUtf8)
import Data.Word (Word8, Word32)
import qualified Codec.Compression.Zlib as Zlib
import qualified Data.Bits as Bits
import qualified Data.ByteString.Builder as Builder
import qualified Data.ByteString.Lazy as BS
import qualified Data.Set as Set
import Codec.Beam.Internal.Syntax
import Codec.Beam.Internal.Table (Table)
import qualified Codec.Beam.Internal.Table as Table
encode
:: (Foldable f1, Foldable f2)
=> Text
-> f1 Metadata
-> f2 Op
-> ByteString
encode name metadata =
let
withMetadata =
foldr _run (initialEnv name) metadata
in
toLazyByteString . foldl encodeOp withMetadata
newtype Metadata = Metadata { _run :: Env -> Env }
export :: Text -> Int -> Metadata
export name arity =
Metadata $ \env -> env { _exporting = Set.insert (name, arity) (_exporting env) }
insertModuleInfo :: Metadata
insertModuleInfo =
Metadata $ \env ->
let
withExports = env
{ _exporting =
_exporting env <> Set.fromList [("module_info", 0), ("module_info", 1)]
}
in
foldl encodeOp withExports
[ Op 1 [FromNewLabel (Label (-1))]
, Op 2 [FromNewFunction "module_info" 0, FromAtom "module_info", FromUntagged 0]
, Op 1 [FromNewLabel (Label (-2))]
, Op 64 [FromAtom (_moduleName env), FromX (X 0)]
, Op 78 [FromUntagged 1, FromImport (Import "erlang" "get_module_info" 1)]
, Op 19 []
, Op 1 [FromNewLabel (Label (-3))]
, Op 2 [FromNewFunction "module_info" 1, FromAtom "module_info", FromUntagged 1]
, Op 1 [FromNewLabel (Label (-4))]
, Op 64 [FromX (X 0), FromX (X 1)]
, Op 64 [FromAtom (_moduleName env), FromX (X 0)]
, Op 78 [FromUntagged 2, FromImport (Import "erlang" "get_module_info" 2)]
, Op 19 []
]
data Env =
Env
{ _moduleName :: Text
, _labelTable :: Table Int
, _atomTable :: Table Text
, _literalTable :: Table Literal
, _lambdaTable :: Table Lambda
, _importTable :: Table Import
, _exportTable :: Table (Text, Int, Int)
, _exportNextLabel :: Maybe (Text, Int)
, _exporting :: Set.Set (Text, Int)
, _functionCount :: Word32
, _maxOpCode :: Word8
, _code :: Builder
}
initialEnv :: Text -> Env
initialEnv name =
Env
{ _moduleName = name
, _labelTable = Table.singleton 0 0
, _atomTable = Table.singleton name 1
, _literalTable = Table.empty
, _lambdaTable = Table.empty
, _importTable = Table.empty
, _exportTable = Table.empty
, _exportNextLabel = Nothing
, _exporting = Set.empty
, _functionCount = 0
, _maxOpCode = 1
, _code = mempty
}
encodeOp :: Env -> Op -> Env
encodeOp env (Op opCode args) =
foldl encodeArgument (appendCode withMaxOp (Builder.word8 opCode)) args
where
withMaxOp =
env { _maxOpCode = max opCode (_maxOpCode env) }
encodeArgument :: Env -> Argument -> Env
encodeArgument env argument =
case argument of
FromUntagged value ->
appendTag (encodeTag 0 value) env
FromNewLabel (Label raw) ->
let
(value, newTable) =
Table.index raw (_labelTable env)
in
appendTag (encodeTag 0 value) $ env
{ _labelTable = newTable
, _exportNextLabel = Nothing
, _exportTable =
maybe id
(\(f, a) -> Table.ensure (f, a, value))
(_exportNextLabel env)
(_exportTable env)
}
FromImport import_ ->
let
(value, newTable) =
Table.index import_ (_importTable env)
in
appendTag (encodeTag 0 value) $ env
{ _importTable = newTable
, _atomTable =
Table.ensure (_import_module import_) $
Table.ensure (_import_function import_) $ _atomTable env
}
FromLambda lambda ->
let
(value, newTable) =
Table.index lambda (_lambdaTable env)
in
appendTag (encodeTag 0 value) $ env
{ _lambdaTable = newTable
, _atomTable = Table.ensure (_lambda_name lambda) $ _atomTable env
}
FromInt value ->
appendTag (encodeTag 1 value) env
FromNil Nil ->
appendTag (encodeTag 2 0) env
FromNewFunction name arity ->
appendTag (encodeTag 2 1) $ env
{ _functionCount = 1 + _functionCount env
, _exportNextLabel =
if Set.member (name, arity) (_exporting env) then
Just (name, arity)
else
Nothing
}
FromAtom name ->
let
(value, newTable) =
Table.index name (_atomTable env)
in
appendTag (encodeTag 2 value) $ env { _atomTable = newTable }
FromX (X value) ->
appendTag (encodeTag 3 value) env
FromY (Y value) ->
appendTag (encodeTag 4 value) env
FromLabel (Label raw) ->
let
(value, newTable) =
Table.index raw (_labelTable env)
in
appendTag (encodeTag 5 value) $ env { _labelTable = newTable }
FromF (F value) ->
appendTag (encodeExt 2 value) env
FromLiteral literal ->
let
(value, newTable) =
Table.index literal (_literalTable env)
in
appendTag (encodeExt 4 value) $ env { _literalTable = newTable }
FromList list ->
foldr (flip encodeArgument) (appendTag (encodeExt 1 (length list)) env) list
appendTag :: [Word8] -> Env -> Env
appendTag bytes env =
appendCode env $ foldMap Builder.word8 bytes
appendCode :: Env -> Builder -> Env
appendCode env bytes =
env { _code = _code env <> bytes }
toLazyByteString :: Env -> ByteString
toLazyByteString
( Env
_
labelTable
atomTable
literalTable
lambdaTable
importTable
exportTable
_
_
functions
maxOpCode
bytes
) =
"FOR1" <> pack32 (BS.length sections + 4) <> "BEAM" <> sections
where
sections =
"AtU8" <> alignSection (atoms atomTable)
<> "StrT" <> alignSection (strings)
<> "LitT" <> alignSection (literals literalTable)
<> "FunT" <> alignSection (lambdas lambdaTable atomTable labelTable)
<> "ImpT" <> alignSection (imports importTable atomTable)
<> "ExpT" <> alignSection (exports exportTable atomTable)
<> "Code" <> alignSection (code bytes labelTable functions maxOpCode)
atoms :: Table Text -> ByteString
atoms table =
pack32 (Table.size table) <> Table.encode (withSize pack8 . packAtom) table
strings :: ByteString
strings =
pack32 (0 :: Integer)
code :: Builder -> Table Int -> Word32 -> Word8 -> ByteString
code builder labelTable functionCount maxOpCode =
mconcat
[ pack32 (16 :: Integer)
, pack32 (0 :: Integer)
, pack32 maxOpCode
, pack32 (Table.size labelTable)
, pack32 functionCount
, Builder.toLazyByteString builder
, pack8 (3 :: Integer)
]
lambdas :: Table Lambda -> Table Text -> Table Int -> ByteString
lambdas lambdaTable atomTable labelTable =
pack32 (Table.size lambdaTable) <> Table.encode fromLambda lambdaTable
where
fromLambda lambda@(Lambda name arity (Label raw) free) =
mconcat
[ pack32 (forceIndex name atomTable)
, pack32 arity
, pack32 (forceIndex raw labelTable)
, pack32 (forceIndex lambda lambdaTable)
, pack32 free
, pack32 (0 :: Integer)
]
imports :: Table Import -> Table Text -> ByteString
imports importTable atomTable =
pack32 (Table.size importTable) <> Table.encode fromImport importTable
where
fromImport (Import module_ function arity) =
mconcat
[ pack32 (forceIndex module_ atomTable)
, pack32 (forceIndex function atomTable)
, pack32 arity
]
exports :: Table (Text, Int, Int) -> Table Text -> ByteString
exports exportTable atomTable =
pack32 (Table.size exportTable) <> Table.encode fromTuple exportTable
where
fromTuple (name, arity, label) =
pack32 (forceIndex name atomTable) <> pack32 arity <> pack32 label
literals :: Table Literal -> ByteString
literals table =
pack32 (BS.length terms) <> Zlib.compress terms
where
terms =
pack32 (Table.size table)
<> Table.encode (withSize pack32 . BS.cons 131 . encodeLiteral) table
encodeLiteral :: Literal -> ByteString
encodeLiteral lit =
case lit of
Atom value ->
encodeAtom value
Integer value ->
encodeInteger value
Float value ->
pack8 (70 :: Integer) <> packDouble value
Binary value ->
pack8 (109 :: Integer) <> withSize pack32 value
Tuple elements | length elements < 256 ->
mconcat
[ pack8 (104 :: Integer)
, pack8 (length elements)
, foldMap encodeLiteral elements
]
Tuple elements ->
mconcat
[ pack8 (105 :: Integer)
, pack32 (length elements)
, foldMap encodeLiteral elements
]
List elements ->
mconcat
[ pack8 (108 :: Integer)
, pack32 (length elements)
, foldMap encodeLiteral elements
, pack8 (106 :: Integer)
]
Map pairs ->
mconcat
[ pack8 (116 :: Integer)
, pack32 (length pairs)
, foldMap (\(x, y) -> encodeLiteral x <> encodeLiteral y) pairs
]
ExternalFun (Import module_ function arity) ->
mconcat
[ pack8 (113 :: Integer)
, encodeAtom module_
, encodeAtom function
, encodeInteger arity
]
encodeAtom :: Text -> ByteString
encodeAtom value =
pack8 (119 :: Integer) <> withSize pack8 (packAtom value)
encodeInteger :: Int -> ByteString
encodeInteger value
| value < 256 = pack8 (97 :: Integer) <> pack8 value
| otherwise = pack8 (98 :: Integer) <> pack32 value
encodeTag :: Word8 -> Int -> [Word8]
encodeTag tag n
| n < 0 = manyBytes tag (negative n [])
| n < 0x10 = oneByte tag n
| n < 0x800 = twoBytes tag n
| otherwise = manyBytes tag (positive n [])
encodeExt :: Int -> Int -> [Word8]
encodeExt tag n =
encodeTag 7 tag ++ encodeTag 0 n
oneByte :: Word8 -> Int -> [Word8]
oneByte tag n =
[ top4 .|. tag ]
where
top4 =
Bits.shiftL (fromIntegral n) 4
twoBytes :: Word8 -> Int -> [Word8]
twoBytes tag n =
[ top3 .|. 0x8 .|. tag, bottom8 ]
where
top3 =
fromIntegral $ Bits.shiftR n 3 .&. 0xE0
bottom8 =
fromIntegral n
manyBytes :: Word8 -> [Word8] -> [Word8]
manyBytes tag bytes =
if count <= 8 then
(packedCount .|. 0x18 .|. tag) : bytes
else
(0xF8 .|. tag) : encodeTag 0 (count - 9) ++ bytes
where
count =
length bytes
packedCount =
fromIntegral $ Bits.shiftL (count - 2) 5
negative :: Int -> [Word8] -> [Word8]
negative n bytes =
case ( n, bytes ) of
( -1, first : _ : _ ) | first > 0x7F ->
bytes
_ ->
withBottom8 negative n bytes
positive :: Int -> [Word8] -> [Word8]
positive n bytes =
case ( n, bytes ) of
( 0, first : _ ) | first < 0x80 ->
bytes
_ ->
withBottom8 positive n bytes
withBottom8 :: (Int -> [Word8] -> a) -> Int -> [Word8] -> a
withBottom8 f n bytes =
f (Bits.shiftR n 8) (fromIntegral n : bytes)
alignSection :: ByteString -> ByteString
alignSection bytes =
withSize pack32 bytes <> padding
where
padding =
case mod (BS.length bytes) 4 of
0 -> BS.empty
r -> BS.replicate (4 - r) 0
withSize :: (Int64 -> ByteString) -> ByteString -> ByteString
withSize f bytes =
f (BS.length bytes) <> bytes
packAtom :: Text -> ByteString
packAtom =
BS.fromStrict . encodeUtf8
pack8 :: Integral n => n -> ByteString
pack8 =
BS.singleton . fromIntegral
pack32 :: Integral n => n -> ByteString
pack32 =
Builder.toLazyByteString . Builder.word32BE . fromIntegral
packDouble :: Double -> ByteString
packDouble =
Builder.toLazyByteString . Builder.doubleBE
forceIndex :: Ord k => k -> Table k -> Int
forceIndex k =
fst . Table.index k