Skip to content

Commit

Permalink
Merge pull request #1637 from GaloisInc/rwd/dev-tools
Browse files Browse the repository at this point in the history
Proof development primitives
  • Loading branch information
robdockins authored Aug 8, 2022
2 parents ff3a41f + 6952456 commit 6cbe62a
Show file tree
Hide file tree
Showing 18 changed files with 741 additions and 289 deletions.
10 changes: 10 additions & 0 deletions cryptol-saw-core/src/Verifier/SAW/CryptolEnv.hs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ module Verifier.SAW.CryptolEnv
, loadCryptolModule
, bindCryptolModule
, lookupCryptolModule
, combineCryptolEnv
, importModule
, bindTypedTerm
, bindType
Expand Down Expand Up @@ -345,6 +346,15 @@ genTermEnv sc modEnv cryEnv0 = do

--------------------------------------------------------------------------------


combineCryptolEnv :: CryptolEnv -> CryptolEnv -> IO CryptolEnv
combineCryptolEnv chkEnv newEnv =
do let newMEnv = eModuleEnv newEnv
let chkMEnv = eModuleEnv chkEnv
let menv' = chkMEnv{ ME.meNameSeeds = ME.meNameSeeds newMEnv }
return chkEnv{ eModuleEnv = menv' }


checkNotParameterized :: T.Module -> IO ()
checkNotParameterized m =
when (T.isParametrizedModule m) $
Expand Down
4 changes: 2 additions & 2 deletions intTests/test_external_abc/test.saw
Original file line number Diff line number Diff line change
Expand Up @@ -44,8 +44,8 @@ let order_res = {{ ([[0x02,0x03],[0x04,0x05]], 0x81050fff) }};

// Check that Verilog counterexamples are in the right order
sr1 <- sat w4_abc_verilog order_term;
caseSatResult sr1 undefined (\t -> prove_print yices {{ t == order_res }});
caseSatResult sr1 (fail "ABC verilog sat fail") (\t -> prove_print yices {{ t == order_res }});

// Check that AIGER counterexamples are in the right order
sr2 <- sat w4_abc_aiger order_term;
caseSatResult sr2 undefined (\t -> prove_print yices {{ t == order_res }});
caseSatResult sr2 (fail "ABC aiger sat fail") (\t -> prove_print yices {{ t == order_res }});
30 changes: 29 additions & 1 deletion saw-core/src/Verifier/SAW/SharedTerm.hs
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,9 @@ module Verifier.SAW.SharedTerm
, SharedContext
, mkSharedContext
, scGetModuleMap
, SharedContextCheckpoint
, checkpointSharedContext
, restoreSharedContext
-- ** Low-level generic term constructors
, scTermF
, scFlatTermF
Expand Down Expand Up @@ -282,7 +285,7 @@ import qualified Data.HashMap.Strict as HMap
import Data.IntMap (IntMap)
import qualified Data.IntMap as IntMap
import qualified Data.IntSet as IntSet
import Data.IORef (IORef,newIORef,readIORef,modifyIORef',atomicModifyIORef')
import Data.IORef (IORef,newIORef,readIORef,modifyIORef',atomicModifyIORef',writeIORef)
import Data.Map (Map)
import qualified Data.Map as Map
import Data.Ref ( C )
Expand Down Expand Up @@ -360,6 +363,31 @@ data SharedContext = SharedContext
, scFreshGlobalVar :: IO VarIndex
}

data SharedContextCheckpoint =
SCC
{ sccModuleMap :: ModuleMap
, sccNamingEnv :: SAWNamingEnv
, sccGlobalEnv :: HashMap Ident Term
}

checkpointSharedContext :: SharedContext -> IO SharedContextCheckpoint
checkpointSharedContext sc =
do mmap <- readIORef (scModuleMap sc)
nenv <- readIORef (scNamingEnv sc)
genv <- readIORef (scGlobalEnv sc)
return SCC
{ sccModuleMap = mmap
, sccNamingEnv = nenv
, sccGlobalEnv = genv
}

