Abstract the string table into a newtype
This commit is contained in:
parent
bf5ec1f3e2
commit
57436f664e
@ -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:
|
||||||
|
@ -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
|
|
||||||
|
128
lib/Language/Elna/Object/ElfCoder.hs
Normal file
128
lib/Language/Elna/Object/ElfCoder.hs
Normal 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
|
44
lib/Language/Elna/Object/StringTable.hs
Normal file
44
lib/Language/Elna/Object/StringTable.hs
Normal 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
|
@ -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 =
|
||||||
|
@ -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)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user