mirror of
https://github.com/GaloisInc/macaw.git
synced 2024-11-24 08:53:12 +03:00
[refinement] Invoke SMT solution at path-focused framework location.
The previous implementation invoked the SMT solver at the top level for prototyping. This version moves the SMT solver invocation to the intended location in the algorithm where the path is successively extended and solutions are compared to identify the "best" refinement solution.
This commit is contained in:
parent
ff2ec55f2c
commit
bd0e57cfc1
@ -5,6 +5,7 @@ module Data.Macaw.Refinement.FuncBlockUtils
|
|||||||
, blockTransferTo
|
, blockTransferTo
|
||||||
, funBlockIDs
|
, funBlockIDs
|
||||||
, funForBlock
|
, funForBlock
|
||||||
|
, getBlock
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
@ -61,6 +62,11 @@ funIncludesBlock :: BlockIdentifier arch
|
|||||||
funIncludesBlock blkID (Some fi) =
|
funIncludesBlock blkID (Some fi) =
|
||||||
isJust ((fi ^. parsedBlocks) Map.!? blkID)
|
isJust ((fi ^. parsedBlocks) Map.!? blkID)
|
||||||
|
|
||||||
|
getBlock :: DiscoveryState arch
|
||||||
|
-> BlockIdentifier arch
|
||||||
|
-> Some (ParsedBlock arch)
|
||||||
|
getBlock ds blkID = undefined
|
||||||
|
|
||||||
-- | This function identifies the possible target addresses (of other
|
-- | This function identifies the possible target addresses (of other
|
||||||
-- blocks within this function) from the terminal statement in the
|
-- blocks within this function) from the terminal statement in the
|
||||||
-- input block. Note that this function is responsible for returning
|
-- input block. Note that this function is responsible for returning
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts #-}
|
||||||
{-# LANGUAGE UndecidableInstances #-}
|
{-# LANGUAGE UndecidableInstances #-}
|
||||||
module Data.Macaw.Refinement.Path
|
module Data.Macaw.Refinement.Path
|
||||||
( FuncBlockPath
|
( FuncBlockPath(..)
|
||||||
, buildFuncPath
|
, buildFuncPath
|
||||||
, pathDepth
|
, pathDepth
|
||||||
, pathTo
|
, pathTo
|
||||||
|
@ -130,8 +130,8 @@ import Data.Macaw.Discovery.State ( DiscoveryFunInfo
|
|||||||
, parsedBlocks
|
, parsedBlocks
|
||||||
, stmtsTerm
|
, stmtsTerm
|
||||||
)
|
)
|
||||||
import Data.Macaw.Refinement.FuncBlockUtils ( BlockIdentifier, blockID, funForBlock )
|
import Data.Macaw.Refinement.FuncBlockUtils ( BlockIdentifier, blockID, funForBlock, getBlock )
|
||||||
import Data.Macaw.Refinement.Path ( FuncBlockPath, buildFuncPath, pathDepth, pathTo, takePath )
|
import Data.Macaw.Refinement.Path ( FuncBlockPath(..), buildFuncPath, pathDepth, pathTo, takePath )
|
||||||
import qualified Data.Macaw.Symbolic as MS
|
import qualified Data.Macaw.Symbolic as MS
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
@ -198,11 +198,9 @@ refineTransfers failedRefine inpDS = do
|
|||||||
. filtered (not . unrefineable)
|
. filtered (not . unrefineable)
|
||||||
thisUnkTransfer = head unkTransfers
|
thisUnkTransfer = head unkTransfers
|
||||||
thisId = blockID thisUnkTransfer
|
thisId = blockID thisUnkTransfer
|
||||||
block_addrs <- liftIO $ withDefaultRefinementContext $ \context ->
|
|
||||||
mapM (refineBlockTransfer' context inpDS) unkTransfers
|
|
||||||
if null unkTransfers
|
if null unkTransfers
|
||||||
then return inpDS
|
then return inpDS
|
||||||
else case refineBlockTransfer inpDS thisUnkTransfer of
|
else refineBlockTransfer inpDS thisUnkTransfer >>= \case
|
||||||
Nothing -> refineTransfers (thisId : failedRefine) inpDS
|
Nothing -> refineTransfers (thisId : failedRefine) inpDS
|
||||||
Just updDS -> refineTransfers failedRefine updDS
|
Just updDS -> refineTransfers failedRefine updDS
|
||||||
|
|
||||||
@ -231,51 +229,81 @@ isUnknownTransfer pb =
|
|||||||
-- that. If it was unable to refine the transfer, it will return
|
-- that. If it was unable to refine the transfer, it will return
|
||||||
-- Nothing and this block will be added to the "unresolvable" list.
|
-- Nothing and this block will be added to the "unresolvable" list.
|
||||||
refineBlockTransfer
|
refineBlockTransfer
|
||||||
:: DiscoveryState arch
|
:: ( MS.SymArchConstraints arch
|
||||||
|
, 16 <= MC.ArchAddrWidth arch
|
||||||
|
, MonadIO m
|
||||||
|
) =>
|
||||||
|
DiscoveryState arch
|
||||||
-> Some (ParsedBlock arch)
|
-> Some (ParsedBlock arch)
|
||||||
-> Maybe (DiscoveryState arch)
|
-> m (Maybe (DiscoveryState arch))
|
||||||
refineBlockTransfer inpDS blk =
|
refineBlockTransfer inpDS blk =
|
||||||
let path = buildFuncPath <$> funForBlock blk inpDS
|
let path = buildFuncPath <$> funForBlock blk inpDS
|
||||||
in case path >>= pathTo (blockID blk) of
|
tgtPath = path >>= pathTo (blockID blk)
|
||||||
|
in case tgtPath of
|
||||||
Nothing -> error "unable to find function path for block" -- internal error
|
Nothing -> error "unable to find function path for block" -- internal error
|
||||||
Just p -> do soln <- refinePath inpDS p (pathDepth p) 0 Nothing
|
Just p -> do soln <- refinePath inpDS p (pathDepth p) 0 Nothing
|
||||||
return $ updateDiscovery soln blk inpDS
|
return $ maybe Nothing (Just . updateDiscovery inpDS blk) soln
|
||||||
|
|
||||||
|
|
||||||
updateDiscovery :: Solution
|
updateDiscovery :: DiscoveryState arch
|
||||||
-> Some (ParsedBlock arch)
|
-> Some (ParsedBlock arch)
|
||||||
-> DiscoveryState arch
|
-> Solution arch
|
||||||
-> DiscoveryState arch
|
-> DiscoveryState arch
|
||||||
updateDiscovery _soln _pblk _inpDS = undefined -- add replace pblk with soln, and add new blocks discoverd by soln
|
updateDiscovery _soln _pblk _inpDS = undefined -- add replace pblk with soln, and add new blocks discoverd by soln
|
||||||
|
|
||||||
|
|
||||||
|
refinePath :: ( MS.SymArchConstraints arch
|
||||||
|
, 16 <= MC.ArchAddrWidth arch
|
||||||
|
, MonadIO m
|
||||||
|
) =>
|
||||||
|
DiscoveryState arch
|
||||||
|
-> FuncBlockPath arch
|
||||||
|
-> Int
|
||||||
|
-> Int
|
||||||
|
-> Maybe (Solution arch)
|
||||||
|
-> m (Maybe (Solution arch))
|
||||||
refinePath inpDS path maxlevel numlevels prevResult =
|
refinePath inpDS path maxlevel numlevels prevResult =
|
||||||
let thispath = takePath numlevels path
|
let thispath = takePath numlevels path
|
||||||
smtEquation = equationFor inpDS thispath
|
smtEquation = equationFor inpDS thispath
|
||||||
in case solve smtEquation of
|
in solve smtEquation >>= \case
|
||||||
Nothing -> prevResult -- divergent, stop here
|
Nothing -> return prevResult -- divergent, stop here
|
||||||
Just soln -> let nextlevel = numlevels + 1
|
s@(Just soln) -> let nextlevel = numlevels + 1
|
||||||
bestResult = case prevResult of
|
bestResult = case prevResult of
|
||||||
Nothing -> Just soln
|
Nothing -> s
|
||||||
Just prevSoln ->
|
Just prevSoln ->
|
||||||
if soln `isBetterSolution` prevSoln
|
if soln < prevSoln
|
||||||
then Just soln
|
then s
|
||||||
else prevResult
|
else prevResult
|
||||||
in if numlevels > maxlevel
|
in if numlevels > maxlevel
|
||||||
then bestResult
|
then return bestResult
|
||||||
else refinePath inpDS path maxlevel nextlevel bestResult
|
else refinePath inpDS path maxlevel nextlevel bestResult
|
||||||
|
|
||||||
data Equation = Equation -- replace by What4 expression to pass to Crucible
|
data Equation arch = Equation (DiscoveryState arch) (Some (ParsedBlock arch))
|
||||||
data Solution = Solution -- replace by Crucible output
|
type Solution arch = [ArchSegmentOff arch] -- identified transfers
|
||||||
|
|
||||||
equationFor :: DiscoveryState arch -> FuncBlockPath arch -> Equation
|
equationFor :: DiscoveryState arch -> FuncBlockPath arch -> Equation arch
|
||||||
equationFor inpDS path = undefined
|
equationFor inpDS (Path bid anc _loop) =
|
||||||
|
let curBlk = getBlock inpDS bid
|
||||||
|
in if null anc
|
||||||
|
then Equation inpDS curBlk
|
||||||
|
else undefined
|
||||||
|
-- Should linearly combine the anc statements with the
|
||||||
|
-- current block's statements and asserts that state that
|
||||||
|
-- the IP from one to the next is expected.
|
||||||
|
|
||||||
solve :: Equation -> Maybe Solution
|
solve :: ( MS.SymArchConstraints arch
|
||||||
solve _eqn = Just Solution
|
, 16 <= MC.ArchAddrWidth arch
|
||||||
|
, MonadIO m
|
||||||
|
) =>
|
||||||
|
Equation arch -> m (Maybe (Solution arch))
|
||||||
|
solve (Equation inpDS blk) = do
|
||||||
|
blockAddrs <- liftIO (withDefaultRefinementContext $ \context ->
|
||||||
|
smtSolveTransfer context inpDS blk)
|
||||||
|
return $ if null blockAddrs then Nothing else Just blockAddrs
|
||||||
|
|
||||||
isBetterSolution :: Solution -> Solution -> Bool
|
--isBetterSolution :: Solution arch -> Solution arch -> Bool
|
||||||
isBetterSolution _solnA _solnB = True -- TBD
|
-- isBetterSolution :: [ArchSegmentOff arch] -> [ArchSegmentOff arch] -> Bool
|
||||||
|
-- isBetterSolution = (<)
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
----------------------------------------------------------------------
|
||||||
-- * Symbolic execution
|
-- * Symbolic execution
|
||||||
@ -369,7 +397,7 @@ initRegs arch_vals sym ip_val = do
|
|||||||
return $ C.RegMap $ Ctx.singleton $
|
return $ C.RegMap $ Ctx.singleton $
|
||||||
(MS.updateReg arch_vals) reg_struct MC.ip_reg ip_val
|
(MS.updateReg arch_vals) reg_struct MC.ip_reg ip_val
|
||||||
|
|
||||||
refineBlockTransfer'
|
smtSolveTransfer
|
||||||
:: forall arch t solver fp m
|
:: forall arch t solver fp m
|
||||||
. ( MS.SymArchConstraints arch
|
. ( MS.SymArchConstraints arch
|
||||||
, C.IsSymInterface (C.OnlineBackend t solver fp)
|
, C.IsSymInterface (C.OnlineBackend t solver fp)
|
||||||
@ -380,7 +408,7 @@ refineBlockTransfer'
|
|||||||
-> DiscoveryState arch
|
-> DiscoveryState arch
|
||||||
-> Some (ParsedBlock arch)
|
-> Some (ParsedBlock arch)
|
||||||
-> m [ArchSegmentOff arch]
|
-> m [ArchSegmentOff arch]
|
||||||
refineBlockTransfer' RefinementContext{..} discovery_state (Some block) = do
|
smtSolveTransfer RefinementContext{..} discovery_state (Some block) = do
|
||||||
let arch = Proxy @arch
|
let arch = Proxy @arch
|
||||||
block_ip_val <- case MC.segoffAsAbsoluteAddr (pblockAddr block) of
|
block_ip_val <- case MC.segoffAsAbsoluteAddr (pblockAddr block) of
|
||||||
Just addr -> liftIO $ LLVM.llvmPointer_bv symbolicBackend
|
Just addr -> liftIO $ LLVM.llvmPointer_bv symbolicBackend
|
||||||
@ -455,4 +483,4 @@ genModels sym expr count
|
|||||||
more_ground_vals <- genModels sym expr (count - 1)
|
more_ground_vals <- genModels sym expr (count - 1)
|
||||||
return $ next_ground_val : more_ground_vals
|
return $ next_ground_val : more_ground_vals
|
||||||
_ -> return []
|
_ -> return []
|
||||||
| otherwise = return []
|
| otherwise = return []
|
||||||
|
Loading…
Reference in New Issue
Block a user