Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove IORef indirection from TheoremDB #1882

Merged
merged 8 commits into from
Jun 30, 2023
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion saw-remote-api/src/SAWServer.hs
Original file line number Diff line number Diff line change
Expand Up @@ -207,7 +207,7 @@ initialState readFileFn =
halloc <- Crucible.newHandleAllocator
jvmTrans <- CJ.mkInitialJVMContext halloc
cwd <- getCurrentDirectory
db <- newTheoremDB
let db = newTheoremDB
let ro = TopLevelRO
{ roJavaCodebase = jcb
, roOptions = opts
Expand Down
12 changes: 6 additions & 6 deletions src/SAWScript/Builtins.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2037,7 +2037,7 @@ core_axiom input =
t <- parseCore input
p <- io (termToProp sc t)
db <- SV.getTheoremDB
thm <- io (admitTheorem db "core_axiom" p pos "core_axiom")
(thm, _db') <- io (admitTheorem db "core_axiom" p pos "core_axiom")
samcowger marked this conversation as resolved.
Show resolved Hide resolved
SV.returnProof thm

core_thm :: String -> TopLevel Theorem
Expand All @@ -2046,15 +2046,15 @@ core_thm input =
sc <- getSharedContext
pos <- SV.getPosition
db <- SV.getTheoremDB
thm <- io (proofByTerm sc db t pos "core_thm")
(thm, _db') <- io (proofByTerm sc db t pos "core_thm")
samcowger marked this conversation as resolved.
Show resolved Hide resolved
SV.returnProof thm

specialize_theorem :: Theorem -> [TypedTerm] -> TopLevel Theorem
specialize_theorem thm ts =
do sc <- getSharedContext
db <- SV.getTheoremDB
pos <- SV.getPosition
thm' <- io (specializeTheorem sc db pos "specialize_theorem" thm (map ttTerm ts))
(thm', _db') <- io (specializeTheorem sc db pos "specialize_theorem" thm (map ttTerm ts))
samcowger marked this conversation as resolved.
Show resolved Hide resolved
SV.returnProof thm'

get_opt :: Int -> TopLevel String
Expand Down Expand Up @@ -2427,7 +2427,7 @@ summarize_verification =
lspecs = [ s | SV.VLLVMCrucibleMethodSpec s <- values ]
thms = [ t | SV.VTheorem t <- values ]
db <- SV.getTheoremDB
summary <- io (computeVerificationSummary db jspecs lspecs thms)
let summary = computeVerificationSummary db jspecs lspecs thms
opts <- fmap (SV.sawPPOpts . rwPPOpts) getTopLevelRW
nenv <- io . scGetNamingEnv =<< getSharedContext
io $ putStrLn $ prettyVerificationSummary opts nenv summary
Expand All @@ -2439,7 +2439,7 @@ summarize_verification_json fpath =
lspecs = [ s | SV.VLLVMCrucibleMethodSpec s <- values ]
thms = [ t | SV.VTheorem t <- values ]
db <- SV.getTheoremDB
summary <- io (computeVerificationSummary db jspecs lspecs thms)
let summary = computeVerificationSummary db jspecs lspecs thms
io (writeFile fpath (jsonVerificationSummary summary))

writeVerificationSummary :: TopLevel ()
Expand All @@ -2450,7 +2450,7 @@ writeVerificationSummary = do
let jspecs = [ s | SV.VJVMMethodSpec s <- values ]
lspecs = [ s | SV.VLLVMCrucibleMethodSpec s <- values ]
thms = [ t | SV.VTheorem t <- values ]
summary <- io (computeVerificationSummary db jspecs lspecs thms)
summary = computeVerificationSummary db jspecs lspecs thms
opts <- roOptions <$> getTopLevelRO
dir <- roInitWorkDir <$> getTopLevelRO
nenv <- io . scGetNamingEnv =<< getSharedContext
Expand Down
2 changes: 1 addition & 1 deletion src/SAWScript/Interpreter.hs
Original file line number Diff line number Diff line change
Expand Up @@ -453,7 +453,7 @@ buildTopLevelEnv proxy opts =
ss <- basic_ss sc
jcb <- JCB.loadCodebase (jarList opts) (classPath opts) (javaBinDirs opts)
currDir <- getCurrentDirectory
thmDB <- newTheoremDB
let thmDB = newTheoremDB
Crucible.withHandleAllocator $ \halloc -> do
let ro0 = TopLevelRO
{ roJavaCodebase = jcb
Expand Down
129 changes: 66 additions & 63 deletions src/SAWScript/Proof.hs
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,6 @@ module SAWScript.Proof

import qualified Control.Monad.Fail as F
import Control.Monad.Except
import Data.IORef
import qualified Data.Foldable as Fold
import Data.List
import Data.Maybe (fromMaybe)
Expand Down Expand Up @@ -902,25 +901,21 @@ data Theorem =
data TheoremDB =
TheoremDB
-- TODO, maybe this should be a summary or something simpler?
{ theoremMap :: IORef (Map.Map TheoremNonce Theorem)
{ theoremMap :: Map.Map TheoremNonce Theorem
}

newTheoremDB :: IO TheoremDB
newTheoremDB = TheoremDB <$> newIORef mempty
newTheoremDB :: TheoremDB
newTheoremDB = TheoremDB mempty
samcowger marked this conversation as resolved.
Show resolved Hide resolved

recordTheorem :: TheoremDB -> Theorem -> IO Theorem
recordTheorem db thm@Theorem{ _thmNonce = n } =
do modifyIORef (theoremMap db) (Map.insert n thm)
return thm
recordTheorem :: TheoremDB -> Theorem -> TheoremDB
recordTheorem db thm@Theorem{ _thmNonce = n } = TheoremDB (Map.insert n thm (theoremMap db))

-- | Given a set of root values, find all the theorems in this database
-- that are transitively used in the proofs of those theorems.
-- This function will panic if any of the roots or reachable theorems
-- are not found in the database.
reachableTheorems :: TheoremDB -> Set TheoremNonce -> IO (Map TheoremNonce Theorem)
reachableTheorems db roots =
do m <- readIORef (theoremMap db)
pure $! Set.foldl' (loop m) mempty roots
reachableTheorems :: TheoremDB -> Set TheoremNonce -> Map TheoremNonce Theorem
reachableTheorems db roots = Set.foldl' (loop (theoremMap db)) mempty roots

where
loop m visited curr
Expand All @@ -945,7 +940,7 @@ reachableTheorems db roots =
validateTheorem :: SharedContext -> TheoremDB -> Theorem -> IO ()

validateTheorem sc db Theorem{ _thmProp = p, _thmEvidence = e, _thmDepends = thmDep } =
do hyps <- Map.keysSet <$> readIORef (theoremMap db)
do let hyps = Map.keysSet (theoremMap db)
(deps,_) <- checkEvidence sc e p
unless (Set.isSubsetOf deps thmDep && Set.isSubsetOf thmDep hyps)
(fail $ unlines ["Theorem failed to declare its dependencies correctly"
Expand Down Expand Up @@ -1148,24 +1143,26 @@ structuralEvidence _sqt (StructuralEvidence sqt' e) = StructuralEvidence sqt' e
structuralEvidence sqt e = StructuralEvidence sqt e

-- | Construct a theorem directly via a proof term.
proofByTerm :: SharedContext -> TheoremDB -> Term -> Pos -> Text -> IO Theorem
proofByTerm :: SharedContext -> TheoremDB -> Term -> Pos -> Text -> IO (Theorem, TheoremDB)
proofByTerm sc db prf loc rsn =
do ty <- scTypeOf sc prf
p <- termToProp sc ty
n <- freshNonce globalNonceGenerator
recordTheorem db
Theorem
{ _thmProp = p
, _thmStats = mempty
, _thmEvidence = ProofTerm prf
, _thmLocation = loc
, _thmProgramLoc = Nothing
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = mempty
, _thmElapsedTime = 0
, _thmSummary = ProvedTheorem mempty
}
let thm =
Theorem
{ _thmProp = p
, _thmStats = mempty
, _thmEvidence = ProofTerm prf
, _thmLocation = loc
, _thmProgramLoc = Nothing
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = mempty
, _thmElapsedTime = 0
, _thmSummary = ProvedTheorem mempty
}
let db' = recordTheorem db thm
pure (thm, db')

-- | Construct a theorem directly from a proposition and evidence
-- for that proposition. The evidence will be validated to
Expand All @@ -1180,31 +1177,33 @@ constructTheorem ::
Maybe ProgramLoc ->
Text ->
NominalDiffTime ->
IO Theorem
IO (Theorem, TheoremDB)
constructTheorem sc db p e loc ploc rsn elapsed =
do (deps,sy) <- checkEvidence sc e p
n <- freshNonce globalNonceGenerator
recordTheorem db
Theorem
{ _thmProp = p
, _thmStats = mempty
, _thmEvidence = e
, _thmLocation = loc
, _thmProgramLoc = ploc
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = deps
, _thmElapsedTime = elapsed
, _thmSummary = sy
}
let thm =
Theorem
{ _thmProp = p
, _thmStats = mempty
, _thmEvidence = e
, _thmLocation = loc
, _thmProgramLoc = ploc
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = deps
, _thmElapsedTime = elapsed
, _thmSummary = sy
}
let db' = recordTheorem db thm
pure (thm, db')


-- | Given a theorem with quantified variables, build a new theorem that
-- specializes the leading quantifiers with the given terms.
-- This will fail if the given terms to not match the quantifier structure
-- of the given theorem.
specializeTheorem :: SharedContext -> TheoremDB -> Pos -> Text -> Theorem -> [Term] -> IO Theorem
specializeTheorem _sc _db _loc _rsn thm [] = return thm
specializeTheorem :: SharedContext -> TheoremDB -> Pos -> Text -> Theorem -> [Term] -> IO (Theorem, TheoremDB)
specializeTheorem _sc db _loc _rsn thm [] = return (thm, db)
specializeTheorem sc db loc rsn thm ts =
do res <- specializeProp sc (_thmProp thm) ts
case res of
Expand All @@ -1231,22 +1230,24 @@ admitTheorem ::
Prop ->
Pos ->
Text ->
IO Theorem
IO (Theorem, TheoremDB)
admitTheorem db msg p loc rsn =
do n <- freshNonce globalNonceGenerator
recordTheorem db
Theorem
{ _thmProp = p
, _thmStats = solverStats "ADMITTED" (propSize p)
, _thmEvidence = Admitted msg loc (propToSequent p)
, _thmLocation = loc
, _thmProgramLoc = Nothing
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = mempty
, _thmElapsedTime = 0
, _thmSummary = AdmittedTheorem msg
}
let thm =
Theorem
{ _thmProp = p
, _thmStats = solverStats "ADMITTED" (propSize p)
, _thmEvidence = Admitted msg loc (propToSequent p)
, _thmLocation = loc
, _thmProgramLoc = Nothing
, _thmReason = rsn
, _thmNonce = n
, _thmDepends = mempty
, _thmElapsedTime = 0
, _thmSummary = AdmittedTheorem msg
}
let db' = recordTheorem db thm
pure (thm, db')

-- | Construct a theorem that an external solver has proved.
solverTheorem ::
Expand All @@ -1256,10 +1257,11 @@ solverTheorem ::
Pos ->
Text ->
NominalDiffTime ->
IO Theorem
IO TheoremDB
samcowger marked this conversation as resolved.
Show resolved Hide resolved
solverTheorem db p stats loc rsn elapsed =
do n <- freshNonce globalNonceGenerator
recordTheorem db
pure $
recordTheorem db
Theorem
{ _thmProp = p
, _thmStats = stats
Expand Down Expand Up @@ -1782,7 +1784,7 @@ finishProof ::
ProofState ->
Bool {- ^ should we record the theorem in the database? -} ->
Bool {- ^ do we need to normalize the sequent to match the final goal ? -} ->
IO ProofResult
IO (ProofResult, TheoremDB)
finishProof sc db conclProp
ps@(ProofState gs (concl,loc,ploc,rsn) stats _ checkEv start)
recordThm useSequentGoals =
Expand All @@ -1795,7 +1797,7 @@ finishProof sc db conclProp
(deps,sy) <- checkEvidence sc e' conclProp
n <- freshNonce globalNonceGenerator
end <- getCurrentTime
thm <- (if recordThm then recordTheorem db else return)
let theorem =
Theorem
{ _thmProp = conclProp
, _thmStats = stats
Expand All @@ -1808,9 +1810,10 @@ finishProof sc db conclProp
, _thmElapsedTime = diffUTCTime end start
, _thmSummary = sy
}
pure (ValidProof stats thm)
let db' = if recordThm then recordTheorem db theorem else db
pure (ValidProof stats theorem, db')
_ : _ ->
pure (UnfinishedProof ps)
pure (UnfinishedProof ps, db)

-- | A type describing counterexamples.
type CEX = [(ExtCns Term, FirstOrderValue)]
Expand Down
17 changes: 13 additions & 4 deletions src/SAWScript/Value.hs
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ import qualified Control.Exception as X
import qualified System.IO.Error as IOError
import Control.Monad.IO.Class (MonadIO, liftIO)
import Control.Monad.Reader (ReaderT(..), ask, asks, local)
import Control.Monad.State (StateT(..), gets)
import Control.Monad.State (StateT(..), gets, modify)
import Control.Monad.Trans.Class (MonadTrans(lift))
import Data.IORef
import Data.Foldable(foldrM)
Expand Down Expand Up @@ -692,7 +692,10 @@ getJavaCodebase :: TopLevel JSS.Codebase
getJavaCodebase = TopLevel_ (asks roJavaCodebase)

getTheoremDB :: TopLevel TheoremDB
getTheoremDB = TopLevel_ (rwTheoremDB <$> get)
getTheoremDB = gets rwTheoremDB

putTheoremDB :: TheoremDB -> TopLevel ()
putTheoremDB db = modifyTopLevelRW (\tl -> tl { rwTheoremDB = db })

getOptions :: TopLevel Options
getOptions = TopLevel_ (asks roOptions)
Expand Down Expand Up @@ -728,6 +731,9 @@ getTopLevelRW = get
putTopLevelRW :: TopLevelRW -> TopLevel ()
putTopLevelRW rw = put rw

modifyTopLevelRW :: (TopLevelRW -> TopLevelRW) -> TopLevel ()
modifyTopLevelRW = modify

returnProof :: IsValue v => v -> TopLevel v
returnProof v = recordProof v >> return v

Expand Down Expand Up @@ -876,8 +882,11 @@ runProofScript (ProofScript m) concl gl ploc rsn recordThm useSequentGoals =
Left (stats,cex) -> return (SAWScript.Proof.InvalidProof stats cex pstate)
Right _ ->
do sc <- getSharedContext
db <- rwTheoremDB <$> getTopLevelRW
io (finishProof sc db concl pstate recordThm useSequentGoals)
db <- getTheoremDB
(thmResult, db') <- io (finishProof sc db concl pstate recordThm useSequentGoals)
putTheoremDB db'
pure thmResult


scriptTopLevel :: TopLevel a -> ProofScript a
scriptTopLevel m = ProofScript (lift (lift m))
Expand Down
8 changes: 4 additions & 4 deletions src/SAWScript/VerificationSummary.hs
Original file line number Diff line number Diff line change
Expand Up @@ -60,14 +60,14 @@ vsTheoremSolvers = Set.unions . map getSolvers . vsTheorems
vsAllSolvers :: VerificationSummary -> Set.Set String
vsAllSolvers vs = Set.union (vsVerifSolvers vs) (vsTheoremSolvers vs)

computeVerificationSummary :: TheoremDB -> [JVMTheorem] -> [LLVMTheorem] -> [Theorem] -> IO VerificationSummary
computeVerificationSummary :: TheoremDB -> [JVMTheorem] -> [LLVMTheorem] -> [Theorem] -> VerificationSummary
computeVerificationSummary db js ls thms =
do let roots = mconcat (
let roots = mconcat (
[ vcDeps vc | j <- js, vc <- j^.psVCStats ] ++
[ vcDeps vc | CMSLLVM.SomeLLVM l <- ls, vc <- l^.psVCStats ] ++
[ Set.singleton (thmNonce t) | t <- thms ])
thms' <- Map.elems <$> reachableTheorems db roots
pure (VerificationSummary js ls thms')
thms' = Map.elems (reachableTheorems db roots)
in VerificationSummary js ls thms'

-- TODO: we could make things instances of a ToJSON typeclass instead of using the two methods below.
msToJSON :: forall ext . Pretty (MethodId ext) => CMS.ProvedSpec ext -> Value
Expand Down