Abstract the string table into a newtype

This commit is contained in:
Eugen Wissner 2024-10-22 01:21:02 +02:00
parent bf5ec1f3e2
commit 57436f664e
Signed by: belka
GPG Key ID: A27FDC1E8EE902C0
6 changed files with 228 additions and 152 deletions

View File

@ -4,12 +4,10 @@ version: 0.1.0.0
synopsis: synopsis:
Elna programming language compiles simple mathematical operations to RISC-V code Elna programming language compiles simple mathematical operations to RISC-V code
-- description:
license: MPL-2.0 license: MPL-2.0
license-file: LICENSE license-file: LICENSE
author: Eugen Wissner author: Eugen Wissner
maintainer: belka@caraus.de maintainer: belka@caraus.de
-- copyright:
category: Language category: Language
build-type: Simple build-type: Simple
extra-doc-files: TODO README extra-doc-files: TODO README
@ -48,6 +46,8 @@ library elna-internal
Language.Elna.Glue Language.Elna.Glue
Language.Elna.Location Language.Elna.Location
Language.Elna.Object.Elf Language.Elna.Object.Elf
Language.Elna.Object.ElfCoder
Language.Elna.Object.StringTable
Language.Elna.RiscV.CodeGenerator Language.Elna.RiscV.CodeGenerator
Language.Elna.RiscV.ElfWriter Language.Elna.RiscV.ElfWriter
build-depends: build-depends:

View File