restoreSharedContext :: SharedContextCheckpoint -> SharedContext -> IO SharedContext
restoreSharedContext scc sc =
do writeIORef (scModuleMap sc) (sccModuleMap scc)
writeIORef (scNamingEnv sc) (sccNamingEnv scc)
writeIORef (scGlobalEnv sc) (sccGlobalEnv scc)
return sc

-- | Create a new term from a lower-level 'FlatTermF' term.
scFlatTermF :: SharedContext -> FlatTermF Term -> IO Term
scFlatTermF sc ftf = scTermF sc (FTermF ftf)
Expand Down
10 changes: 7 additions & 3 deletions saw-remote-api/src/SAWServer.hs
Original file line number Diff line number Diff line change
Expand Up @@ -199,8 +199,7 @@ initialState readFileFn =
cwd <- getCurrentDirectory
db <- newTheoremDB
let ro = TopLevelRO
{ roSharedContext = sc
, roJavaCodebase = jcb
{ roJavaCodebase = jcb
, roOptions = opts
, roHandleAlloc = halloc
, roPosition = PosInternal "SAWServer"
Expand All @@ -211,7 +210,10 @@ initialState readFileFn =
#endif
, roInitWorkDir = cwd
, roBasicSS = ss
, roTheoremDB = db
, roStackTrace = []
, roSubshell = fail "SAW server does not support subshells."
, roProofSubshell = fail "SAW server does not support subshells."
, roLocalEnv = []
}
rw = TopLevelRW
{ rwValues = mempty
Expand All @@ -222,6 +224,8 @@ initialState readFileFn =
, rwMonadify = defaultMonEnv
, rwMRSolverEnv = emptyMREnv
, rwPPOpts = defaultPPOpts
, rwTheoremDB = db
, rwSharedContext = sc
, rwJVMTrans = jvmTrans
, rwPrimsAvail = mempty
, rwSMTArrayMemoryModel = False
Expand Down
3 changes: 2 additions & 1 deletion saw-remote-api/src/SAWServer/Eval.hs
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,8 @@ instance Doc.DescribedMethod (EvalParams Bool cryptolExpr) (EvalResult Bool) whe
Doc.Paragraph [Doc.Text "The boolean value of the expresssion."])
]

eval :: (TypedTerm -> SV.TopLevel a) -> EvalParams a Expression -> Argo.Command SAWState (EvalResult a)
eval :: (SV.FromValue a, SV.IsValue a) =>
(TypedTerm -> SV.TopLevel a) -> EvalParams a Expression -> Argo.Command SAWState (EvalResult a)
eval f params = do
state <- Argo.getState
fileReader <- Argo.getFileReader
Expand Down
15 changes: 6 additions & 9 deletions saw-remote-api/src/SAWServer/TopLevel.hs
Original file line number Diff line number Diff line change
Expand Up @@ -5,31 +5,28 @@ module SAWServer.TopLevel (tl) where
import Control.Exception ( try, SomeException(..) )
import Control.Lens ( view, set )
import Control.Monad.State ( MonadIO(liftIO) )
import Data.IORef
import Data.Typeable (cast)

import SAWScript.Value ( TopLevel, runTopLevel )
import SAWScript.Value ( TopLevel, runTopLevel, IsValue(..), FromValue(..) )

import qualified Argo
import CryptolServer.Exceptions (cryptolError)
import SAWServer ( SAWState, sawTopLevelRO, sawTopLevelRW )
import SAWServer.CryptolExpression (CryptolModuleException(..))
import SAWServer.Exceptions ( verificationException )

