mirror of
https://github.com/GaloisInc/macaw.git
synced 2024-12-29 00:59:09 +03:00
Merge remote-tracking branch 'public/master' into jhx/plt-support
This commit is contained in:
commit
ebc5d9575e
@ -3,7 +3,7 @@ version: 0.3.4
|
|||||||
author: Galois, Inc.
|
author: Galois, Inc.
|
||||||
maintainer: jhendrix@galois.com
|
maintainer: jhendrix@galois.com
|
||||||
build-type: Simple
|
build-type: Simple
|
||||||
cabal-version: >= 1.9.2
|
cabal-version: >= 1.10
|
||||||
license: BSD3
|
license: BSD3
|
||||||
license-file: LICENSE
|
license-file: LICENSE
|
||||||
description:
|
description:
|
||||||
@ -80,3 +80,7 @@ library
|
|||||||
ghc-options: -Wall
|
ghc-options: -Wall
|
||||||
ghc-options: -fno-warn-unticked-promoted-constructors
|
ghc-options: -fno-warn-unticked-promoted-constructors
|
||||||
ghc-prof-options: -O2 -fprof-auto-top
|
ghc-prof-options: -O2 -fprof-auto-top
|
||||||
|
default-language: Haskell2010
|
||||||
|
|
||||||
|
if impl(ghc >= 8.6)
|
||||||
|
default-extensions: NoStarIsType
|
||||||
|
@ -66,6 +66,7 @@ import Data.Bits
|
|||||||
import Data.Foldable
|
import Data.Foldable
|
||||||
import Data.Functor
|
import Data.Functor
|
||||||
import Data.Int
|
import Data.Int
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
import qualified Data.Map.Strict as Map
|
import qualified Data.Map.Strict as Map
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
@ -1316,7 +1317,7 @@ transferApp r a = do
|
|||||||
_ -> TopV
|
_ -> TopV
|
||||||
|
|
||||||
-- | Minimal information needed to parse a function call/system call
|
-- | Minimal information needed to parse a function call/system call
|
||||||
data CallParams (r :: Type -> *)
|
data CallParams (r :: Type -> Kind.Type)
|
||||||
= CallParams { postCallStackDelta :: Integer
|
= CallParams { postCallStackDelta :: Integer
|
||||||
-- ^ Amount stack should shift by when going before/after call.
|
-- ^ Amount stack should shift by when going before/after call.
|
||||||
, preserveReg :: forall tp . r tp -> Bool
|
, preserveReg :: forall tp . r tp -> Bool
|
||||||
|
@ -34,6 +34,7 @@ module Data.Macaw.Analysis.FunctionArgs
|
|||||||
import Control.Lens
|
import Control.Lens
|
||||||
import Control.Monad.State.Strict
|
import Control.Monad.State.Strict
|
||||||
import Data.Foldable as Fold (traverse_)
|
import Data.Foldable as Fold (traverse_)
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Map.Strict (Map)
|
import Data.Map.Strict (Map)
|
||||||
import qualified Data.Map.Strict as Map
|
import qualified Data.Map.Strict as Map
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
@ -87,13 +88,13 @@ instance (Applicative f, Monoid a) => Monoid (Ap f a) where
|
|||||||
-- demanded?
|
-- demanded?
|
||||||
|
|
||||||
-- | A set of registrs
|
-- | A set of registrs
|
||||||
type RegisterSet (r :: Type -> *) = Set (Some r)
|
type RegisterSet (r :: Type -> Kind.Type) = Set (Some r)
|
||||||
|
|
||||||
-- | A memory segment offset compatible with the architecture registers.
|
-- | A memory segment offset compatible with the architecture registers.
|
||||||
type RegSegmentOff r = MemSegmentOff (RegAddrWidth r)
|
type RegSegmentOff r = MemSegmentOff (RegAddrWidth r)
|
||||||
|
|
||||||
-- | This stores the registers needed by a specific address
|
-- | This stores the registers needed by a specific address
|
||||||
data DemandSet (r :: Type -> *) =
|
data DemandSet (r :: Type -> Kind.Type) =
|
||||||
DemandSet { registerDemands :: !(RegisterSet r)
|
DemandSet { registerDemands :: !(RegisterSet r)
|
||||||
-- | This maps a function address to the registers
|
-- | This maps a function address to the registers
|
||||||
-- that it needs.
|
-- that it needs.
|
||||||
|
@ -21,6 +21,7 @@ module Data.Macaw.CFG.App
|
|||||||
, ppAppA
|
, ppAppA
|
||||||
) where
|
) where
|
||||||
|
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Control.Monad.Identity
|
import Control.Monad.Identity
|
||||||
import Data.Parameterized.Classes
|
import Data.Parameterized.Classes
|
||||||
import qualified Data.Parameterized.List as P
|
import qualified Data.Parameterized.List as P
|
||||||
@ -40,7 +41,7 @@ import Data.Macaw.Utils.Pretty
|
|||||||
-- These operations are all total functions. Different architecture tend to have
|
-- These operations are all total functions. Different architecture tend to have
|
||||||
-- different ways of raising signals or exceptions, and so partial functions are
|
-- different ways of raising signals or exceptions, and so partial functions are
|
||||||
-- all architecture specific.
|
-- all architecture specific.
|
||||||
data App (f :: Type -> *) (tp :: Type) where
|
data App (f :: Type -> Kind.Type) (tp :: Type) where
|
||||||
|
|
||||||
-- Compare for equality.
|
-- Compare for equality.
|
||||||
Eq :: !(f tp) -> !(f tp) -> App f BoolType
|
Eq :: !(f tp) -> !(f tp) -> App f BoolType
|
||||||
|
@ -6,6 +6,8 @@
|
|||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
{-# LANGUAGE TypeFamilyDependencies #-}
|
{-# LANGUAGE TypeFamilyDependencies #-}
|
||||||
{-# LANGUAGE TypeOperators #-}
|
{-# LANGUAGE TypeOperators #-}
|
||||||
|
{-# LANGUAGE UndecidableInstances #-}
|
||||||
|
|
||||||
module Data.Macaw.CFG.AssignRhs
|
module Data.Macaw.CFG.AssignRhs
|
||||||
( AssignRhs(..)
|
( AssignRhs(..)
|
||||||
-- * MemRepr
|
-- * MemRepr
|
||||||
@ -25,6 +27,7 @@ module Data.Macaw.CFG.AssignRhs
|
|||||||
, ArchMemAddr
|
, ArchMemAddr
|
||||||
) where
|
) where
|
||||||
|
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Macaw.CFG.App
|
import Data.Macaw.CFG.App
|
||||||
import Data.Macaw.Memory (Endianness(..), MemSegmentOff, MemWord, MemAddr)
|
import Data.Macaw.Memory (Endianness(..), MemSegmentOff, MemWord, MemAddr)
|
||||||
import Data.Macaw.Types
|
import Data.Macaw.Types
|
||||||
@ -36,8 +39,11 @@ import Data.Parameterized.TraversableFC (FoldableFC(..))
|
|||||||
import Data.Proxy
|
import Data.Proxy
|
||||||
import Text.PrettyPrint.ANSI.Leijen hiding ((<$>), (<>))
|
import Text.PrettyPrint.ANSI.Leijen hiding ((<$>), (<>))
|
||||||
|
|
||||||
|
import Prelude
|
||||||
|
|
||||||
|
|
||||||
-- | Width of register used to store addresses.
|
-- | Width of register used to store addresses.
|
||||||
type family RegAddrWidth (r :: Type -> *) :: Nat
|
type family RegAddrWidth (r :: Type -> Kind.Type) :: Nat
|
||||||
|
|
||||||
-- | A word for the given architecture register type.
|
-- | A word for the given architecture register type.
|
||||||
type RegAddrWord r = MemWord (RegAddrWidth r)
|
type RegAddrWord r = MemWord (RegAddrWidth r)
|
||||||
@ -46,7 +52,7 @@ type RegAddrWord r = MemWord (RegAddrWidth r)
|
|||||||
--
|
--
|
||||||
-- Registers include things like the general purpose registers, any flag
|
-- Registers include things like the general purpose registers, any flag
|
||||||
-- registers that can be read and written without side effects,
|
-- registers that can be read and written without side effects,
|
||||||
type family ArchReg (arch :: *) = (reg :: Type -> *) | reg -> arch
|
type family ArchReg (arch :: Kind.Type) = (reg :: Type -> Kind.Type) | reg -> arch
|
||||||
-- Note the injectivity constraint. This makes GHC quit bothering us
|
-- Note the injectivity constraint. This makes GHC quit bothering us
|
||||||
-- about ambigous types for functions taking ArchRegs as arguments.
|
-- about ambigous types for functions taking ArchRegs as arguments.
|
||||||
|
|
||||||
@ -57,13 +63,13 @@ type family ArchReg (arch :: *) = (reg :: Type -> *) | reg -> arch
|
|||||||
--
|
--
|
||||||
-- The function may depend on the set of registers defined so far, and the type
|
-- The function may depend on the set of registers defined so far, and the type
|
||||||
-- of the result.
|
-- of the result.
|
||||||
type family ArchFn (arch :: *) = (fn :: (Type -> *) -> Type -> *) | fn -> arch
|
type family ArchFn (arch :: Kind.Type) = (fn :: (Type -> Kind.Type) -> Type -> Kind.Type) | fn -> arch
|
||||||
|
|
||||||
-- | A type family for defining architecture-specific statements.
|
-- | A type family for defining architecture-specific statements.
|
||||||
--
|
--
|
||||||
-- The second parameter is used to denote the underlying values in the
|
-- The second parameter is used to denote the underlying values in the
|
||||||
-- statements so that we can use ArchStmts with multiple CFGs.
|
-- statements so that we can use ArchStmts with multiple CFGs.
|
||||||
type family ArchStmt (arch :: *) = (stmt :: (Type -> *) -> *) | stmt -> arch
|
type family ArchStmt (arch :: Kind.Type) = (stmt :: (Type -> Kind.Type) -> Kind.Type) | stmt -> arch
|
||||||
|
|
||||||
-- | A type family for defining architecture-specific statements that
|
-- | A type family for defining architecture-specific statements that
|
||||||
-- may have instruction-specific effects on control-flow and register state.
|
-- may have instruction-specific effects on control-flow and register state.
|
||||||
@ -75,7 +81,7 @@ type family ArchStmt (arch :: *) = (stmt :: (Type -> *) -> *) | stmt -> arch
|
|||||||
-- values, it may or may not return to the current function. If it does return to the
|
-- values, it may or may not return to the current function. If it does return to the
|
||||||
-- current function, it is assumed to be at most one location, and the block-translator
|
-- current function, it is assumed to be at most one location, and the block-translator
|
||||||
-- must provide that value at translation time.
|
-- must provide that value at translation time.
|
||||||
type family ArchTermStmt (arch :: *) :: * -> *
|
type family ArchTermStmt (arch :: Kind.Type) :: Kind.Type -> Kind.Type
|
||||||
-- NOTE: Not injective because PPC32 and PPC64 use the same type.
|
-- NOTE: Not injective because PPC32 and PPC64 use the same type.
|
||||||
|
|
||||||
-- | Number of bits in addreses for architecture.
|
-- | Number of bits in addreses for architecture.
|
||||||
@ -134,7 +140,7 @@ instance HasRepr MemRepr TypeRepr where
|
|||||||
|
|
||||||
-- | The right hand side of an assignment is an expression that
|
-- | The right hand side of an assignment is an expression that
|
||||||
-- returns a value.
|
-- returns a value.
|
||||||
data AssignRhs (arch :: *) (f :: Type -> *) tp where
|
data AssignRhs (arch :: Kind.Type) (f :: Type -> Kind.Type) tp where
|
||||||
-- | An expression that is computed from evaluating subexpressions.
|
-- | An expression that is computed from evaluating subexpressions.
|
||||||
EvalApp :: !(App f tp)
|
EvalApp :: !(App f tp)
|
||||||
-> AssignRhs arch f tp
|
-> AssignRhs arch f tp
|
||||||
|
@ -6,6 +6,8 @@ This exports the pre-classification term statement and block data
|
|||||||
types.
|
types.
|
||||||
-}
|
-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
|
{-# LANGUAGE UndecidableInstances #-}
|
||||||
|
|
||||||
module Data.Macaw.CFG.Block
|
module Data.Macaw.CFG.Block
|
||||||
( Block(..)
|
( Block(..)
|
||||||
, ppBlock
|
, ppBlock
|
||||||
|
@ -7,6 +7,7 @@ Defines data types needed to represent values, assignments, and statements from
|
|||||||
This is a low-level CFG representation where the entire program is a
|
This is a low-level CFG representation where the entire program is a
|
||||||
single CFG.
|
single CFG.
|
||||||
-}
|
-}
|
||||||
|
{-# LANGUAGE AllowAmbiguousTypes #-}
|
||||||
{-# LANGUAGE ConstraintKinds #-}
|
{-# LANGUAGE ConstraintKinds #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
@ -21,7 +22,8 @@ single CFG.
|
|||||||
{-# LANGUAGE StandaloneDeriving #-}
|
{-# LANGUAGE StandaloneDeriving #-}
|
||||||
{-# LANGUAGE TypeFamilies #-}
|
{-# LANGUAGE TypeFamilies #-}
|
||||||
{-# LANGUAGE TypeOperators #-}
|
{-# LANGUAGE TypeOperators #-}
|
||||||
{-# LANGUAGE AllowAmbiguousTypes #-}
|
{-# LANGUAGE UndecidableInstances #-}
|
||||||
|
|
||||||
module Data.Macaw.CFG.Core
|
module Data.Macaw.CFG.Core
|
||||||
( -- * Stmt level declarations
|
( -- * Stmt level declarations
|
||||||
Stmt(..)
|
Stmt(..)
|
||||||
@ -83,6 +85,7 @@ import Control.Monad.Identity
|
|||||||
import Control.Monad.State.Strict
|
import Control.Monad.State.Strict
|
||||||
import Data.Bits
|
import Data.Bits
|
||||||
import Data.Int (Int64)
|
import Data.Int (Int64)
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Maybe (isNothing, catMaybes)
|
import Data.Maybe (isNothing, catMaybes)
|
||||||
import Data.Parameterized.Classes
|
import Data.Parameterized.Classes
|
||||||
import Data.Parameterized.Map (MapF)
|
import Data.Parameterized.Map (MapF)
|
||||||
@ -124,7 +127,7 @@ class PrettyPrec v where
|
|||||||
prettyPrec :: Int -> v -> Doc
|
prettyPrec :: Int -> v -> Doc
|
||||||
|
|
||||||
-- | Pretty print over all instances of a type.
|
-- | Pretty print over all instances of a type.
|
||||||
class PrettyF (f :: k -> *) where
|
class PrettyF (f :: k -> Kind.Type) where
|
||||||
prettyF :: f tp -> Doc
|
prettyF :: f tp -> Doc
|
||||||
|
|
||||||
-- | Pretty print a document with parens if condition is true
|
-- | Pretty print a document with parens if condition is true
|
||||||
@ -152,7 +155,7 @@ addrWidthTypeRepr Addr64 = BVTypeRepr knownNat
|
|||||||
-- form. 'AssignId's are typed, and also include a type variable @ids@
|
-- form. 'AssignId's are typed, and also include a type variable @ids@
|
||||||
-- that intuitively denotes the set of identifiers from which they are
|
-- that intuitively denotes the set of identifiers from which they are
|
||||||
-- drawn.
|
-- drawn.
|
||||||
newtype AssignId (ids :: *) (tp :: Type) = AssignId (Nonce ids tp)
|
newtype AssignId (ids :: Kind.Type) (tp :: Type) = AssignId (Nonce ids tp)
|
||||||
|
|
||||||
ppAssignId :: AssignId ids tp -> Doc
|
ppAssignId :: AssignId ids tp -> Doc
|
||||||
ppAssignId (AssignId w) = text ("r" ++ show (indexValue w))
|
ppAssignId (AssignId w) = text ("r" ++ show (indexValue w))
|
||||||
@ -391,7 +394,7 @@ class IPAlignment arch where
|
|||||||
-- RegState
|
-- RegState
|
||||||
|
|
||||||
-- | This represents the state of the processor registers.
|
-- | This represents the state of the processor registers.
|
||||||
newtype RegState (r :: k -> *) (f :: k -> *) = RegState (MapF.MapF r f)
|
newtype RegState (r :: k -> Kind.Type) (f :: k -> Kind.Type) = RegState (MapF.MapF r f)
|
||||||
|
|
||||||
deriving instance (OrdF r, EqF f) => Eq (RegState r f)
|
deriving instance (OrdF r, EqF f) => Eq (RegState r f)
|
||||||
|
|
||||||
@ -538,7 +541,7 @@ instance RegisterInfo (ArchReg arch) => Show (Value arch ids tp) where
|
|||||||
show = show . pretty
|
show = show . pretty
|
||||||
|
|
||||||
-- | Typeclass for architecture-specific functions
|
-- | Typeclass for architecture-specific functions
|
||||||
class IsArchFn (f :: (Type -> *) -> Type -> *) where
|
class IsArchFn (f :: (Type -> Kind.Type) -> Type -> Kind.Type) where
|
||||||
-- | A function for pretty printing an archFn of a given type.
|
-- | A function for pretty printing an archFn of a given type.
|
||||||
ppArchFn :: Applicative m
|
ppArchFn :: Applicative m
|
||||||
=> (forall u . v u -> m Doc)
|
=> (forall u . v u -> m Doc)
|
||||||
@ -547,7 +550,7 @@ class IsArchFn (f :: (Type -> *) -> Type -> *) where
|
|||||||
-> m Doc
|
-> m Doc
|
||||||
|
|
||||||
-- | Typeclass for architecture-specific statements
|
-- | Typeclass for architecture-specific statements
|
||||||
class IsArchStmt (f :: (Type -> *) -> *) where
|
class IsArchStmt (f :: (Type -> Kind.Type) -> Kind.Type) where
|
||||||
-- | A function for pretty printing an architecture statement of a given type.
|
-- | A function for pretty printing an architecture statement of a given type.
|
||||||
ppArchStmt :: (forall u . v u -> Doc)
|
ppArchStmt :: (forall u . v u -> Doc)
|
||||||
-- ^ Function for pretty printing value.
|
-- ^ Function for pretty printing value.
|
||||||
@ -641,7 +644,7 @@ ppValueAssignmentList vals =
|
|||||||
|
|
||||||
-- | This class provides a way of optionally pretty printing the contents
|
-- | This class provides a way of optionally pretty printing the contents
|
||||||
-- of a register or omitting them.
|
-- of a register or omitting them.
|
||||||
class PrettyRegValue r (f :: Type -> *) where
|
class PrettyRegValue r (f :: Type -> Kind.Type) where
|
||||||
-- | ppValueEq should return a doc if the contents of the given register
|
-- | ppValueEq should return a doc if the contents of the given register
|
||||||
-- should be printed, and Nothing if the contents should be ignored.
|
-- should be printed, and Nothing if the contents should be ignored.
|
||||||
ppValueEq :: r tp -> f tp -> Maybe Doc
|
ppValueEq :: r tp -> f tp -> Maybe Doc
|
||||||
|
@ -125,7 +125,7 @@ appendRewrittenStmt stmt = Rewriter $ do
|
|||||||
stmts <- use rwRevStmts
|
stmts <- use rwRevStmts
|
||||||
let stmts' = stmt : stmts
|
let stmts' = stmt : stmts
|
||||||
seq stmt $ seq stmts' $ do
|
seq stmt $ seq stmts' $ do
|
||||||
rwRevStmts .= stmts'
|
rwRevStmts .= stmts'
|
||||||
|
|
||||||
-- | Add a statment to the list
|
-- | Add a statment to the list
|
||||||
appendRewrittenArchStmt :: ArchStmt arch (Value arch tgt) -> Rewriter arch s src tgt ()
|
appendRewrittenArchStmt :: ArchStmt arch (Value arch tgt) -> Rewriter arch s src tgt ()
|
||||||
@ -151,10 +151,10 @@ addBinding :: AssignId src tp -> Value arch tgt tp -> Rewriter arch s src tgt ()
|
|||||||
addBinding srcId val = Rewriter $ do
|
addBinding srcId val = Rewriter $ do
|
||||||
ref <- gets $ rwctxCache . rwContext
|
ref <- gets $ rwctxCache . rwContext
|
||||||
lift $ do
|
lift $ do
|
||||||
m <- readSTRef ref
|
m <- readSTRef ref
|
||||||
when (MapF.member srcId m) $ do
|
when (MapF.member srcId m) $ do
|
||||||
fail $ "Assignment " ++ show srcId ++ " is already bound."
|
fail $ "Assignment " ++ show srcId ++ " is already bound."
|
||||||
writeSTRef ref $! MapF.insert srcId val m
|
writeSTRef ref $! MapF.insert srcId val m
|
||||||
|
|
||||||
-- | Return true if values are identical
|
-- | Return true if values are identical
|
||||||
identValue :: TestEquality (ArchReg arch) => Value arch tgt tp -> Value arch tgt tp -> Bool
|
identValue :: TestEquality (ArchReg arch) => Value arch tgt tp -> Value arch tgt tp -> Bool
|
||||||
@ -171,7 +171,7 @@ rewriteApp :: App (Value arch tgt) tp -> Rewriter arch s src tgt (Value arch tgt
|
|||||||
rewriteApp app = do
|
rewriteApp app = do
|
||||||
ctx <- Rewriter $ gets rwContext
|
ctx <- Rewriter $ gets rwContext
|
||||||
rwctxConstraints ctx $ do
|
rwctxConstraints ctx $ do
|
||||||
case app of
|
case app of
|
||||||
|
|
||||||
Trunc (BVValue _ x) w -> do
|
Trunc (BVValue _ x) w -> do
|
||||||
pure $ BVValue w $ toUnsigned w x
|
pure $ BVValue w $ toUnsigned w x
|
||||||
|
@ -423,13 +423,13 @@ mergeIntraJump :: ArchSegmentOff arch
|
|||||||
mergeIntraJump src ab tgt = do
|
mergeIntraJump src ab tgt = do
|
||||||
info <- uses curFunCtx archInfo
|
info <- uses curFunCtx archInfo
|
||||||
withArchConstraints info $ do
|
withArchConstraints info $ do
|
||||||
when (not (absStackHasReturnAddr ab)) $ do
|
when (not (absStackHasReturnAddr ab)) $ do
|
||||||
debug DCFG ("WARNING: Missing return value in jump from " ++ show src ++ " to\n" ++ show ab) $
|
debug DCFG ("WARNING: Missing return value in jump from " ++ show src ++ " to\n" ++ show ab) $
|
||||||
pure ()
|
pure ()
|
||||||
let rsn = NextIP src
|
let rsn = NextIP src
|
||||||
-- Associate a new abstract state with the code region.
|
-- Associate a new abstract state with the code region.
|
||||||
foundMap <- use foundAddrs
|
foundMap <- use foundAddrs
|
||||||
case Map.lookup tgt foundMap of
|
case Map.lookup tgt foundMap of
|
||||||
-- We have seen this block before, so need to join and see if
|
-- We have seen this block before, so need to join and see if
|
||||||
-- the results is changed.
|
-- the results is changed.
|
||||||
Just old_info -> do
|
Just old_info -> do
|
||||||
@ -776,8 +776,8 @@ recordWriteStmt arch_info mem regs stmt = do
|
|||||||
WriteMem _addr repr v
|
WriteMem _addr repr v
|
||||||
| Just Refl <- testEquality repr (addrMemRepr arch_info) -> do
|
| Just Refl <- testEquality repr (addrMemRepr arch_info) -> do
|
||||||
withArchConstraints arch_info $ do
|
withArchConstraints arch_info $ do
|
||||||
let addrs = identifyConcreteAddresses mem (transferValue regs v)
|
let addrs = identifyConcreteAddresses mem (transferValue regs v)
|
||||||
writtenCodeAddrs %= (filter isExecutableSegOff addrs ++)
|
writtenCodeAddrs %= (filter isExecutableSegOff addrs ++)
|
||||||
_ -> return ()
|
_ -> return ()
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
@ -919,10 +919,8 @@ parseFetchAndExecute ctx idx initRegs stmts absProcState finalRegs = do
|
|||||||
let ainfo = pctxArchInfo ctx
|
let ainfo = pctxArchInfo ctx
|
||||||
let absProcState' = absEvalStmts ainfo absProcState stmts
|
let absProcState' = absEvalStmts ainfo absProcState stmts
|
||||||
withArchConstraints ainfo $ do
|
withArchConstraints ainfo $ do
|
||||||
-- See if next statement appears to end with a call.
|
-- Try to figure out what control flow statement we have.
|
||||||
-- We define calls as statements that end with a write that
|
case () of
|
||||||
-- stores the pc to an address.
|
|
||||||
case () of
|
|
||||||
-- The block ends with a Mux, so we turn this into a `ParsedIte` statement.
|
-- The block ends with a Mux, so we turn this into a `ParsedIte` statement.
|
||||||
_ | Just (Mux _ c t f) <- valueAsApp (finalRegs^.boundValue ip_reg) -> do
|
_ | Just (Mux _ c t f) <- valueAsApp (finalRegs^.boundValue ip_reg) -> do
|
||||||
mapM_ (recordWriteStmt ainfo mem absProcState') stmts
|
mapM_ (recordWriteStmt ainfo mem absProcState') stmts
|
||||||
@ -955,9 +953,8 @@ parseFetchAndExecute ctx idx initRegs stmts absProcState finalRegs = do
|
|||||||
_ | Just (prev_stmts, ret) <- identifyCall ainfo mem stmts finalRegs -> do
|
_ | Just (prev_stmts, ret) <- identifyCall ainfo mem stmts finalRegs -> do
|
||||||
mapM_ (recordWriteStmt ainfo mem absProcState') prev_stmts
|
mapM_ (recordWriteStmt ainfo mem absProcState') prev_stmts
|
||||||
let abst = finalAbsBlockState absProcState' finalRegs
|
let abst = finalAbsBlockState absProcState' finalRegs
|
||||||
seq abst $ do
|
|
||||||
-- Merge caller return information
|
-- Merge caller return information
|
||||||
intraJumpTargets %= ((ret, postCallAbsState ainfo abst ret):)
|
seq abst $ intraJumpTargets %= ((ret, postCallAbsState ainfo abst ret):)
|
||||||
-- Use the abstract domain to look for new code pointers for the current IP.
|
-- Use the abstract domain to look for new code pointers for the current IP.
|
||||||
addNewFunctionAddrs $
|
addNewFunctionAddrs $
|
||||||
identifyCallTargets mem abst finalRegs
|
identifyCallTargets mem abst finalRegs
|
||||||
@ -1090,16 +1087,25 @@ parseFetchAndExecute ctx idx initRegs stmts absProcState finalRegs = do
|
|||||||
let abst = finalAbsBlockState absProcState' finalRegs
|
let abst = finalAbsBlockState absProcState' finalRegs
|
||||||
seq abst $ do
|
seq abst $ do
|
||||||
|
|
||||||
-- Look for new instruction pointers
|
-- Look for new instruction pointers
|
||||||
addNewFunctionAddrs $
|
addNewFunctionAddrs $
|
||||||
identifyConcreteAddresses mem (abst^.absRegState^.curIP)
|
identifyConcreteAddresses mem (abst^.absRegState^.curIP)
|
||||||
|
|
||||||
|
<<<<<<< HEAD
|
||||||
let ret = StatementList { stmtsIdent = idx
|
let ret = StatementList { stmtsIdent = idx
|
||||||
, stmtsNonterm = toList stmts
|
, stmtsNonterm = toList stmts
|
||||||
, stmtsTerm = ParsedCall finalRegs Nothing
|
, stmtsTerm = ParsedCall finalRegs Nothing
|
||||||
, stmtsAbsState = absProcState'
|
, stmtsAbsState = absProcState'
|
||||||
}
|
}
|
||||||
seq ret $ pure (ret,idx+1)
|
seq ret $ pure (ret,idx+1)
|
||||||
|
=======
|
||||||
|
let ret = StatementList { stmtsIdent = idx
|
||||||
|
, stmtsNonterm = stmts
|
||||||
|
, stmtsTerm = ParsedCall s Nothing
|
||||||
|
, stmtsAbsState = absProcState'
|
||||||
|
}
|
||||||
|
seq ret $ pure (ret,idx+1)
|
||||||
|
>>>>>>> public/master
|
||||||
|
|
||||||
-- | this evalutes the statements in a block to expand the information known
|
-- | this evalutes the statements in a block to expand the information known
|
||||||
-- about control flow targets of this block.
|
-- about control flow targets of this block.
|
||||||
@ -1118,7 +1124,7 @@ parseBlock ctx idx initRegs b absProcState = do
|
|||||||
let mem = pctxMemory ctx
|
let mem = pctxMemory ctx
|
||||||
let ainfo = pctxArchInfo ctx
|
let ainfo = pctxArchInfo ctx
|
||||||
withArchConstraints ainfo $ do
|
withArchConstraints ainfo $ do
|
||||||
case blockTerm b of
|
case blockTerm b of
|
||||||
Branch c lb rb -> do
|
Branch c lb rb -> do
|
||||||
let blockMap = pctxBlockMap ctx
|
let blockMap = pctxBlockMap ctx
|
||||||
-- FIXME: we should propagate c back to the initial block, not just b
|
-- FIXME: we should propagate c back to the initial block, not just b
|
||||||
@ -1246,6 +1252,7 @@ transfer addr = do
|
|||||||
s <- use curFunCtx
|
s <- use curFunCtx
|
||||||
let ainfo = archInfo s
|
let ainfo = archInfo s
|
||||||
withArchConstraints ainfo $ do
|
withArchConstraints ainfo $ do
|
||||||
|
<<<<<<< HEAD
|
||||||
mfinfo <- use $ foundAddrs . at addr
|
mfinfo <- use $ foundAddrs . at addr
|
||||||
let finfo = fromMaybe (error $ "transfer called on unfound address " ++ show addr ++ ".") $
|
let finfo = fromMaybe (error $ "transfer called on unfound address " ++ show addr ++ ".") $
|
||||||
mfinfo
|
mfinfo
|
||||||
@ -1284,6 +1291,58 @@ transfer addr = do
|
|||||||
-- Call transfer blocks to calculate parsedblocks
|
-- Call transfer blocks to calculate parsedblocks
|
||||||
let blockMap = Map.fromList [ (blockLabel b, b) | b <- bs ]
|
let blockMap = Map.fromList [ (blockLabel b, b) | b <- bs ]
|
||||||
addBlocks addr finfo initRegs sz blockMap
|
addBlocks addr finfo initRegs sz blockMap
|
||||||
|
=======
|
||||||
|
mfinfo <- use $ foundAddrs . at addr
|
||||||
|
let finfo = fromMaybe (error $ "transfer called on unfound address " ++ show addr ++ ".") $
|
||||||
|
mfinfo
|
||||||
|
let mem = memory s
|
||||||
|
nonceGen <- gets funNonceGen
|
||||||
|
prev_block_map <- use $ curFunBlocks
|
||||||
|
-- Get maximum number of bytes to disassemble
|
||||||
|
let maxSize :: Int
|
||||||
|
maxSize =
|
||||||
|
case Map.lookupGT addr prev_block_map of
|
||||||
|
Just (next,_) | Just o <- diffSegmentOff next addr -> fromInteger o
|
||||||
|
_ -> fromInteger (segoffBytesLeft addr)
|
||||||
|
let ab = foundAbstractState finfo
|
||||||
|
(bs0, sz, maybeError) <- liftST $ disassembleFn ainfo nonceGen addr maxSize ab
|
||||||
|
|
||||||
|
#ifdef USE_REWRITER
|
||||||
|
bs1 <- do
|
||||||
|
let archStmt = rewriteArchStmt ainfo
|
||||||
|
let secAddrMap = memSectionIndexMap mem
|
||||||
|
liftST $ do
|
||||||
|
ctx <- mkRewriteContext nonceGen (rewriteArchFn ainfo) archStmt secAddrMap
|
||||||
|
traverse (rewriteBlock ainfo ctx) bs0
|
||||||
|
#else
|
||||||
|
bs1 <- pure bs0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
-- If no blocks are returned, then we just add an empty parsed block.
|
||||||
|
if null bs1 then do
|
||||||
|
let errMsg = Text.pack $ fromMaybe "Unknown error" maybeError
|
||||||
|
let stmts = StatementList
|
||||||
|
{ stmtsIdent = 0
|
||||||
|
, stmtsNonterm = []
|
||||||
|
, stmtsTerm = ParsedTranslateError errMsg
|
||||||
|
, stmtsAbsState = initAbsProcessorState mem (foundAbstractState finfo)
|
||||||
|
}
|
||||||
|
let pb = ParsedBlock { pblockAddr = addr
|
||||||
|
, blockSize = sz
|
||||||
|
, blockReason = foundReason finfo
|
||||||
|
, blockAbstractState = foundAbstractState finfo
|
||||||
|
, blockStatementList = stmts
|
||||||
|
}
|
||||||
|
id %= addFunBlock addr pb
|
||||||
|
else do
|
||||||
|
-- Rewrite returned blocks to simplify expressions
|
||||||
|
|
||||||
|
-- Compute demand set
|
||||||
|
let bs = bs1 -- eliminateDeadStmts ainfo bs1
|
||||||
|
-- Call transfer blocks to calculate parsedblocks
|
||||||
|
let blockMap = Map.fromList [ (blockLabel b, b) | b <- bs ]
|
||||||
|
addBlocks addr finfo sz blockMap
|
||||||
|
>>>>>>> public/master
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- Main loop
|
-- Main loop
|
||||||
|
@ -160,6 +160,8 @@ import Data.Parameterized.NatRepr
|
|||||||
|
|
||||||
import qualified Data.Macaw.Memory.Permissions as Perm
|
import qualified Data.Macaw.Memory.Permissions as Perm
|
||||||
|
|
||||||
|
import Prelude
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- AddrWidthRepr
|
-- AddrWidthRepr
|
||||||
|
|
||||||
|
@ -72,7 +72,7 @@ import qualified Data.IntervalMap.Strict as IMap
|
|||||||
import Data.Map.Strict (Map)
|
import Data.Map.Strict (Map)
|
||||||
import qualified Data.Map.Strict as Map
|
import qualified Data.Map.Strict as Map
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Monoid
|
import Data.Semigroup
|
||||||
import Data.Set (Set)
|
import Data.Set (Set)
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
import qualified Data.Vector as V
|
import qualified Data.Vector as V
|
||||||
@ -85,6 +85,9 @@ import Data.Macaw.Memory.LoadCommon
|
|||||||
import qualified Data.Macaw.Memory.Permissions as Perm
|
import qualified Data.Macaw.Memory.Permissions as Perm
|
||||||
import Data.Macaw.Memory.Symbols
|
import Data.Macaw.Memory.Symbols
|
||||||
|
|
||||||
|
import Prelude
|
||||||
|
|
||||||
|
|
||||||
-- | Return a subrange of a bytestring.
|
-- | Return a subrange of a bytestring.
|
||||||
sliceL :: Integral w => Elf.Range w -> L.ByteString -> L.ByteString
|
sliceL :: Integral w => Elf.Range w -> L.ByteString -> L.ByteString
|
||||||
sliceL (i,c) = L.take (fromIntegral c) . L.drop (fromIntegral i)
|
sliceL (i,c) = L.take (fromIntegral c) . L.drop (fromIntegral i)
|
||||||
@ -918,8 +921,8 @@ insertElfSegment :: RegionIndex
|
|||||||
-> MemLoader w ()
|
-> MemLoader w ()
|
||||||
insertElfSegment regIdx addrOff shdrMap contents relocMap phdr = do
|
insertElfSegment regIdx addrOff shdrMap contents relocMap phdr = do
|
||||||
w <- uses mlsMemory memAddrWidth
|
w <- uses mlsMemory memAddrWidth
|
||||||
reprConstraints w $ do
|
reprConstraints w $
|
||||||
when (Elf.phdrMemSize phdr > 0) $ do
|
when (Elf.phdrMemSize phdr > 0) $ do
|
||||||
let segIdx = Elf.phdrSegmentIndex phdr
|
let segIdx = Elf.phdrSegmentIndex phdr
|
||||||
seg <- do
|
seg <- do
|
||||||
let linkBaseOff = fromIntegral (Elf.phdrSegmentVirtAddr phdr)
|
let linkBaseOff = fromIntegral (Elf.phdrSegmentVirtAddr phdr)
|
||||||
@ -962,21 +965,21 @@ memoryForElfSegments regIndex addrOff e = do
|
|||||||
let hdr = Elf.elfLayoutHeader l
|
let hdr = Elf.elfLayoutHeader l
|
||||||
let w = elfAddrWidth (elfClass e)
|
let w = elfAddrWidth (elfClass e)
|
||||||
reprConstraints w $ do
|
reprConstraints w $ do
|
||||||
let ph = Elf.allPhdrs l
|
let ph = Elf.allPhdrs l
|
||||||
let contents = elfLayoutBytes l
|
let contents = elfLayoutBytes l
|
||||||
-- Create relocation map
|
-- Create relocation map
|
||||||
relocMap <- dynamicRelocationMap hdr ph contents
|
relocMap <- dynamicRelocationMap hdr ph contents
|
||||||
|
|
||||||
let intervals :: ElfFileSectionMap (ElfWordType w)
|
let intervals :: ElfFileSectionMap (ElfWordType w)
|
||||||
intervals = IMap.fromList
|
intervals = IMap.fromList
|
||||||
[ (IntervalCO start end, sec)
|
[ (IntervalCO start end, sec)
|
||||||
| shdr <- Map.elems (l ^. Elf.shdrs)
|
| shdr <- Map.elems (l ^. Elf.shdrs)
|
||||||
, let start = shdr^._3
|
, let start = shdr^._3
|
||||||
, let sec = shdr^._1
|
, let sec = shdr^._1
|
||||||
, let end = start + elfSectionFileSize sec
|
, let end = start + elfSectionFileSize sec
|
||||||
]
|
]
|
||||||
mapM_ (insertElfSegment regIndex addrOff intervals contents relocMap)
|
mapM_ (insertElfSegment regIndex addrOff intervals contents relocMap)
|
||||||
(filter (\p -> Elf.phdrSegmentType p == Elf.PT_LOAD) ph)
|
(filter (\p -> Elf.phdrSegmentType p == Elf.PT_LOAD) ph)
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- Elf section loading
|
-- Elf section loading
|
||||||
@ -1038,8 +1041,8 @@ insertAllocatedSection :: Elf.ElfHeader w
|
|||||||
insertAllocatedSection hdr symtab sectionMap regIdx nm = do
|
insertAllocatedSection hdr symtab sectionMap regIdx nm = do
|
||||||
w <- uses mlsMemory memAddrWidth
|
w <- uses mlsMemory memAddrWidth
|
||||||
reprConstraints w $ do
|
reprConstraints w $ do
|
||||||
msec <- findSection sectionMap nm
|
msec <- findSection sectionMap nm
|
||||||
case msec of
|
case msec of
|
||||||
Nothing -> pure ()
|
Nothing -> pure ()
|
||||||
Just sec -> do
|
Just sec -> do
|
||||||
mRelBuffer <- fmap (fmap (L.fromStrict . elfSectionData)) $
|
mRelBuffer <- fmap (fmap (L.fromStrict . elfSectionData)) $
|
||||||
|
@ -29,6 +29,7 @@ module Data.Macaw.Types
|
|||||||
, Data.Parameterized.NatRepr.knownNat
|
, Data.Parameterized.NatRepr.knownNat
|
||||||
) where
|
) where
|
||||||
|
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Parameterized.Classes
|
import Data.Parameterized.Classes
|
||||||
import qualified Data.Parameterized.List as P
|
import qualified Data.Parameterized.List as P
|
||||||
import Data.Parameterized.NatRepr
|
import Data.Parameterized.NatRepr
|
||||||
@ -251,7 +252,7 @@ instance OrdF FloatInfoRepr where
|
|||||||
|
|
||||||
-- | A multi-parameter type class that allows one to represent that a
|
-- | A multi-parameter type class that allows one to represent that a
|
||||||
-- parameterized type value has some representative type such as a TypeRepr.
|
-- parameterized type value has some representative type such as a TypeRepr.
|
||||||
class HasRepr (f :: k -> *) (v :: k -> *) | f -> v where
|
class HasRepr (f :: k -> Kind.Type) (v :: k -> Kind.Type) | f -> v where
|
||||||
typeRepr :: f tp -> v tp
|
typeRepr :: f tp -> v tp
|
||||||
|
|
||||||
typeWidth :: HasRepr f TypeRepr => f (BVType w) -> NatRepr w
|
typeWidth :: HasRepr f TypeRepr => f (BVType w) -> NatRepr w
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
{-# LANGUAGE CPP #-}
|
{-# LANGUAGE CPP #-}
|
||||||
|
{-# LANGUAGE ConstraintKinds #-}
|
||||||
{-# LANGUAGE DataKinds #-}
|
{-# LANGUAGE DataKinds #-}
|
||||||
{-# LANGUAGE EmptyCase #-}
|
{-# LANGUAGE EmptyCase #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
@ -43,10 +44,15 @@ module Data.Macaw.Symbolic
|
|||||||
, Regs
|
, Regs
|
||||||
, freshValue
|
, freshValue
|
||||||
, GlobalMap
|
, GlobalMap
|
||||||
|
-- * Symbolic architecture-specific types
|
||||||
|
, ArchBits
|
||||||
|
, ArchInfo(..)
|
||||||
|
, ArchVals(..)
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Control.Lens ((^.))
|
import Control.Lens ((^.))
|
||||||
import Control.Monad (forM, join)
|
import Control.Monad (forM, join)
|
||||||
|
import Control.Monad.IO.Class
|
||||||
import Control.Monad.ST (ST, RealWorld, stToIO)
|
import Control.Monad.ST (ST, RealWorld, stToIO)
|
||||||
import Data.Foldable
|
import Data.Foldable
|
||||||
import Data.Map.Strict (Map)
|
import Data.Map.Strict (Map)
|
||||||
@ -477,13 +483,17 @@ type MacawArchEvalFn sym arch =
|
|||||||
|
|
||||||
|
|
||||||
-- | This evaluates a Macaw statement extension in the simulator.
|
-- | This evaluates a Macaw statement extension in the simulator.
|
||||||
execMacawStmtExtension ::
|
execMacawStmtExtension
|
||||||
IsSymInterface sym =>
|
:: IsSymInterface sym
|
||||||
MacawArchEvalFn sym arch {- ^ Function for executing -} ->
|
=> ( C.GlobalVar MM.Mem
|
||||||
C.GlobalVar MM.Mem ->
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
GlobalMap sym (M.ArchAddrWidth arch) ->
|
-> MacawArchEvalFn sym arch
|
||||||
LookupFunctionHandle sym arch ->
|
)
|
||||||
EvalStmtFunc (MacawStmtExtension arch) (MacawSimulatorState sym) sym (MacawExt arch)
|
{- ^ Function for executing -}
|
||||||
|
-> C.GlobalVar MM.Mem
|
||||||
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
|
-> LookupFunctionHandle sym arch
|
||||||
|
-> EvalStmtFunc (MacawStmtExtension arch) (MacawSimulatorState sym) sym (MacawExt arch)
|
||||||
execMacawStmtExtension archStmtFn mvar globs (LFH lookupH) s0 st =
|
execMacawStmtExtension archStmtFn mvar globs (LFH lookupH) s0 st =
|
||||||
case s0 of
|
case s0 of
|
||||||
MacawReadMem w mr x -> doReadMem st mvar globs w mr x
|
MacawReadMem w mr x -> doReadMem st mvar globs w mr x
|
||||||
@ -506,7 +516,7 @@ execMacawStmtExtension archStmtFn mvar globs (LFH lookupH) s0 st =
|
|||||||
(hv, st') <- doLookupFunctionHandle lookupH st mvar (C.regValue args)
|
(hv, st') <- doLookupFunctionHandle lookupH st mvar (C.regValue args)
|
||||||
return (C.HandleFnVal hv, st')
|
return (C.HandleFnVal hv, st')
|
||||||
|
|
||||||
MacawArchStmtExtension s -> archStmtFn s st
|
MacawArchStmtExtension s -> archStmtFn mvar globs s st
|
||||||
MacawArchStateUpdate {} -> return ((), st)
|
MacawArchStateUpdate {} -> return ((), st)
|
||||||
|
|
||||||
PtrEq w x y -> doPtrEq st mvar w x y
|
PtrEq w x y -> doPtrEq st mvar w x y
|
||||||
@ -568,37 +578,84 @@ freshValue sym str w ty =
|
|||||||
|
|
||||||
|
|
||||||
-- | Return macaw extension evaluation functions.
|
-- | Return macaw extension evaluation functions.
|
||||||
macawExtensions ::
|
macawExtensions
|
||||||
IsSymInterface sym =>
|
:: IsSymInterface sym
|
||||||
MacawArchEvalFn sym arch ->
|
=> ( C.GlobalVar MM.Mem
|
||||||
C.GlobalVar MM.Mem ->
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
GlobalMap sym (M.ArchAddrWidth arch) ->
|
-> MacawArchEvalFn sym arch
|
||||||
LookupFunctionHandle sym arch ->
|
)
|
||||||
C.ExtensionImpl (MacawSimulatorState sym) sym (MacawExt arch)
|
-> C.GlobalVar MM.Mem
|
||||||
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
|
-> LookupFunctionHandle sym arch
|
||||||
|
-> C.ExtensionImpl (MacawSimulatorState sym) sym (MacawExt arch)
|
||||||
macawExtensions f mvar globs lookupH =
|
macawExtensions f mvar globs lookupH =
|
||||||
C.ExtensionImpl { C.extensionEval = evalMacawExprExtension
|
C.ExtensionImpl { C.extensionEval = evalMacawExprExtension
|
||||||
, C.extensionExec = execMacawStmtExtension f mvar globs lookupH
|
, C.extensionExec = execMacawStmtExtension f mvar globs lookupH
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ArchBits arch =
|
||||||
|
( C.IsSyntaxExtension (MacawExt arch)
|
||||||
|
, M.ArchConstraints arch
|
||||||
|
, M.RegisterInfo (M.ArchReg arch)
|
||||||
|
, M.HasRepr (M.ArchReg arch) M.TypeRepr
|
||||||
|
, M.MemWidth (M.ArchAddrWidth arch)
|
||||||
|
, Show (M.ArchReg arch (M.BVType (M.ArchAddrWidth arch)))
|
||||||
|
, ArchInfo arch
|
||||||
|
)
|
||||||
|
|
||||||
|
type SymArchConstraints arch =
|
||||||
|
( C.IsSyntaxExtension (MacawExt arch)
|
||||||
|
, M.MemWidth (M.ArchAddrWidth arch)
|
||||||
|
, M.PrettyF (M.ArchReg arch)
|
||||||
|
)
|
||||||
|
|
||||||
|
data ArchVals arch = ArchVals
|
||||||
|
{ archFunctions :: MacawSymbolicArchFunctions arch
|
||||||
|
, withArchEval
|
||||||
|
:: forall a m sym
|
||||||
|
. (IsSymInterface sym, MonadIO m)
|
||||||
|
=> sym
|
||||||
|
-> ( ( C.GlobalVar MM.Mem
|
||||||
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
|
-> MacawArchEvalFn sym arch
|
||||||
|
)
|
||||||
|
-> m a
|
||||||
|
)
|
||||||
|
-> m a
|
||||||
|
, withArchConstraints :: forall a . (SymArchConstraints arch => a) -> a
|
||||||
|
}
|
||||||
|
|
||||||
|
-- | A class to capture the architecture-specific information required to
|
||||||
|
-- perform block recovery and translation into a Crucible CFG.
|
||||||
|
--
|
||||||
|
-- For architectures that do not have a symbolic backend yet, have this function
|
||||||
|
-- return 'Nothing'.
|
||||||
|
class ArchInfo arch where
|
||||||
|
archVals :: proxy arch -> Maybe (ArchVals arch)
|
||||||
|
|
||||||
-- | Run the simulator over a contiguous set of code.
|
-- | Run the simulator over a contiguous set of code.
|
||||||
runCodeBlock :: forall sym arch blocks
|
runCodeBlock
|
||||||
. (C.IsSyntaxExtension (MacawExt arch), IsSymInterface sym)
|
:: forall sym arch blocks
|
||||||
=> sym
|
. (C.IsSyntaxExtension (MacawExt arch), IsSymInterface sym)
|
||||||
-> MacawSymbolicArchFunctions arch
|
=> sym
|
||||||
-- ^ Translation functions
|
-> MacawSymbolicArchFunctions arch
|
||||||
-> MacawArchEvalFn sym arch
|
-- ^ Translation functions
|
||||||
-> C.HandleAllocator RealWorld
|
-> ( C.GlobalVar MM.Mem
|
||||||
-> (MM.MemImpl sym, GlobalMap sym (M.ArchAddrWidth arch))
|
-> GlobalMap sym (M.ArchAddrWidth arch)
|
||||||
-> LookupFunctionHandle sym arch
|
-> MacawArchEvalFn sym arch
|
||||||
-> C.CFG (MacawExt arch) blocks (EmptyCtx ::> ArchRegStruct arch) (ArchRegStruct arch)
|
)
|
||||||
-> Ctx.Assignment (C.RegValue' sym) (MacawCrucibleRegTypes arch)
|
-> C.HandleAllocator RealWorld
|
||||||
-- ^ Register assignment
|
-> (MM.MemImpl sym, GlobalMap sym (M.ArchAddrWidth arch))
|
||||||
-> IO ( C.GlobalVar MM.Mem
|
-> LookupFunctionHandle sym arch
|
||||||
, C.ExecResult
|
-> C.CFG (MacawExt arch) blocks (EmptyCtx ::> ArchRegStruct arch) (ArchRegStruct arch)
|
||||||
(MacawSimulatorState sym)
|
-> Ctx.Assignment (C.RegValue' sym) (MacawCrucibleRegTypes arch)
|
||||||
sym
|
-- ^ Register assignment
|
||||||
(MacawExt arch)
|
-> IO ( C.GlobalVar MM.Mem
|
||||||
(C.RegEntry sym (ArchRegStruct arch)))
|
, C.ExecResult
|
||||||
|
(MacawSimulatorState sym)
|
||||||
|
sym
|
||||||
|
(MacawExt arch)
|
||||||
|
(C.RegEntry sym (ArchRegStruct arch)))
|
||||||
runCodeBlock sym archFns archEval halloc (initMem,globs) lookupH g regStruct = do
|
runCodeBlock sym archFns archEval halloc (initMem,globs) lookupH g regStruct = do
|
||||||
mvar <- stToIO (MM.mkMemVar halloc)
|
mvar <- stToIO (MM.mkMemVar halloc)
|
||||||
let crucRegTypes = crucArchRegTypes archFns
|
let crucRegTypes = crucArchRegTypes archFns
|
||||||
|
@ -46,6 +46,8 @@ library
|
|||||||
ghc-options: -Wall
|
ghc-options: -Wall
|
||||||
ghc-options: -fno-warn-unticked-promoted-constructors
|
ghc-options: -fno-warn-unticked-promoted-constructors
|
||||||
ghc-prof-options: -O2 -fprof-auto-top
|
ghc-prof-options: -O2 -fprof-auto-top
|
||||||
|
if impl(ghc >= 8.6)
|
||||||
|
default-extensions: NoStarIsType
|
||||||
|
|
||||||
test-suite macaw-x86-tests
|
test-suite macaw-x86-tests
|
||||||
type: exitcode-stdio-1.0
|
type: exitcode-stdio-1.0
|
||||||
|
@ -33,11 +33,13 @@ module Data.Macaw.X86.ArchTypes
|
|||||||
, X86PrimLoc(..)
|
, X86PrimLoc(..)
|
||||||
, SIMDWidth(..)
|
, SIMDWidth(..)
|
||||||
, RepValSize(..)
|
, RepValSize(..)
|
||||||
|
, SomeRepValSize(..)
|
||||||
, repValSizeByteCount
|
, repValSizeByteCount
|
||||||
, repValSizeMemRepr
|
, repValSizeMemRepr
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Data.Bits
|
import Data.Bits
|
||||||
|
import qualified Data.Kind as Kind
|
||||||
import Data.Word(Word8)
|
import Data.Word(Word8)
|
||||||
import Data.Macaw.CFG
|
import Data.Macaw.CFG
|
||||||
import Data.Macaw.CFG.Rewriter
|
import Data.Macaw.CFG.Rewriter
|
||||||
@ -80,6 +82,9 @@ data RepValSize w
|
|||||||
| (w ~ 32) => DWordRepVal
|
| (w ~ 32) => DWordRepVal
|
||||||
| (w ~ 64) => QWordRepVal
|
| (w ~ 64) => QWordRepVal
|
||||||
|
|
||||||
|
data SomeRepValSize where
|
||||||
|
SomeRepValSize :: (1 <= w) => RepValSize w -> SomeRepValSize
|
||||||
|
|
||||||
repValSizeMemRepr :: RepValSize w -> MemRepr (BVType w)
|
repValSizeMemRepr :: RepValSize w -> MemRepr (BVType w)
|
||||||
repValSizeMemRepr v =
|
repValSizeMemRepr v =
|
||||||
case v of
|
case v of
|
||||||
@ -121,7 +126,7 @@ data X86PrimLoc tp
|
|||||||
| (tp ~ BVType 16) => FS
|
| (tp ~ BVType 16) => FS
|
||||||
-- ^ This refers to the selector of the 'FS' register.
|
-- ^ This refers to the selector of the 'FS' register.
|
||||||
| (tp ~ BVType 16) => GS
|
| (tp ~ BVType 16) => GS
|
||||||
-- ^ This refers to the se lector of the 'GS' register.
|
-- ^ This refers to the selector of the 'GS' register.
|
||||||
| forall w . (tp ~ BVType w) => X87_ControlLoc !(X87_ControlReg w)
|
| forall w . (tp ~ BVType w) => X87_ControlLoc !(X87_ControlReg w)
|
||||||
-- ^ One of the x87 control registers
|
-- ^ One of the x87 control registers
|
||||||
|
|
||||||
@ -877,7 +882,7 @@ x86PrimFnHasSideEffects f =
|
|||||||
-- X86Stmt
|
-- X86Stmt
|
||||||
|
|
||||||
-- | An X86 specific statement.
|
-- | An X86 specific statement.
|
||||||
data X86Stmt (v :: Type -> *) where
|
data X86Stmt (v :: Type -> Kind.Type) where
|
||||||
WriteLoc :: !(X86PrimLoc tp) -> !(v tp) -> X86Stmt v
|
WriteLoc :: !(X86PrimLoc tp) -> !(v tp) -> X86Stmt v
|
||||||
|
|
||||||
-- | Store the X87 control register in the given address.
|
-- | Store the X87 control register in the given address.
|
||||||
@ -894,12 +899,14 @@ data X86Stmt (v :: Type -> *) where
|
|||||||
-- * @dir@ is a flag that indicates the direction of move ('True' ==
|
-- * @dir@ is a flag that indicates the direction of move ('True' ==
|
||||||
-- decrement, 'False' == increment) for updating the buffer
|
-- decrement, 'False' == increment) for updating the buffer
|
||||||
-- pointers.
|
-- pointers.
|
||||||
RepMovs :: !(RepValSize w)
|
RepMovs
|
||||||
-> !(v (BVType 64))
|
:: (1 <= w)
|
||||||
-> !(v (BVType 64))
|
=> !(RepValSize w)
|
||||||
-> !(v (BVType 64))
|
-> !(v (BVType 64))
|
||||||
-> !(v BoolType)
|
-> !(v (BVType 64))
|
||||||
-> X86Stmt v
|
-> !(v (BVType 64))
|
||||||
|
-> !(v BoolType)
|
||||||
|
-> X86Stmt v
|
||||||
|
|
||||||
-- | Assign all elements in an array in memory a specific value.
|
-- | Assign all elements in an array in memory a specific value.
|
||||||
--
|
--
|
||||||
@ -911,16 +918,18 @@ data X86Stmt (v :: Type -> *) where
|
|||||||
-- * @dir@ is a flag that indicates the direction of move ('True' ==
|
-- * @dir@ is a flag that indicates the direction of move ('True' ==
|
||||||
-- decrement, 'False' == increment) for updating the buffer
|
-- decrement, 'False' == increment) for updating the buffer
|
||||||
-- pointers.
|
-- pointers.
|
||||||
RepStos :: !(RepValSize w)
|
RepStos
|
||||||
-> !(v (BVType 64))
|
:: (1 <= w)
|
||||||
-- /\ Address to start assigning to.
|
=> !(RepValSize w)
|
||||||
-> !(v (BVType w))
|
-> !(v (BVType 64))
|
||||||
-- /\ Value to assign
|
-- /\ Address to start assigning to.
|
||||||
-> !(v (BVType 64))
|
-> !(v (BVType w))
|
||||||
-- /\ Number of values to assign
|
-- /\ Value to assign
|
||||||
-> !(v BoolType)
|
-> !(v (BVType 64))
|
||||||
-- /\ Direction flag
|
-- /\ Number of values to assign
|
||||||
-> X86Stmt v
|
-> !(v BoolType)
|
||||||
|
-- /\ Direction flag
|
||||||
|
-> X86Stmt v
|
||||||
|
|
||||||
-- | Empty MMX technology State. Sets the x87 FPU tag word to empty.
|
-- | Empty MMX technology State. Sets the x87 FPU tag word to empty.
|
||||||
--
|
--
|
||||||
|
@ -61,6 +61,7 @@ module Data.Macaw.X86.Generator
|
|||||||
import Control.Lens
|
import Control.Lens
|
||||||
import Control.Monad.Cont
|
import Control.Monad.Cont
|
||||||
import Control.Monad.Except
|
import Control.Monad.Except
|
||||||
|
import Control.Monad.Fail
|
||||||
import Control.Monad.Reader
|
import Control.Monad.Reader
|
||||||
import Control.Monad.ST
|
import Control.Monad.ST
|
||||||
import Control.Monad.State.Strict
|
import Control.Monad.State.Strict
|
||||||
@ -267,6 +268,8 @@ instance Monad (X86Generator st_s ids) where
|
|||||||
return v = seq v $ X86G $ return v
|
return v = seq v $ X86G $ return v
|
||||||
(X86G m) >>= h = X86G $ m >>= \v -> seq v (unX86G (h v))
|
(X86G m) >>= h = X86G $ m >>= \v -> seq v (unX86G (h v))
|
||||||
X86G m >> X86G n = X86G $ m >> n
|
X86G m >> X86G n = X86G $ m >> n
|
||||||
|
|
||||||
|
instance MonadFail (X86Generator st_s ids) where
|
||||||
fail msg = seq t $ X86G $ ContT $ \_ -> throwError t
|
fail msg = seq t $ X86G $ ContT $ \_ -> throwError t
|
||||||
where t = Text.pack msg
|
where t = Text.pack msg
|
||||||
|
|
||||||
|
@ -1223,13 +1223,12 @@ def_ret = defVariadic "ret" $ \_ vs ->
|
|||||||
def_movs :: InstructionDef
|
def_movs :: InstructionDef
|
||||||
def_movs = defBinary "movs" $ \ii loc _ -> do
|
def_movs = defBinary "movs" $ \ii loc _ -> do
|
||||||
let pfx = F.iiPrefixes ii
|
let pfx = F.iiPrefixes ii
|
||||||
Some w <-
|
SomeRepValSize w <- case loc of
|
||||||
case loc of
|
F.Mem8{} -> pure (SomeRepValSize ByteRepVal)
|
||||||
F.Mem8{} -> pure (Some ByteRepVal)
|
F.Mem16{} -> pure (SomeRepValSize WordRepVal)
|
||||||
F.Mem16{} -> pure (Some WordRepVal)
|
F.Mem32{} -> pure (SomeRepValSize DWordRepVal)
|
||||||
F.Mem32{} -> pure (Some DWordRepVal)
|
F.Mem64{} -> pure (SomeRepValSize QWordRepVal)
|
||||||
F.Mem64{} -> pure (Some QWordRepVal)
|
_ -> error "Bad argument to movs"
|
||||||
_ -> error "Bad argument to movs"
|
|
||||||
let bytesPerOp = bvLit n64 (repValSizeByteCount w)
|
let bytesPerOp = bvLit n64 (repValSizeByteCount w)
|
||||||
dest <- get rdi
|
dest <- get rdi
|
||||||
src <- get rsi
|
src <- get rsi
|
||||||
@ -1484,16 +1483,16 @@ def_lodsx suf elsz = defNullaryPrefix ("lods" ++ suf) $ \pfx -> do
|
|||||||
def_stos :: InstructionDef
|
def_stos :: InstructionDef
|
||||||
def_stos = defBinary "stos" $ \ii loc loc' -> do
|
def_stos = defBinary "stos" $ \ii loc loc' -> do
|
||||||
let pfx = F.iiPrefixes ii
|
let pfx = F.iiPrefixes ii
|
||||||
Some rep <-
|
SomeRepValSize rep <-
|
||||||
case (loc, loc') of
|
case (loc, loc') of
|
||||||
(F.Mem8 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.ByteReg F.AL) -> do
|
(F.Mem8 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.ByteReg F.AL) -> do
|
||||||
pure (Some ByteRepVal)
|
pure (SomeRepValSize ByteRepVal)
|
||||||
(F.Mem16 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.WordReg F.AX) -> do
|
(F.Mem16 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.WordReg F.AX) -> do
|
||||||
pure (Some WordRepVal)
|
pure (SomeRepValSize WordRepVal)
|
||||||
(F.Mem32 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.DWordReg F.EAX) -> do
|
(F.Mem32 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.DWordReg F.EAX) -> do
|
||||||
pure (Some DWordRepVal)
|
pure (SomeRepValSize DWordRepVal)
|
||||||
(F.Mem64 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.QWordReg F.RAX) -> do
|
(F.Mem64 (F.Addr_64 F.ES (Just F.RDI) Nothing F.NoDisplacement), F.QWordReg F.RAX) -> do
|
||||||
pure (Some QWordRepVal)
|
pure (SomeRepValSize QWordRepVal)
|
||||||
_ -> error $ "stos given bad arguments " ++ show (loc, loc')
|
_ -> error $ "stos given bad arguments " ++ show (loc, loc')
|
||||||
-- The direction flag indicates post decrement or post increment.
|
-- The direction flag indicates post decrement or post increment.
|
||||||
dest <- get rdi
|
dest <- get rdi
|
||||||
|
@ -3,12 +3,13 @@ version: 0.0.1
|
|||||||
author: Galois, Inc.
|
author: Galois, Inc.
|
||||||
maintainer: jhendrix@galois.com
|
maintainer: jhendrix@galois.com
|
||||||
build-type: Simple
|
build-type: Simple
|
||||||
cabal-version: >= 1.9.2
|
cabal-version: >= 1.10
|
||||||
license: BSD3
|
license: BSD3
|
||||||
license-file: LICENSE
|
license-file: LICENSE
|
||||||
|
|
||||||
library
|
library
|
||||||
build-depends: base >= 4,
|
build-depends: base >= 4,
|
||||||
|
ansi-wl-pprint,
|
||||||
crucible >= 0.4,
|
crucible >= 0.4,
|
||||||
crucible-llvm,
|
crucible-llvm,
|
||||||
flexdis86 >= 0.1.2,
|
flexdis86 >= 0.1.2,
|
||||||
@ -20,6 +21,7 @@ library
|
|||||||
parameterized-utils,
|
parameterized-utils,
|
||||||
what4 >= 0.4
|
what4 >= 0.4
|
||||||
hs-source-dirs: src
|
hs-source-dirs: src
|
||||||
|
default-language: Haskell2010
|
||||||
|
|
||||||
exposed-modules:
|
exposed-modules:
|
||||||
Data.Macaw.X86.Symbolic
|
Data.Macaw.X86.Symbolic
|
||||||
@ -27,6 +29,8 @@ library
|
|||||||
|
|
||||||
ghc-options: -Wall
|
ghc-options: -Wall
|
||||||
ghc-prof-options: -O2 -fprof-auto-top
|
ghc-prof-options: -O2 -fprof-auto-top
|
||||||
|
if impl(ghc >= 8.6)
|
||||||
|
default-extensions: NoStarIsType
|
||||||
|
|
||||||
test-suite macaw-x86-symbolic-tests
|
test-suite macaw-x86-symbolic-tests
|
||||||
type: exitcode-stdio-1.0
|
type: exitcode-stdio-1.0
|
||||||
|
@ -33,20 +33,26 @@ module Data.Macaw.X86.Crucible
|
|||||||
) where
|
) where
|
||||||
|
|
||||||
import Control.Lens ((^.))
|
import Control.Lens ((^.))
|
||||||
|
import Control.Monad
|
||||||
import Data.Bits hiding (xor)
|
import Data.Bits hiding (xor)
|
||||||
|
import Data.Kind ( Type )
|
||||||
import Data.Parameterized.Context.Unsafe (empty,extend)
|
import Data.Parameterized.Context.Unsafe (empty,extend)
|
||||||
import Data.Parameterized.Utils.Endian (Endian(..))
|
|
||||||
import Data.Parameterized.NatRepr
|
import Data.Parameterized.NatRepr
|
||||||
|
import Data.Parameterized.Utils.Endian (Endian(..))
|
||||||
import qualified Data.Parameterized.Vector as PV
|
import qualified Data.Parameterized.Vector as PV
|
||||||
|
import Data.Semigroup
|
||||||
import Data.Word (Word8)
|
import Data.Word (Word8)
|
||||||
import GHC.TypeLits (KnownNat)
|
import GHC.TypeLits (KnownNat)
|
||||||
|
import Text.PrettyPrint.ANSI.Leijen hiding ( (<$>), (<>), empty )
|
||||||
|
|
||||||
|
import What4.Concrete
|
||||||
import What4.Interface hiding (IsExpr)
|
import What4.Interface hiding (IsExpr)
|
||||||
import What4.InterpretedFloatingPoint
|
import What4.InterpretedFloatingPoint
|
||||||
import What4.Symbol (userSymbol)
|
import What4.Symbol (userSymbol)
|
||||||
|
|
||||||
import Lang.Crucible.Backend (IsSymInterface)
|
import Lang.Crucible.Backend (IsSymInterface)
|
||||||
import Lang.Crucible.CFG.Expr
|
import Lang.Crucible.CFG.Expr
|
||||||
|
import qualified Lang.Crucible.Simulator as C
|
||||||
import qualified Lang.Crucible.Simulator.Evaluation as C
|
import qualified Lang.Crucible.Simulator.Evaluation as C
|
||||||
import Lang.Crucible.Simulator.ExecutionTree
|
import Lang.Crucible.Simulator.ExecutionTree
|
||||||
import Lang.Crucible.Simulator.Intrinsics (IntrinsicTypes)
|
import Lang.Crucible.Simulator.Intrinsics (IntrinsicTypes)
|
||||||
@ -56,14 +62,25 @@ import Lang.Crucible.Types
|
|||||||
import qualified Lang.Crucible.Vector as V
|
import qualified Lang.Crucible.Vector as V
|
||||||
|
|
||||||
import Lang.Crucible.LLVM.MemModel
|
import Lang.Crucible.LLVM.MemModel
|
||||||
(LLVMPointerType, projectLLVM_bv,
|
( LLVMPointerType
|
||||||
pattern LLVMPointerRepr, llvmPointer_bv)
|
, Mem
|
||||||
|
, ptrAdd
|
||||||
|
, projectLLVM_bv
|
||||||
|
, pattern LLVMPointerRepr
|
||||||
|
, llvmPointer_bv
|
||||||
|
)
|
||||||
|
|
||||||
|
import qualified Data.Macaw.CFG.Core as M
|
||||||
|
import qualified Data.Macaw.Memory as M
|
||||||
import qualified Data.Macaw.Types as M
|
import qualified Data.Macaw.Types as M
|
||||||
import Data.Macaw.Symbolic.CrucGen (MacawExt)
|
import Data.Macaw.Symbolic.CrucGen (MacawExt)
|
||||||
import Data.Macaw.Symbolic
|
import Data.Macaw.Symbolic.MemOps
|
||||||
|
import Data.Macaw.Symbolic.PersistentState
|
||||||
import qualified Data.Macaw.X86 as M
|
import qualified Data.Macaw.X86 as M
|
||||||
import qualified Data.Macaw.X86.ArchTypes as M
|
import qualified Data.Macaw.X86.ArchTypes as M
|
||||||
|
import qualified Data.Macaw.CFG.Core as MC
|
||||||
|
|
||||||
|
import Prelude
|
||||||
|
|
||||||
|
|
||||||
type S sym rtp bs r ctx =
|
type S sym rtp bs r ctx =
|
||||||
@ -81,19 +98,83 @@ funcSemantics fs x s = do let sym = Sym { symIface = s^.stateSymInterface
|
|||||||
v <- pureSem sym x
|
v <- pureSem sym x
|
||||||
return (v,s)
|
return (v,s)
|
||||||
|
|
||||||
stmtSemantics :: (IsSymInterface sym)
|
withConcreteCountAndDir
|
||||||
=> SymFuns sym
|
:: (IsSymInterface sym, 1 <= w)
|
||||||
-> M.X86Stmt (AtomWrapper (RegEntry sym))
|
=> S sym rtp bs r ctx
|
||||||
-> S sym rtp bs r ctx
|
-> M.RepValSize w
|
||||||
-> IO (RegValue sym UnitType, S sym rtp bs r ctx)
|
-> (AtomWrapper (RegEntry sym) (M.BVType 64))
|
||||||
stmtSemantics = error "Symbolic-execution time semantics for x86 statements are not implemented yet"
|
-> (AtomWrapper (RegEntry sym) M.BoolType)
|
||||||
|
-> (S sym rtp bs r ctx -> (SymBV sym 64) -> IO (S sym rtp bs r ctx))
|
||||||
|
-> IO (RegValue sym UnitType, S sym rtp bs r ctx)
|
||||||
|
withConcreteCountAndDir state val_size wrapped_count wrapped_dir func = do
|
||||||
|
let sym = state^.stateSymInterface
|
||||||
|
let val_byte_size = M.repValSizeByteCount val_size
|
||||||
|
bv_count <- toValBV sym wrapped_count
|
||||||
|
case asConcrete bv_count of
|
||||||
|
Just (ConcreteBV _ count) -> do
|
||||||
|
res_crux_state <- foldM func state
|
||||||
|
=<< mapM (\index -> bvLit sym knownNat $ index * val_byte_size)
|
||||||
|
-- [0..((if dir then 1 else -1) * (count - 1))]
|
||||||
|
[0..(count - 1)]
|
||||||
|
return ((), res_crux_state)
|
||||||
|
Nothing -> error $ "Unsupported symbolic count in rep stmt: "
|
||||||
|
|
||||||
|
stmtSemantics
|
||||||
|
:: IsSymInterface sym
|
||||||
|
=> SymFuns sym
|
||||||
|
-> C.GlobalVar Mem
|
||||||
|
-> GlobalMap sym (M.ArchAddrWidth M.X86_64)
|
||||||
|
-> M.X86Stmt (AtomWrapper (RegEntry sym))
|
||||||
|
-> S sym rtp bs r ctx
|
||||||
|
-> IO (RegValue sym UnitType, S sym rtp bs r ctx)
|
||||||
|
stmtSemantics _sym_funs global_var_mem globals stmt state = do
|
||||||
|
let sym = state^.stateSymInterface
|
||||||
|
case stmt of
|
||||||
|
M.RepMovs val_size (AtomWrapper dest) (AtomWrapper src) count dir ->
|
||||||
|
withConcreteCountAndDir state val_size count dir $ \acc_state offset -> do
|
||||||
|
let mem_repr = M.repValSizeMemRepr val_size
|
||||||
|
curr_dest_ptr <- ptrAdd sym knownNat (regValue dest) offset
|
||||||
|
curr_src_ptr <- ptrAdd sym knownNat (regValue src) offset
|
||||||
|
(val, after_read_state) <- doReadMem
|
||||||
|
acc_state
|
||||||
|
global_var_mem
|
||||||
|
globals
|
||||||
|
M.Addr64
|
||||||
|
mem_repr
|
||||||
|
(RegEntry knownRepr curr_src_ptr)
|
||||||
|
(_, after_write_state) <- doWriteMem
|
||||||
|
after_read_state
|
||||||
|
global_var_mem
|
||||||
|
globals
|
||||||
|
M.Addr64
|
||||||
|
mem_repr
|
||||||
|
(RegEntry knownRepr curr_dest_ptr)
|
||||||
|
(RegEntry (typeToCrucible $ M.typeRepr mem_repr) val)
|
||||||
|
return after_write_state
|
||||||
|
M.RepStos val_size (AtomWrapper dest) (AtomWrapper val) count dir ->
|
||||||
|
withConcreteCountAndDir state val_size count dir $ \acc_state offset -> do
|
||||||
|
let mem_repr = M.repValSizeMemRepr val_size
|
||||||
|
curr_dest_ptr <- ptrAdd sym knownNat (regValue dest) offset
|
||||||
|
(_, after_write_state) <- doWriteMem
|
||||||
|
acc_state
|
||||||
|
global_var_mem
|
||||||
|
globals
|
||||||
|
M.Addr64
|
||||||
|
mem_repr
|
||||||
|
(RegEntry knownRepr curr_dest_ptr)
|
||||||
|
val
|
||||||
|
return after_write_state
|
||||||
|
_ -> error $
|
||||||
|
"Symbolic execution semantics for x86 statement are not implemented yet: "
|
||||||
|
<> (show $ MC.ppArchStmt (liftAtomIn (pretty . regType)) stmt)
|
||||||
|
|
||||||
termSemantics :: (IsSymInterface sym)
|
termSemantics :: (IsSymInterface sym)
|
||||||
=> SymFuns sym
|
=> SymFuns sym
|
||||||
-> M.X86TermStmt ids
|
-> M.X86TermStmt ids
|
||||||
-> S sym rtp bs r ctx
|
-> S sym rtp bs r ctx
|
||||||
-> IO (RegValue sym UnitType, S sym rtp bs r ctx)
|
-> IO (RegValue sym UnitType, S sym rtp bs r ctx)
|
||||||
termSemantics = error "Symbolic-execution time semantics for x86 terminators are not implemented yet"
|
termSemantics _fs x _s = error ("Symbolic execution semantics for x86 terminators are not implemented yet: " <>
|
||||||
|
(show $ MC.prettyF x))
|
||||||
|
|
||||||
data Sym s = Sym { symIface :: s
|
data Sym s = Sym { symIface :: s
|
||||||
, symTys :: IntrinsicTypes s
|
, symTys :: IntrinsicTypes s
|
||||||
@ -154,10 +235,10 @@ pureSem sym fn = do
|
|||||||
M.RDTSC{} -> error "RDTSC"
|
M.RDTSC{} -> error "RDTSC"
|
||||||
M.MemCmp{} -> error "MemCmp"
|
M.MemCmp{} -> error "MemCmp"
|
||||||
M.RepnzScas{} -> error "RepnzScas"
|
M.RepnzScas{} -> error "RepnzScas"
|
||||||
M.X86IDiv {} -> error "X86IDiv"
|
M.X86IDiv w n d -> sDiv sym w n d
|
||||||
M.X86IRem {} -> error "X86IRem"
|
M.X86IRem w n d -> sRem sym w n d
|
||||||
M.X86Div {} -> error "X86Div"
|
M.X86Div w n d -> uDiv sym w n d
|
||||||
M.X86Rem {} -> error "X86Rem"
|
M.X86Rem w n d -> uRem sym w n d
|
||||||
M.X87_Extend{} -> error "X87_Extend"
|
M.X87_Extend{} -> error "X87_Extend"
|
||||||
M.X87_FAdd{} -> error "X87_FAdd"
|
M.X87_FAdd{} -> error "X87_FAdd"
|
||||||
M.X87_FSub{} -> error "X87_FSub"
|
M.X87_FSub{} -> error "X87_FSub"
|
||||||
@ -381,6 +462,127 @@ shuffleB xs is = fmap lkp is
|
|||||||
(bv 0)
|
(bv 0)
|
||||||
(bvLookup xs (app $ BVTrunc n4 knownNat i)))
|
(bvLookup xs (app $ BVTrunc n4 knownNat i)))
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
-- | Performs a simple unsigned division operation.
|
||||||
|
--
|
||||||
|
-- The x86 numerator is twice the size as the denominator.
|
||||||
|
--
|
||||||
|
-- This function is only reponsible for the dividend (not any
|
||||||
|
-- remainder--see uRem for that), and any divide-by-zero exception was
|
||||||
|
-- already handled via an Assert.
|
||||||
|
uDiv :: ( IsSymInterface sym ) =>
|
||||||
|
Sym sym
|
||||||
|
-> M.RepValSize w
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType (w + w))
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType w)
|
||||||
|
-> IO (RegValue sym (LLVMPointerType w))
|
||||||
|
uDiv sym repsz n d = do
|
||||||
|
let dw = M.typeWidth $ M.repValSizeMemRepr repsz
|
||||||
|
withAddLeq dw dw $ \nw ->
|
||||||
|
case testLeq n1 nw of
|
||||||
|
Just LeqProof ->
|
||||||
|
divOp sym nw n dw d BVUdiv
|
||||||
|
Nothing -> error "uDiv unable to verify numerator is >= 1 bit"
|
||||||
|
|
||||||
|
-- | Performs a simple unsigned division operation.
|
||||||
|
--
|
||||||
|
-- The x86 numerator is twice the size as the denominator.
|
||||||
|
--
|
||||||
|
-- This function is only reponsible for the remainder (the dividend is
|
||||||
|
-- computed separately by uDiv), and any divide-by-zero exception was
|
||||||
|
-- already handled via an Assert.
|
||||||
|
uRem :: ( IsSymInterface sym ) =>
|
||||||
|
Sym sym
|
||||||
|
-> M.RepValSize w
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType (w + w))
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType w)
|
||||||
|
-> IO (RegValue sym (LLVMPointerType w))
|
||||||
|
uRem sym repsz n d = do
|
||||||
|
let dw = M.typeWidth $ M.repValSizeMemRepr repsz
|
||||||
|
withAddLeq dw dw $ \nw ->
|
||||||
|
case testLeq n1 nw of
|
||||||
|
Just LeqProof ->
|
||||||
|
divOp sym nw n dw d BVUrem
|
||||||
|
Nothing -> error "uRem unable to verify numerator is >= 1 bit"
|
||||||
|
|
||||||
|
-- | Performs a simple signed division operation.
|
||||||
|
--
|
||||||
|
-- The x86 numerator is twice the size as the denominator.
|
||||||
|
--
|
||||||
|
-- This function is only reponsible for the dividend (not any
|
||||||
|
-- remainder--see sRem for that), and any divide-by-zero exception was
|
||||||
|
-- already handled via an Assert.
|
||||||
|
sDiv :: ( IsSymInterface sym ) =>
|
||||||
|
Sym sym
|
||||||
|
-> M.RepValSize w
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType (w + w))
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType w)
|
||||||
|
-> IO (RegValue sym (LLVMPointerType w))
|
||||||
|
sDiv sym repsz n d = do
|
||||||
|
let dw = M.typeWidth $ M.repValSizeMemRepr repsz
|
||||||
|
withAddLeq dw dw $ \nw ->
|
||||||
|
case testLeq n1 nw of
|
||||||
|
Just LeqProof ->
|
||||||
|
divOp sym nw n dw d BVSdiv
|
||||||
|
Nothing -> error "sDiv unable to verify numerator is >= 1 bit"
|
||||||
|
|
||||||
|
-- | Performs a simple signed division operation.
|
||||||
|
--
|
||||||
|
-- The x86 numerator is twice the size as the denominator.
|
||||||
|
--
|
||||||
|
-- This function is only reponsible for the remainder (the dividend is
|
||||||
|
-- computed separately by sDiv), and any divide-by-zero exception was
|
||||||
|
-- already handled via an Assert.
|
||||||
|
sRem :: ( IsSymInterface sym ) =>
|
||||||
|
Sym sym
|
||||||
|
-> M.RepValSize w
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType (w + w))
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType w)
|
||||||
|
-> IO (RegValue sym (LLVMPointerType w))
|
||||||
|
sRem sym repsz n d = do
|
||||||
|
let dw = M.typeWidth $ M.repValSizeMemRepr repsz
|
||||||
|
withAddLeq dw dw $ \nw ->
|
||||||
|
case testLeq n1 nw of
|
||||||
|
Just LeqProof ->
|
||||||
|
divOp sym nw n dw d BVSrem
|
||||||
|
Nothing -> error "sRem unable to verify numerator is >= 1 bit"
|
||||||
|
|
||||||
|
-- | Common function for division and remainder computation for both
|
||||||
|
-- signed and unsigned BV expressions.
|
||||||
|
--
|
||||||
|
-- The x86 numerator is twice the size as the denominator, so
|
||||||
|
-- zero-extend the denominator, perform the division, then truncate
|
||||||
|
-- the result.
|
||||||
|
divOp :: ( IsSymInterface sym
|
||||||
|
, 1 <= (w + w)
|
||||||
|
, w <= (w + w)
|
||||||
|
) =>
|
||||||
|
Sym sym
|
||||||
|
-> NatRepr (w + w)
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType (w + w))
|
||||||
|
-> NatRepr w
|
||||||
|
-> AtomWrapper (RegEntry sym) (M.BVType w)
|
||||||
|
-> (NatRepr (w + w) -> E sym (BVType (w + w)) -> E sym (BVType (w + w)) -> App () (E sym) (BVType (w + w)))
|
||||||
|
-> IO (RegValue sym (LLVMPointerType w))
|
||||||
|
divOp sym nw n' dw d' op = do
|
||||||
|
let symi = symIface sym
|
||||||
|
n <- getBitVal symi n'
|
||||||
|
d <- getBitVal symi d'
|
||||||
|
case testLeq n1 dw of
|
||||||
|
Just LeqProof ->
|
||||||
|
case testLeq (incNat dw) nw of
|
||||||
|
Just LeqProof ->
|
||||||
|
llvmPointer_bv symi =<< (evalE sym
|
||||||
|
-- (assertExpr (app $ BVEq dw (app $ BVLit dw 0)
|
||||||
|
(app $ BVTrunc dw nw $ app $ op nw (app $ BVZext nw dw d) n))
|
||||||
|
-- "must not be zero"
|
||||||
|
-- )
|
||||||
|
-- )
|
||||||
|
Nothing -> error "divOp unable to prove numerator size > denominator size + 1"
|
||||||
|
Nothing -> error "divOp unable to prove denominator size > 1 bit"
|
||||||
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
divExact ::
|
divExact ::
|
||||||
NatRepr n ->
|
NatRepr n ->
|
||||||
@ -602,7 +804,7 @@ evalApp x = C.evalApp (symIface x) (symTys x) logger evalExt (evalE x)
|
|||||||
evalExt :: fun -> EmptyExprExtension f a -> IO (RegValue sym a)
|
evalExt :: fun -> EmptyExprExtension f a -> IO (RegValue sym a)
|
||||||
evalExt _ y = case y of {}
|
evalExt _ y = case y of {}
|
||||||
|
|
||||||
data E :: * -> CrucibleType -> * where
|
data E :: Type -> CrucibleType -> Type where
|
||||||
ValBool :: RegValue sym BoolType -> E sym BoolType
|
ValBool :: RegValue sym BoolType -> E sym BoolType
|
||||||
ValBV :: (1 <= w) => NatRepr w -> RegValue sym (BVType w) -> E sym (BVType w)
|
ValBV :: (1 <= w) => NatRepr w -> RegValue sym (BVType w) -> E sym (BVType w)
|
||||||
Expr :: App () (E sym) t -> E sym t
|
Expr :: App () (E sym) t -> E sym t
|
||||||
@ -682,7 +884,7 @@ n128 = knownNat
|
|||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
newtype AtomWrapper (f :: CrucibleType -> *) (tp :: M.Type)
|
newtype AtomWrapper (f :: CrucibleType -> Type) (tp :: M.Type)
|
||||||
= AtomWrapper (f (ToCrucibleType tp))
|
= AtomWrapper (f (ToCrucibleType tp))
|
||||||
|
|
||||||
liftAtomMap :: (forall s. f s -> g s) -> AtomWrapper f t -> AtomWrapper g t
|
liftAtomMap :: (forall s. f s -> g s) -> AtomWrapper f t -> AtomWrapper g t
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
{-# LANGUAGE TypeApplications #-}
|
{-# LANGUAGE TypeApplications #-}
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE AllowAmbiguousTypes #-}
|
{-# LANGUAGE AllowAmbiguousTypes #-}
|
||||||
|
{-# OPTIONS_GHC -fno-warn-orphans #-}
|
||||||
module Data.Macaw.X86.Symbolic
|
module Data.Macaw.X86.Symbolic
|
||||||
( x86_64MacawSymbolicFns
|
( x86_64MacawSymbolicFns
|
||||||
, x86_64MacawEvalFn
|
, x86_64MacawEvalFn
|
||||||
@ -28,7 +29,9 @@ module Data.Macaw.X86.Symbolic
|
|||||||
|
|
||||||
import Control.Lens ((^.),(%~),(&))
|
import Control.Lens ((^.),(%~),(&))
|
||||||
import Control.Monad ( void )
|
import Control.Monad ( void )
|
||||||
|
import Control.Monad.IO.Class ( liftIO )
|
||||||
import Data.Functor.Identity (Identity(..))
|
import Data.Functor.Identity (Identity(..))
|
||||||
|
import Data.Kind
|
||||||
import Data.Parameterized.Context as Ctx
|
import Data.Parameterized.Context as Ctx
|
||||||
import Data.Parameterized.Map as MapF
|
import Data.Parameterized.Map as MapF
|
||||||
import Data.Parameterized.TraversableF
|
import Data.Parameterized.TraversableF
|
||||||
@ -54,6 +57,7 @@ import qualified Lang.Crucible.CFG.Extension as C
|
|||||||
import qualified Lang.Crucible.CFG.Reg as C
|
import qualified Lang.Crucible.CFG.Reg as C
|
||||||
import Lang.Crucible.Simulator.RegValue (RegValue'(..))
|
import Lang.Crucible.Simulator.RegValue (RegValue'(..))
|
||||||
import qualified Lang.Crucible.Types as C
|
import qualified Lang.Crucible.Types as C
|
||||||
|
import qualified Lang.Crucible.LLVM.MemModel as MM
|
||||||
|
|
||||||
------------------------------------------------------------------------
|
------------------------------------------------------------------------
|
||||||
-- Utilities for generating a type-level context with repeated elements.
|
-- Utilities for generating a type-level context with repeated elements.
|
||||||
@ -185,7 +189,7 @@ freshX86Reg sym r =
|
|||||||
|
|
||||||
-- | We currently make a type like this, we could instead a generic
|
-- | We currently make a type like this, we could instead a generic
|
||||||
-- X86PrimFn function
|
-- X86PrimFn function
|
||||||
data X86StmtExtension (f :: C.CrucibleType -> *) (ctp :: C.CrucibleType) where
|
data X86StmtExtension (f :: C.CrucibleType -> Type) (ctp :: C.CrucibleType) where
|
||||||
-- | To reduce clutter, but potentially increase clutter, we just make every
|
-- | To reduce clutter, but potentially increase clutter, we just make every
|
||||||
-- Macaw X86PrimFn a Macaw-Crucible statement extension.
|
-- Macaw X86PrimFn a Macaw-Crucible statement extension.
|
||||||
X86PrimFn :: !(M.X86PrimFn (AtomWrapper f) t) ->
|
X86PrimFn :: !(M.X86PrimFn (AtomWrapper f) t) ->
|
||||||
@ -263,8 +267,23 @@ x86_64MacawSymbolicFns =
|
|||||||
|
|
||||||
|
|
||||||
-- | X86_64 specific function for evaluating a Macaw X86_64 program in Crucible.
|
-- | X86_64 specific function for evaluating a Macaw X86_64 program in Crucible.
|
||||||
x86_64MacawEvalFn ::
|
x86_64MacawEvalFn
|
||||||
C.IsSymInterface sym => SymFuns sym -> MacawArchEvalFn sym M.X86_64
|
:: C.IsSymInterface sym
|
||||||
x86_64MacawEvalFn fs (X86PrimFn x) s = funcSemantics fs x s
|
=> SymFuns sym
|
||||||
x86_64MacawEvalFn fs (X86PrimStmt stmt) s = stmtSemantics fs stmt s
|
-> C.GlobalVar MM.Mem
|
||||||
x86_64MacawEvalFn fs (X86PrimTerm term) s = termSemantics fs term s
|
-> GlobalMap sym (M.ArchAddrWidth M.X86_64)
|
||||||
|
-> MacawArchEvalFn sym M.X86_64
|
||||||
|
x86_64MacawEvalFn fs global_var_mem globals ext_stmt crux_state =
|
||||||
|
case ext_stmt of
|
||||||
|
X86PrimFn x -> funcSemantics fs x crux_state
|
||||||
|
X86PrimStmt stmt -> stmtSemantics fs global_var_mem globals stmt crux_state
|
||||||
|
X86PrimTerm term -> termSemantics fs term crux_state
|
||||||
|
|
||||||
|
instance ArchInfo M.X86_64 where
|
||||||
|
archVals _ = Just $ ArchVals
|
||||||
|
{ archFunctions = x86_64MacawSymbolicFns
|
||||||
|
, withArchEval = \sym -> \k -> do
|
||||||
|
sfns <- liftIO $ newSymFuns sym
|
||||||
|
k $ x86_64MacawEvalFn sfns
|
||||||
|
, withArchConstraints = \x -> x
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user