@ -20,9 +20,7 @@ module Language.Elna.Object.Elf
, ElfSectionType(..) , ElfSectionType(..)
, ElfSymbolBinding(..) , ElfSymbolBinding(..)
, ElfSymbolType(..) , ElfSymbolType(..)
, ElfWriter(..) , byteOrder
, ElfHeaderResult(..)
, addSectionHeader
, elf32Addr , elf32Addr
, elf32Half , elf32Half
, elf32Off , elf32Off
@ -33,10 +31,7 @@ module Language.Elna.Object.Elf
, elf32Rel , elf32Rel
, elf32Rela , elf32Rela
, elf32Sym , elf32Sym
, elfHeaderSize
, elfIdentification , elfIdentification
, elfObject
, elfSectionsSize
, rInfo , rInfo
, shfWrite , shfWrite
, shfAlloc , shfAlloc
@ -46,19 +41,12 @@ module Language.Elna.Object.Elf
, stInfo , stInfo
) where ) where
import Control.Exception (Exception(..), throwIO) import Control.Exception (Exception(..))
import Data.Bits (Bits(..)) import Data.Bits (Bits(..))
import qualified Data.ByteString.Builder as ByteString.Builder import qualified Data.ByteString.Builder as ByteString.Builder
import Data.Int (Int32) import Data.Int (Int32)
import Data.Word (Word8, Word16, Word32) import Data.Word (Word8, Word16, Word32)
import qualified Data.ByteString as ByteString import qualified Data.ByteString as ByteString
import Data.Vector (Vector)
import qualified Data.Vector as Vector
import System.IO (Handle, IOMode(..), SeekMode(..), hSeek, withFile)
import Data.Foldable (traverse_)
import Control.Monad.Trans.State (StateT, runStateT, modify')
import Control.Monad.IO.Class (MonadIO(..))
import Data.ByteString (ByteString)
-- * Data types. -- * Data types.
@ -498,90 +486,3 @@ instance Exception ElfEncodingError
fromIntegralEnum :: (Enum a, Num b) => a -> b fromIntegralEnum :: (Enum a, Num b) => a -> b
fromIntegralEnum = fromIntegral . fromEnum fromIntegralEnum = fromIntegral . fromEnum
-- * Object file generation.
newtype ElfWriter a = ElfWriter
{ runElfWriter :: StateT (ElfHeaderResult Elf32_Shdr) IO a
}
data ElfHeaderResult a = ElfHeaderResult
{ sectionNames :: ByteString
, sectionHeaders :: Vector a
} deriving Eq
instance Functor ElfWriter
where
fmap f (ElfWriter x) = ElfWriter $ f <$> x
instance Applicative ElfWriter
where
pure = ElfWriter . pure
(ElfWriter f) <*> (ElfWriter x) = ElfWriter $ f <*> x
instance Monad ElfWriter
where
(ElfWriter x) >>= f = ElfWriter $ x >>= (runElfWriter . f)
instance MonadIO ElfWriter
where
liftIO = ElfWriter . liftIO
-- | ELF header size.
elfHeaderSize :: Elf32_Off
elfHeaderSize = 52
-- | Calculates the size of all sections based on the 'sh_size' in the given
-- headers and adds 'elfHeaderSize' to it.
elfSectionsSize :: Vector Elf32_Shdr -> Elf32_Off
elfSectionsSize = (elfHeaderSize +)
. Vector.foldr ((+) . sh_size) 0
addSectionHeader :: ByteString -> Elf32_Shdr -> ElfWriter ()
addSectionHeader name newHeader = ElfWriter $ modify' modifier
where
modifier ElfHeaderResult{..} =
ElfHeaderResult
{ sectionHeaders = Vector.snoc sectionHeaders newHeader
, sectionNames = sectionNames <> name <> "\0"
}
-- Writes an ELF object to the provided file path. The callback writes the
-- sections, collects headers for those sections and returns the ELF header.
elfObject :: FilePath -> (Handle -> ElfWriter Elf32_Ehdr) -> IO ()
elfObject outFile putContents = withFile outFile WriteMode withObjectFile
where
withObjectFile objectHandle
= hSeek objectHandle AbsoluteSeek (fromIntegral elfHeaderSize)
>> putContents' objectHandle
>>= uncurry (afterContents objectHandle)
putContents' objectHandle
= flip runStateT initialState
$ runElfWriter
$ putContents objectHandle
zeroHeader = Elf32_Shdr
{ sh_type = SHT_NULL
, sh_size = 0
, sh_offset = 0
, sh_name = 0
, sh_link = 0
, sh_info = 0
, sh_flags = 0
, sh_entsize = 0
, sh_addralign = 0
, sh_addr = 0
}
initialState = ElfHeaderResult
{ sectionHeaders = Vector.singleton zeroHeader
, sectionNames = "\0"
}
afterContents objectHandle header ElfHeaderResult{..} =
let hPutBuilder = ByteString.Builder.hPutBuilder objectHandle
writeSectionHeaders byteOrder' =
traverse_ (hPutBuilder . elf32Shdr byteOrder') sectionHeaders
in either throwIO pure (byteOrder (e_ident header))
>>= writeSectionHeaders
>> either throwIO (putHeaders objectHandle) (elf32Ehdr header)
putHeaders objectHandle encodedHeader
= hSeek objectHandle AbsoluteSeek 0
>> ByteString.Builder.hPutBuilder objectHandle encodedHeader

View File

@ -0,0 +1,128 @@
-- | Object file generation.
module Language.Elna.Object.ElfCoder
( ElfEnvironment(..)
, ElfWriter(..)
, ElfHeaderResult(..)
, elfHeaderSize
, addSectionHeader
, elfObject
, elfSectionsSize
, putSectionHeader
, module Language.Elna.Object.Elf
) where
import Control.Exception (throwIO)
import Control.Monad.IO.Class (MonadIO(..))
import Control.Monad.Trans.State (StateT, runStateT, modify', gets)
import Data.ByteString (StrictByteString)
import qualified Data.ByteString as ByteString
import qualified Data.ByteString.Builder as ByteString.Builder
import Data.Vector (Vector)
import qualified Data.Vector as Vector
import System.IO (Handle, IOMode(..), SeekMode(..), hSeek, withFile)
import Data.Foldable (traverse_)
import Language.Elna.Object.Elf
import Language.Elna.Object.StringTable (StringTable)
import qualified Language.Elna.Object.StringTable as StringTable
import GHC.Records (HasField(..))
data ElfEnvironment = ElfEnvironment
{ objectHeaders :: ElfHeaderResult Elf32_Shdr
, objectHandle :: Handle
}
newtype ElfWriter a = ElfWriter
{ runElfWriter :: StateT ElfEnvironment IO a
}
data ElfHeaderResult a = ElfHeaderResult
{ sectionNames :: StringTable
, sectionHeaders :: Vector a
} deriving Eq
instance Functor ElfWriter
where
fmap f (ElfWriter x) = ElfWriter $ f <$> x
instance Applicative ElfWriter
where
pure = ElfWriter . pure
(ElfWriter f) <*> (ElfWriter x) = ElfWriter $ f <*> x
instance Monad ElfWriter
where
(ElfWriter x) >>= f = ElfWriter $ x >>= (runElfWriter . f)
instance MonadIO ElfWriter
where
liftIO = ElfWriter . liftIO
-- | ELF header size.
elfHeaderSize :: Elf32_Off
elfHeaderSize = 52
-- | Calculates the size of all sections based on the 'sh_size' in the given
-- headers and adds 'elfHeaderSize' to it.
elfSectionsSize :: Vector Elf32_Shdr -> Elf32_Off
elfSectionsSize = (elfHeaderSize +)
. Vector.foldr ((+) . sh_size) 0
addSectionHeader :: StrictByteString -> Elf32_Shdr -> ElfWriter ()
addSectionHeader name newHeader = ElfWriter $ modify' modifier
where
modifier elfEnvironment@ElfEnvironment{ objectHeaders } =
let ElfHeaderResult{..} = objectHeaders
in elfEnvironment
{ objectHeaders = ElfHeaderResult
{ sectionHeaders = Vector.snoc sectionHeaders newHeader
, sectionNames = StringTable.append name sectionNames
}
}
putSectionHeader :: StrictByteString -> Elf32_Shdr -> StrictByteString -> ElfWriter ()
putSectionHeader name newHeader encoded = do
objectHandle' <- ElfWriter $ gets $ getField @"objectHandle"
liftIO $ ByteString.hPut objectHandle' encoded
addSectionHeader name newHeader
-- Writes an ELF object to the provided file path. The callback writes the
-- sections, collects headers for those sections and returns the ELF header.
elfObject :: FilePath -> ElfWriter Elf32_Ehdr -> IO ()
elfObject outFile putContents = withFile outFile WriteMode withObjectFile
where
withObjectFile objectHandle
= hSeek objectHandle AbsoluteSeek (fromIntegral elfHeaderSize)
>> putContents' objectHandle
>>= uncurry afterContents
putContents' objectHandle
= flip runStateT (initialState objectHandle)
$ runElfWriter putContents
zeroHeader = Elf32_Shdr
{ sh_type = SHT_NULL
, sh_size = 0
, sh_offset = 0
, sh_name = 0
, sh_link = 0
, sh_info = 0
, sh_flags = 0
, sh_entsize = 0
, sh_addralign = 0
, sh_addr = 0
}
initialState objectHandle = ElfEnvironment
{ objectHeaders = ElfHeaderResult
{ sectionHeaders = Vector.singleton zeroHeader
, sectionNames = mempty
}
, objectHandle = objectHandle
}
afterContents header ElfEnvironment{ objectHeaders = ElfHeaderResult{..}, ..} =
let hPutBuilder = ByteString.Builder.hPutBuilder objectHandle
writeSectionHeaders byteOrder' =
traverse_ (hPutBuilder . elf32Shdr byteOrder') sectionHeaders
in either throwIO pure (byteOrder (e_ident header))
>>= writeSectionHeaders
>> either throwIO (putHeaders objectHandle) (elf32Ehdr header)
putHeaders objectHandle encodedHeader
= hSeek objectHandle AbsoluteSeek 0
>> ByteString.Builder.hPutBuilder objectHandle encodedHeader

View File

@ -0,0 +1,44 @@
module Language.Elna.Object.StringTable
( StringTable
, append
, elem
, index
, encode
, size
) where
import Data.ByteString (StrictByteString)
import qualified Data.ByteString as ByteString
import Language.Elna.Object.Elf
import Prelude hiding (elem)
newtype StringTable = StringTable StrictByteString
deriving Eq
instance Semigroup StringTable
where
(StringTable x) <> (StringTable y) = StringTable $ x <> ByteString.drop 1 y
instance Monoid StringTable
where
mempty = StringTable "\0"
size :: StringTable -> Elf32_Word
size (StringTable container) =
fromIntegral $ ByteString.length container
elem :: StrictByteString -> StringTable -> Bool
elem needle (StringTable container) =
("\0" <> needle <> "\0") `ByteString.isInfixOf` container
append :: StrictByteString -> StringTable -> StringTable
append element (StringTable container) =
StringTable $ container <> element <> "\0"
index :: Elf32_Word -> StringTable -> StrictByteString
index stringTableIndex (StringTable stringTable)
= ByteString.takeWhile (/= 0)
$ ByteString.drop (fromIntegral stringTableIndex) stringTable
encode :: StringTable -> StrictByteString
encode (StringTable container) = container

View File

@ -4,13 +4,13 @@ module Language.Elna.RiscV.ElfWriter
) where ) where
import Data.Word (Word8) import Data.Word (Word8)
import Data.ByteString (ByteString) import Data.ByteString (StrictByteString)
import qualified Data.ByteString as ByteString import qualified Data.ByteString as ByteString
import qualified Data.ByteString.Builder as ByteString.Builder import qualified Data.ByteString.Builder as ByteString.Builder
import qualified Data.ByteString.Lazy as LazyByteString import qualified Data.ByteString.Lazy as LazyByteString
import Data.Vector (Vector) import Data.Vector (Vector)
import qualified Data.Vector as Vector import qualified Data.Vector as Vector
import Language.Elna.Object.Elf import Language.Elna.Object.ElfCoder
( ByteOrder(..) ( ByteOrder(..)
, Elf32_Addr , Elf32_Addr
, Elf32_Ehdr(..) , Elf32_Ehdr(..)
@ -30,6 +30,7 @@ import Language.Elna.Object.Elf
, Elf32_Rel(..) , Elf32_Rel(..)
, ElfWriter(..) , ElfWriter(..)
, ElfHeaderResult(..) , ElfHeaderResult(..)
, ElfEnvironment(..)
, elf32Sym , elf32Sym
, elfHeaderSize , elfHeaderSize
, elfSectionsSize , elfSectionsSize
@ -38,21 +39,22 @@ import Language.Elna.Object.Elf
, elf32Rel , elf32Rel
, shfInfoLink , shfInfoLink
, addSectionHeader , addSectionHeader
, putSectionHeader
) )
import System.IO (Handle)
import qualified Language.Elna.Architecture.RiscV as RiscV import qualified Language.Elna.Architecture.RiscV as RiscV
import qualified Data.Text.Encoding as Text import qualified Data.Text.Encoding as Text
import Control.Monad.IO.Class (MonadIO(..)) import Control.Monad.IO.Class (MonadIO(..))
import Control.Monad.Trans.State (get) import Control.Monad.Trans.State (get, gets)
import Language.Elna.RiscV.CodeGenerator (Statement(..)) import Language.Elna.RiscV.CodeGenerator (Statement(..))
import qualified Language.Elna.Object.StringTable as StringTable
import qualified Data.HashSet as HashSet import qualified Data.HashSet as HashSet
import GHC.Records (HasField(..)) import GHC.Records (HasField(..))
data UnresolvedRelocation = UnresolvedRelocation ByteString Elf32_Addr Word8 data UnresolvedRelocation = UnresolvedRelocation StrictByteString Elf32_Addr Word8
data UnresolvedRelocations = UnresolvedRelocations (Vector UnresolvedRelocation) Elf32_Word data UnresolvedRelocations = UnresolvedRelocations (Vector UnresolvedRelocation) Elf32_Word
riscv32Elf :: Vector Statement -> Handle -> ElfWriter Elf32_Ehdr riscv32Elf :: Vector Statement -> ElfWriter Elf32_Ehdr
riscv32Elf code objectHandle = text riscv32Elf code = text
>>= uncurry symrel >>= uncurry symrel
>>= strtab >>= strtab
>> shstrtab >> shstrtab
@ -60,13 +62,15 @@ riscv32Elf code objectHandle = text
where where
shstrtab :: ElfWriter Elf32_Half shstrtab :: ElfWriter Elf32_Half
shstrtab = do shstrtab = do
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
let stringTable = sectionNames <> ".shstrtab\0" let stringTable = ".shstrtab"
currentNamesSize = StringTable.size sectionNames
nextHeader = Elf32_Shdr nextHeader = Elf32_Shdr
{ sh_type = SHT_STRTAB { sh_type = SHT_STRTAB
, sh_size = fromIntegral $ ByteString.length stringTable , sh_size = currentNamesSize -- Adding trailing null character.
+ fromIntegral (succ $ ByteString.length stringTable)
, sh_offset = elfSectionsSize sectionHeaders , sh_offset = elfSectionsSize sectionHeaders
, sh_name = fromIntegral $ ByteString.length sectionNames , sh_name = currentNamesSize
, sh_link = 0 , sh_link = 0
, sh_info = 0 , sh_info = 0
, sh_flags = 0 , sh_flags = 0
@ -74,12 +78,16 @@ riscv32Elf code objectHandle = text
, sh_addralign = 1 , sh_addralign = 1
, sh_addr = 0 , sh_addr = 0
} }
liftIO $ ByteString.hPut objectHandle stringTable addSectionHeader stringTable nextHeader
addSectionHeader ".shstrtab" nextHeader
ElfEnvironment{..} <- ElfWriter get
liftIO $ ByteString.hPut objectHandle
$ StringTable.encode
$ getField @"sectionNames" objectHeaders
pure $ fromIntegral $ Vector.length sectionHeaders pure $ fromIntegral $ Vector.length sectionHeaders
riscv32Header :: Elf32_Half -> ElfWriter Elf32_Ehdr riscv32Header :: Elf32_Half -> ElfWriter Elf32_Ehdr
riscv32Header shstrndx = do riscv32Header shstrndx = do
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
pure $ Elf32_Ehdr pure $ Elf32_Ehdr
{ e_version = EV_CURRENT { e_version = EV_CURRENT
, e_type = ET_REL , e_type = ET_REL
@ -97,8 +105,7 @@ riscv32Elf code objectHandle = text
, e_ehsize = fromIntegral elfHeaderSize , e_ehsize = fromIntegral elfHeaderSize
} }
takeStringZ stringTable Elf32_Sym{ st_name } takeStringZ stringTable Elf32_Sym{ st_name }
= ByteString.takeWhile (/= 0) = StringTable.index st_name stringTable
$ ByteString.drop (fromIntegral st_name) stringTable
resolveRelocation (ElfHeaderResult stringTable entries) unresolvedRelocation resolveRelocation (ElfHeaderResult stringTable entries) unresolvedRelocation
| UnresolvedRelocation symbolName offset type' <- unresolvedRelocation | UnresolvedRelocation symbolName offset type' <- unresolvedRelocation
, Just entry <- Vector.findIndex ((== symbolName) . takeStringZ stringTable) entries = , Just entry <- Vector.findIndex ((== symbolName) . takeStringZ stringTable) entries =
@ -108,7 +115,7 @@ riscv32Elf code objectHandle = text
} }
| otherwise = Left unresolvedRelocation | otherwise = Left unresolvedRelocation
symtab entries = do symtab entries = do
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
let encodedSymbols = LazyByteString.toStrict let encodedSymbols = LazyByteString.toStrict
$ ByteString.Builder.toLazyByteString $ ByteString.Builder.toLazyByteString
$ foldMap (elf32Sym LSB) entries $ foldMap (elf32Sym LSB) entries
@ -116,7 +123,7 @@ riscv32Elf code objectHandle = text
{ sh_type = SHT_SYMTAB { sh_type = SHT_SYMTAB
, sh_size = fromIntegral $ ByteString.length encodedSymbols , sh_size = fromIntegral $ ByteString.length encodedSymbols
, sh_offset = elfSectionsSize sectionHeaders , sh_offset = elfSectionsSize sectionHeaders
, sh_name = fromIntegral $ ByteString.length sectionNames , sh_name = StringTable.size sectionNames
, sh_link = fromIntegral $ Vector.length sectionHeaders + 2 , sh_link = fromIntegral $ Vector.length sectionHeaders + 2
, sh_info = 1 , sh_info = 1
, sh_flags = 0 , sh_flags = 0
@ -124,15 +131,14 @@ riscv32Elf code objectHandle = text
, sh_addralign = 4 , sh_addralign = 4
, sh_addr = 0 , sh_addr = 0
} }
liftIO $ ByteString.hPut objectHandle encodedSymbols putSectionHeader ".symtab" symHeader encodedSymbols
addSectionHeader ".symtab" symHeader
pure $ fromIntegral $ Vector.length sectionHeaders pure $ fromIntegral $ Vector.length sectionHeaders
symrel symbols relocations = do symrel symbols relocations = do
let UnresolvedRelocations relocationList index = relocations let UnresolvedRelocations relocationList index = relocations
ElfHeaderResult stringTable entries = symbols ElfHeaderResult stringTable entries = symbols
sectionHeadersLength <- symtab entries sectionHeadersLength <- symtab entries
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
let encodedRelocations = LazyByteString.toStrict let encodedRelocations = LazyByteString.toStrict
$ ByteString.Builder.toLazyByteString $ ByteString.Builder.toLazyByteString
@ -142,7 +148,7 @@ riscv32Elf code objectHandle = text
{ sh_type = SHT_REL { sh_type = SHT_REL
, sh_size = fromIntegral $ ByteString.length encodedRelocations , sh_size = fromIntegral $ ByteString.length encodedRelocations
, sh_offset = elfSectionsSize sectionHeaders , sh_offset = elfSectionsSize sectionHeaders
, sh_name = fromIntegral $ ByteString.length sectionNames , sh_name = StringTable.size sectionNames
, sh_link = sectionHeadersLength , sh_link = sectionHeadersLength
, sh_info = index , sh_info = index
, sh_flags = shfInfoLink , sh_flags = shfInfoLink
@ -150,16 +156,15 @@ riscv32Elf code objectHandle = text
, sh_addralign = 4 , sh_addralign = 4
, sh_addr = 0 , sh_addr = 0
} }
liftIO $ ByteString.hPut objectHandle encodedRelocations putSectionHeader ".rel.text" relHeader encodedRelocations
addSectionHeader ".rel.text" relHeader
pure stringTable pure stringTable
strtab stringTable = do strtab stringTable = do
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
let strHeader = Elf32_Shdr let strHeader = Elf32_Shdr
{ sh_type = SHT_STRTAB { sh_type = SHT_STRTAB
, sh_size = fromIntegral $ ByteString.length stringTable , sh_size = StringTable.size stringTable
, sh_offset = elfSectionsSize sectionHeaders , sh_offset = elfSectionsSize sectionHeaders
, sh_name = fromIntegral $ ByteString.length sectionNames , sh_name = StringTable.size sectionNames
, sh_link = 0 , sh_link = 0
, sh_info = 0 , sh_info = 0
, sh_flags = 0 , sh_flags = 0
@ -167,12 +172,11 @@ riscv32Elf code objectHandle = text
, sh_addralign = 1 , sh_addralign = 1
, sh_addr = 0 , sh_addr = 0
} }
liftIO $ ByteString.hPut objectHandle stringTable putSectionHeader ".strtab" strHeader $ StringTable.encode stringTable
addSectionHeader ".strtab" strHeader
text = do text = do
ElfHeaderResult{..} <- ElfWriter get ElfHeaderResult{..} <- ElfWriter $ gets $ getField @"objectHeaders"
let textTabIndex = fromIntegral $ Vector.length sectionHeaders let textTabIndex = fromIntegral $ Vector.length sectionHeaders
initialHeaders = ElfHeaderResult "\0" initialHeaders = ElfHeaderResult mempty
$ Vector.singleton $ Vector.singleton
$ Elf32_Sym $ Elf32_Sym
{ st_value = 0 { st_value = 0
@ -183,19 +187,13 @@ riscv32Elf code objectHandle = text
, st_info = 0 , st_info = 0
} }
(encoded, updatedRelocations, symbols, definitions) = (encoded, updatedRelocations, symbols, definitions) =
encodeAsm textTabIndex (mempty, Vector.empty, initialHeaders, HashSet.empty) code encodeFunctions textTabIndex (mempty, Vector.empty, initialHeaders, HashSet.empty) code
filterPredicate = not
. (`ByteString.isInfixOf` getField @"sectionNames" symbols)
. ("\0" <>) . (<> "\0")
symbolResult = HashSet.foldl' encodeEmptyDefinitions symbols
$ HashSet.filter filterPredicate definitions
size = fromIntegral $ LazyByteString.length encoded size = fromIntegral $ LazyByteString.length encoded
newHeader = Elf32_Shdr newHeader = Elf32_Shdr
{ sh_type = SHT_PROGBITS { sh_type = SHT_PROGBITS
, sh_size = size , sh_size = size
, sh_offset = elfSectionsSize sectionHeaders , sh_offset = elfSectionsSize sectionHeaders
, sh_name = fromIntegral $ ByteString.length sectionNames , sh_name = StringTable.size sectionNames
, sh_link = 0 , sh_link = 0
, sh_info = 0 , sh_info = 0
, sh_flags = 0b110 , sh_flags = 0b110
@ -203,8 +201,12 @@ riscv32Elf code objectHandle = text
, sh_addralign = 4 , sh_addralign = 4
, sh_addr = 0 , sh_addr = 0
} }
liftIO $ ByteString.hPut objectHandle $ LazyByteString.toStrict encoded putSectionHeader ".text" newHeader $ LazyByteString.toStrict encoded
addSectionHeader ".text" newHeader let filterPredicate :: StrictByteString -> Bool
filterPredicate = not
. (`StringTable.elem` getField @"sectionNames" symbols)
symbolResult = HashSet.foldl' encodeEmptyDefinitions symbols
$ HashSet.filter filterPredicate definitions
pure (symbolResult, UnresolvedRelocations updatedRelocations $ fromIntegral $ Vector.length sectionHeaders) pure (symbolResult, UnresolvedRelocations updatedRelocations $ fromIntegral $ Vector.length sectionHeaders)
encodeEmptyDefinitions (ElfHeaderResult names entries) definition = encodeEmptyDefinitions (ElfHeaderResult names entries) definition =
let nextEntry = Elf32_Sym let nextEntry = Elf32_Sym
@ -212,18 +214,18 @@ riscv32Elf code objectHandle = text
, st_size = 0 , st_size = 0
, st_shndx = 0 , st_shndx = 0
, st_other = 0 , st_other = 0
, st_name = fromIntegral (ByteString.length names) , st_name = StringTable.size names
, st_info = stInfo STB_GLOBAL STT_FUNC , st_info = stInfo STB_GLOBAL STT_FUNC
} }
in ElfHeaderResult (names <> definition <> "\0") in ElfHeaderResult (StringTable.append definition names)
$ Vector.snoc entries nextEntry $ Vector.snoc entries nextEntry
encodeAsm shndx (encoded, relocations, ElfHeaderResult names symbols, definitions) instructions encodeFunctions shndx (encoded, relocations, ElfHeaderResult names symbols, definitions) instructions
| Just (instruction, rest) <- Vector.uncons instructions = | Just (instruction, rest) <- Vector.uncons instructions =
case instruction of case instruction of
Instruction _ -> Instruction _ ->
let (encoded', relocations', rest', definitions') = let (encoded', relocations', rest', definitions') =
encodeInstructions (encoded, relocations, instructions, definitions) encodeInstructions (encoded, relocations, instructions, definitions)
in encodeAsm shndx (encoded', relocations', ElfHeaderResult names symbols, definitions') rest' in encodeFunctions shndx (encoded', relocations', ElfHeaderResult names symbols, definitions') rest'
JumpLabel labelName _ -> JumpLabel labelName _ ->
let (encoded', relocations', rest', definitions') = let (encoded', relocations', rest', definitions') =
encodeInstructions (encoded, relocations, rest, definitions) encodeInstructions (encoded, relocations, rest, definitions)
@ -232,16 +234,17 @@ riscv32Elf code objectHandle = text
, st_size = fromIntegral $ LazyByteString.length encoded' , st_size = fromIntegral $ LazyByteString.length encoded'
, st_shndx = shndx , st_shndx = shndx
, st_other = 0 , st_other = 0
, st_name = fromIntegral $ ByteString.length names , st_name = StringTable.size names
, st_info = stInfo STB_GLOBAL STT_FUNC , st_info = stInfo STB_GLOBAL STT_FUNC
} }
result = result =
( encoded' ( encoded'
, relocations' , relocations'
, ElfHeaderResult (names <> Text.encodeUtf8 labelName <> "\0") (Vector.snoc symbols newEntry) , ElfHeaderResult (StringTable.append (Text.encodeUtf8 labelName) names)
$ Vector.snoc symbols newEntry
, definitions' , definitions'
) )
in encodeAsm shndx result rest' in encodeFunctions shndx result rest'
| otherwise = (encoded, relocations, ElfHeaderResult names symbols, definitions) | otherwise = (encoded, relocations, ElfHeaderResult names symbols, definitions)
encodeInstructions (encoded, relocations, instructions, definitions) encodeInstructions (encoded, relocations, instructions, definitions)
| Just (Instruction instruction, rest) <- Vector.uncons instructions = | Just (Instruction instruction, rest) <- Vector.uncons instructions =

View File

@ -3,7 +3,7 @@ module Main
) where ) where
import Language.Elna.CommandLine (CommandLine(..), commandLine, execParser) import Language.Elna.CommandLine (CommandLine(..), commandLine, execParser)
import Language.Elna.Object.Elf (elfObject) import Language.Elna.Object.ElfCoder (elfObject)
import Language.Elna.Backend.Allocator (allocate) import Language.Elna.Backend.Allocator (allocate)
import Language.Elna.Glue (glue) import Language.Elna.Glue (glue)
import Language.Elna.Frontend.NameAnalysis (nameAnalysis) import Language.Elna.Frontend.NameAnalysis (nameAnalysis)