tl :: TopLevel a -> Argo.Command SAWState a
tl :: (FromValue a, IsValue a) => TopLevel a -> Argo.Command SAWState a
tl act =
do st <- Argo.getState
let ro = view sawTopLevelRO st
rw = view sawTopLevelRW st
ref <- liftIO (newIORef rw)
liftIO (try (runTopLevel act ro ref)) >>=
liftIO (try (runTopLevel act ro rw)) >>=
\case
Left e@(SomeException e')
| Just (CryptolModuleException err warnings) <- cast e'
-> Argo.raise (cryptolError err warnings)
| otherwise
-> Argo.raise (verificationException e)
Right res ->
do rw' <- liftIO (readIORef ref)
Argo.modifyState $ set sawTopLevelRW rw'
return res
Right (res, rw') ->
do Argo.modifyState $ set sawTopLevelRW rw'
return (fromValue res)
1 change: 1 addition & 0 deletions saw-script.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -213,6 +213,7 @@ executable saw
, exceptions
, filepath
, haskeline
, mtl
, QuickCheck
, transformers
, saw-script
Expand Down
9 changes: 7 additions & 2 deletions saw/Main.hs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ import SAWScript.Options
import SAWScript.Utils
import SAWScript.Interpreter (processFile)
import qualified SAWScript.REPL as REPL
import qualified SAWScript.REPL.Haskeline as REPL
import qualified SAWScript.REPL.Monad as REPL
import SAWScript.Version (shortVersionText)
import SAWScript.Value (AIGProxy(..))
import qualified Data.AIG.CompactGraph as AIG
Expand All @@ -41,12 +43,14 @@ main = do
[] -> checkZ3 opts'' *> REPL.run opts''
_ | runInteractively opts'' -> checkZ3 opts'' *> REPL.run opts''
[file] -> checkZ3 opts'' *>
processFile (AIGProxy AIG.compactProxy) opts'' file `catch`
processFile (AIGProxy AIG.compactProxy) opts'' file subsh proofSubsh`catch`
(\(ErrorCall msg) -> err opts'' msg)
(_:_) -> err opts'' "Multiple files not yet supported."
(_, _, errs) -> do hPutStrLn stderr (concat errs ++ usageInfo header options)
exitProofUnknown
where header = "Usage: saw [OPTION...] [-I | file]"
where subsh = Just (REPL.subshell (REPL.replBody Nothing (return ())))
proofSubsh = Just (REPL.proof_subshell (REPL.replBody Nothing (return ())))
header = "Usage: saw [OPTION...] [-I | file]"
checkZ3 opts = do
p <- findExecutable "z3"
unless (isJust p)
Expand All @@ -55,3 +59,4 @@ main = do
when (verbLevel opts >= Error)
(hPutStrLn stderr msg)
exitProofUnknown

26 changes: 11 additions & 15 deletions saw/SAWScript/REPL/Command.hs
Original file line number Diff line number Diff line change
Expand Up @@ -30,14 +30,15 @@ module SAWScript.REPL.Command (

--import Verifier.SAW.SharedTerm (SharedContext)


import SAWScript.REPL.Monad
import SAWScript.REPL.Trie
import SAWScript.Position (getPos)

import Cryptol.Parser (ParseError())
import Cryptol.Utils.PP hiding ((</>))

import Control.Monad (guard)
import Control.Monad (guard, void)

import Data.Char (isSpace,isPunctuation,isSymbol)
import Data.Function (on)
import Data.List (intercalate)
Expand All @@ -56,7 +57,7 @@ import SAWScript.MGU (checkDecl)
import SAWScript.Interpreter (interpretStmt)
import qualified SAWScript.Lexer (lexSAW)
import qualified SAWScript.Parser (parseStmtSemi, parseExpression)
import SAWScript.TopLevel (TopLevelRW(..), runTopLevel)
import SAWScript.TopLevel (TopLevelRW(..))


-- Commands --------------------------------------------------------------------
Expand Down Expand Up @@ -139,13 +140,7 @@ genHelp cs = map cmdHelp cs
runCommand :: Command -> REPL ()
runCommand c = case c of

Command cmd -> cmd `SAWScript.REPL.Monad.catch` handlerPP
`SAWScript.REPL.Monad.catchFail` handlerFail
`SAWScript.REPL.Monad.catchOther` handlerPrint
where
handlerPP re = io (putStrLn "" >> print (pp re))
handlerPrint e = io (putStrLn "" >> print e)
handlerFail s = io (putStrLn "" >> putStrLn s)
Command cmd -> exceptionProtect cmd

Unknown cmd -> io (putStrLn ("Unknown command: " ++ cmd))

Expand All @@ -161,7 +156,7 @@ typeOfCmd str =
Left err -> fail (show err)
Right expr -> return expr
let decl = SS.Decl (getPos expr) (SS.PWild Nothing) Nothing expr
rw <- getEnvironment
rw <- getValueEnvironment
~(SS.Decl _pos _ (Just schema) _expr') <-
either failTypecheck return $ checkDecl (rwTypes rw) (rwTypedef rw) decl
io $ putStrLn $ SS.pShow schema
Expand All @@ -172,7 +167,7 @@ quitCmd = stop

envCmd :: REPL ()
envCmd = do
env <- getEnvironment
env <- getValueEnvironment
let showLName = SS.getVal
io $ sequence_ [ putStrLn (showLName x ++ " : " ++ SS.pShow v) | (x, v) <- Map.assocs (rwTypes env) ]

Expand Down Expand Up @@ -227,9 +222,10 @@ sawScriptCmd str = do
case SAWScript.Parser.parseStmtSemi tokens of
Left err -> io $ print err
Right stmt ->
do ro <- getTopLevelRO
rwRef <- getEnvironmentRef
io $ runTopLevel (interpretStmt True stmt) ro rwRef
do mr <- getProofStateRef
case mr of
Nothing -> void $ liftTopLevel (interpretStmt True stmt)
Just r -> void $ liftProofScript (interpretStmt True stmt) r

replFileName :: String
replFileName = "<stdin>"
Expand Down
23 changes: 17 additions & 6 deletions saw/SAWScript/REPL/Haskeline.hs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import qualified Control.Monad.Catch as E
import Data.Char (isAlphaNum, isSpace)
import Data.Function (on)
import Data.List (isPrefixOf,sortBy)
import Data.Maybe (isJust)
import System.Console.Haskeline
import System.Directory(getAppUserDataDirectory,createDirectoryIfMissing)
import System.FilePath((</>))
Expand All @@ -29,6 +30,7 @@ import qualified Control.Monad.Trans.Class as MTL
import qualified Control.Exception as X

import SAWScript.Options (Options)
import SAWScript.TopLevel( TopLevelRO(..) )


-- | Haskeline-specific repl implementation.
Expand All @@ -38,22 +40,31 @@ import SAWScript.Options (Options)
-- handle this.
repl :: Maybe FilePath -> Options -> REPL () -> IO ()
repl mbBatch opts begin =
do settings <- setHistoryFile replSettings
runREPL isBatch opts (runInputTBehavior style settings body)
runREPL (isJust mbBatch) opts (replBody mbBatch begin)


replBody :: Maybe FilePath -> REPL () -> REPL ()
replBody mbBatch begin =
do settings <- MTL.liftIO (setHistoryFile replSettings)
enableSubshell (runInputTBehavior style settings body)
where
body = withInterrupt $ do
MTL.lift begin
loop

(isBatch,style) = case mbBatch of
Nothing -> (False,defaultBehavior)
Just path -> (True,useFile path)
style = case mbBatch of
Nothing -> defaultBehavior
Just path -> useFile path

enableSubshell m =
REPL $ \refs ->
do let ro' = (eTopLevelRO refs){ roSubshell = subshell (runInputT replSettings (withInterrupt loop)) }
unREPL m refs{ eTopLevelRO = ro' }

loop = do
prompt <- MTL.lift getPrompt
mb <- handleInterrupt (return (Just "")) (getInputLines prompt [])
case mb of

Just line
| Just cmd <- parseCommand findCommand line -> do
continue <- MTL.lift $ do
Expand Down
Loading

0 comments on commit 6cbe62a

Please sign in to comment.