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

Adding a feature to ask for abducts #213

Merged
merged 41 commits into from
Sep 1, 2022
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
39b6c36
Added offline get-abduct feature with an example file of how to use it
arjunvish Jul 5, 2022
be4ea86
Undid a small change we don't want to push
arjunvish Jul 5, 2022
99bfa5a
Unstable: trying to add get-abduct to the online solver
arjunvish Jul 7, 2022
1f8dcf9
Stable: added a way to get single abduct
arjunvish Jul 7, 2022
c1158a9
Working online version of get-abduct
arjunvish Jul 9, 2022
c82fb4f
Minor
arjunvish Jul 12, 2022
7f9999f
Refactored abduct commands so that variable name bound to abducts is …
arjunvish Jul 12, 2022
0dc8331
Updated comments
arjunvish Jul 12, 2022
a9accfa
Changed name from getAbduct to getAbducts
arjunvish Jul 12, 2022
b75ce45
Added ability for user to specify grammar
arjunvish Jul 18, 2022
39872ce
Turning off unsat-cores in cvc5 since unsat cores and abduction are n…
arjunvish Jul 18, 2022
a1326cb
Enabling unsat cores in cvc5
arjunvish Jul 28, 2022
d4b29ed
Temp. turning off unsat cores with cvc5
arjunvish Jul 30, 2022
58704b9
Adding a way to push and pop frame 2
arjunvish Aug 3, 2022
af3b58d
Adding a way to push and pop frame 2, part 2
arjunvish Aug 3, 2022
28b81cc
Turned unsat cores back on in cvc5
arjunvish Aug 4, 2022
2a1c375
Corrected an oops
arjunvish Aug 4, 2022
e91dfc1
Turning cvc5 unsat core back off
arjunvish Aug 4, 2022
99dba52
Unfinished work commented
arjunvish Aug 6, 2022
7fb74a9
Re-enabled unsat cores in cvc5, other minor changes to ready for merge
arjunvish Aug 8, 2022
ebe4c24
Partially finished formula to cfg function implemented
arjunvish Aug 10, 2022
dc24a2b
Removing all the grammar related stuff, readying for merge
arjunvish Aug 10, 2022
92cefb5
qMerge remote-tracking branch 'origin' into get-abduct
arjunvish Aug 10, 2022
ef05e76
Fixed some bugs
arjunvish Aug 10, 2022
99a6808
Housekeeping stuff
arjunvish Aug 12, 2022
7f315df
Addressing most of the PR feedback
arjunvish Aug 15, 2022
35fe9c6
Minor
arjunvish Aug 15, 2022
680ab0f
Correcting an oops from the last commits
arjunvish Aug 15, 2022
b55680f
Update what4/src/What4/Protocol/SMTWriter.hs
arjunvish Aug 29, 2022
8830292
Update what4/src/What4/Protocol/Online.hs
arjunvish Aug 29, 2022
295a0d1
Addressing some feedback
arjunvish Aug 29, 2022
3e0cdab
Fixing merge conflict
arjunvish Aug 29, 2022
fd0d547
Unstable - working on abduct test cases
arjunvish Aug 29, 2022
ef18c3f
Stable abduction tests
arjunvish Aug 30, 2022
09bf15c
Update what4/test/Abduct.hs
arjunvish Aug 30, 2022
fadc839
Update what4/test/Abduct.hs
arjunvish Aug 30, 2022
16f826f
Update what4/test/Abduct.hs
arjunvish Aug 30, 2022
a3116cd
Minor
arjunvish Aug 30, 2022
a97aafd
merging minor changes done from Github
arjunvish Aug 30, 2022
c79d6aa
Fixing imports in abduct test
arjunvish Aug 31, 2022
20e1e87
cvc5-1.0.2
arjunvish Aug 31, 2022
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
3 changes: 2 additions & 1 deletion .github/workflows/gen_matrix.pl
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@
main_version(yices, "2_6_4").
main_version(stp, "2_3_3").
main_version(cvc4, "1_8").
main_version(cvc5, "1_0_0").
main_version(cvc5, "1_0_1").
main_version(boolector, "3_2_2").
main_version(abc, "2021_12_30").

Expand Down Expand Up @@ -117,6 +117,7 @@
version(cvc4, "1_7").

version(cvc5, "1_0_0").
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
version(cvc5, "1_0_1").

version(boolector, "3_2_2").
version(boolector, "3_2_1").
Expand Down
10 changes: 5 additions & 5 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@ views or policies of the Department of Defense or the U.S. Government.

# Solver Compatibility

| Feature | ABC | Boolector | CVC4 | CVC5 | Dreal | STP | Yices | Z3 |
|---------------------------------------|-----|-------------|-----------|------|-------|-------------|----------|----------------------|
| Supported | yes | >= 3.2.0, ? | >= 1.8, ? | yes | yes | >= 2.3.3, ? | 2.6.x, ? | 4.8.8 -- 4.8.14 |
| goal timeouts | ? | yes | yes | yes | ? | yes | yes | ! (4.8.11 or 4.8.12) |
| strings with unicode and escape codes | ? | ? | >= 1.8 | yes | ? | ? | ? | >= 4.8.11 |
| Feature | ABC | Boolector | CVC4 | CVC5 | Dreal | STP | Yices | Z3 |
|---------------------------------------|-----|-------------|-----------|-------|-------|-------------|----------|----------------------|
| Supported | yes | >= 3.2.0, ? | >= 1.8, ? | 1.0.1 | yes | >= 2.3.3, ? | 2.6.x, ? | 4.8.8 -- 4.8.14 |
| goal timeouts | ? | yes | yes | yes | ? | yes | yes | ! (4.8.11 or 4.8.12) |
| strings with unicode and escape codes | ? | ? | >= 1.8 | yes | ? | ? | ? | >= 4.8.11 |
2 changes: 2 additions & 0 deletions what4/CHANGES.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@

* Added support for the cvc5 SMT solver.

* Added a `get-abduct` feature which is compatible with cvc5.

# 1.3 (April 2022)

* Allow building with GHC 9.2.
Expand Down
2 changes: 1 addition & 1 deletion what4/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -273,7 +273,7 @@ on as wide a collection of solvers as is reasonable.
- Z3 versions 4.8.7 through 4.8.12
- Yices 2.6.1 and 2.6.2
- CVC4 1.7 and 1.8
- CVC5 1.0.0
- CVC5 1.0.1
- Boolector 3.2.1 and 3.2.2
- STP 2.3.3
(However, note https://github.com/stp/stp/issues/363, which prevents
Expand Down
6 changes: 6 additions & 0 deletions what4/src/What4/ProblemFeatures.hs
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ module What4.ProblemFeatures
, useUnsatAssumptions
, useUninterpFunctions
, useDefinedFunctions
, useProduceAbducts
, hasProblemFeature
) where

Expand Down Expand Up @@ -130,6 +131,11 @@ useUninterpFunctions = ProblemFeatures 0x2000
useDefinedFunctions :: ProblemFeatures
useDefinedFunctions = ProblemFeatures 0x4000

-- | Indicates if the solver is able and configured to
-- produce abducts.
useProduceAbducts :: ProblemFeatures
useProduceAbducts = ProblemFeatures 0x8000

-- | Tests if one set of problem features subsumes another.
-- In particular, @hasProblemFeature x y@ is true iff
-- the set of features in @x@ is a superset of those in @y@.
Expand Down
47 changes: 47 additions & 0 deletions what4/src/What4/Protocol/Online.hs
Original file line number Diff line number Diff line change
Expand Up @@ -32,12 +32,15 @@ module What4.Protocol.Online
, reset
, inNewFrame
, inNewFrameWithVars
, inNewFrame2Open
, inNewFrame2Close
, check
, checkAndGetModel
, checkWithAssumptions
, checkWithAssumptionsAndModel
, getModel
, getUnsatCore
, getAbducts
, getUnsatAssumptions
, getSatResult
, checkSatisfiable
Expand Down Expand Up @@ -236,6 +239,39 @@ checkSatisfiable proc rsn p =
do assume conn p
check proc rsn

-- | `get-abuct nm t` queries the solver for the first abduct, which is returned
-- as an SMT function definition named `nm`. The remaining abducts are obtained
-- from the solver by successive invocations of the `get-abduct-next` command,
-- which return SMT functions bound to the same `nm` as the first. The name `nm`
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
-- is bound within the current assertion frame.
-- Note that this is an unstable API; we expect that the return type will change
-- to a parsed expression in the future
getAbducts ::
SMTReadWriter solver =>
SolverProcess scope solver ->
Int ->
Text ->
BoolExpr scope ->
IO [String]
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
getAbducts proc n nm t =
if (n > 0) then do
let conn = solverConn proc
unless (supportedFeatures conn `hasProblemFeature` useProduceAbducts) $
fail $ show $ pretty (smtWriterName conn) <+> pretty "is not configured to produce abducts"
f <- mkFormula conn t
-- get the first abduct using the get-abduct command
addCommandNoAck conn (getAbductCommand conn nm f)
abd1 <- smtAbductResult conn conn nm f
-- get the remaining abducts using get-abduct-next commands
if (n > 1) then do
let rest = n - 1
abdRest <- forM [1..rest] $ \_ -> do
addCommandNoAck conn (getAbductNextCommand conn)
smtAbductNextResult conn conn
return (abd1:abdRest)
else return [abd1]
else return []

-- | Check if the formula is satisifiable in the current
-- solver state. This is done in a
-- fresh frame, which is exited after the continuation
Expand Down Expand Up @@ -327,6 +363,17 @@ tryPop p =
inNewFrame :: (MonadIO m, MonadMask m, SMTReadWriter solver) => SolverProcess scope solver -> m a -> m a
inNewFrame p action = inNewFrameWithVars p [] action

-- For abduction, we want the final assertion to be a in a new frame, so that it
-- can be closed before asking for abducts. The following two commands allow frame 2
-- to be pushed and popped independently of other commands
-- | Open a second solver assumption frame.
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
inNewFrame2Open :: SMTReadWriter solver => SolverProcess scope solver -> IO ()
inNewFrame2Open sp = let c = solverConn sp in addCommand c (push2Command c)

-- | Close a second solver assumption frame.
inNewFrame2Close :: SMTReadWriter solver => SolverProcess scope solver -> IO ()
inNewFrame2Close sp = let c = solverConn sp in addCommand c (pop2Command c)

-- | Perform an action in the scope of a solver assumption frame, where the given
-- bound variables are considered free within that frame.
inNewFrameWithVars :: (MonadIO m, MonadMask m, SMTReadWriter solver)
Expand Down
6 changes: 6 additions & 0 deletions what4/src/What4/Protocol/SExp.hs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ module What4.Protocol.SExp
, asAtomList
, asNegAtomList
, skipSpaceOrNewline
, sExpToString
) where

#if !MIN_VERSION_base(4,13,0)
Expand Down Expand Up @@ -111,3 +112,8 @@ asAtomList (SApp xs) = go xs
go (SAtom a:ys) = (a:) <$> go ys
go _ = Nothing
asAtomList _ = Nothing

sExpToString :: SExp -> String
sExpToString (SAtom t) = Text.unpack t
sExpToString (SString t) = ('"' : Text.unpack t) ++ ['"']
sExpToString (SApp ss) = ('(' : Data.String.unwords (map sExpToString ss)) ++ [')']
54 changes: 54 additions & 0 deletions what4/src/What4/Protocol/SMTLib2.hs
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,10 @@ module What4.Protocol.SMTLib2
, writeCheckSat
, writeExit
, writeGetValue
, writeGetAbduct
, writeGetAbductNext
, runCheckSat
, runGetAbducts
, asSMT2Type
, setOption
, getVersion
Expand Down Expand Up @@ -676,6 +679,8 @@ instance SMTLib2Tweaks a => SMTWriter (Writer a) where

pushCommand _ = SMT2.push 1
popCommand _ = SMT2.pop 1
push2Command _ = SMT2.push 2
pop2Command _ = SMT2.pop 2
resetCommand _ = SMT2.resetAssertions
popManyCommands _ n = [SMT2.pop (toInteger n)]

Expand All @@ -684,6 +689,9 @@ instance SMTLib2Tweaks a => SMTWriter (Writer a) where

getUnsatAssumptionsCommand _ = SMT2.getUnsatAssumptions
getUnsatCoreCommand _ = SMT2.getUnsatCore
getAbductCommand _ nm e = SMT2.getAbduct nm e
getAbductNextCommand _ = SMT2.getAbductNext

setOptCommand _ = SMT2.setOption

declareCommand _proxy v argTypes retType =
Expand Down Expand Up @@ -755,6 +763,12 @@ setProduceModels w b = addCommand w $ SMT2.setProduceModels b
writeGetValue :: SMTLib2Tweaks a => WriterConn t (Writer a) -> [Term] -> IO ()
writeGetValue w l = addCommandNoAck w $ SMT2.getValue l

writeGetAbduct :: SMTLib2Tweaks a => WriterConn t (Writer a) -> Text -> Term -> IO ()
writeGetAbduct w nm p = addCommandNoAck w $ SMT2.getAbduct nm p

writeGetAbductNext :: SMTLib2Tweaks a => WriterConn t (Writer a) -> IO ()
writeGetAbductNext w = addCommandNoAck w SMT2.getAbductNext

parseBoolSolverValue :: MonadFail m => SExp -> m Bool
parseBoolSolverValue (SAtom "true") = return True
parseBoolSolverValue (SAtom "false") = return False
Expand Down Expand Up @@ -891,6 +905,32 @@ runGetValue s e = do
_ -> Nothing
getLimitedSolverResponse "get value" valRsp (sessionWriter s) (SMT2.getValue [e])

-- | runGetAbducts s nm p n, returns n formulas (as strings) the disjunction of which entails p (along with all
-- the assertions in the context)
runGetAbducts :: SMTLib2Tweaks a
=> Session t a
-> Int
-> Text
-> Term
-> IO [String]
runGetAbducts s n nm p =
if (n > 0) then do
writeGetAbduct (sessionWriter s) nm p
let valRsp = \x -> case x of
-- SMT solver returns `(define-fun nm () Bool X)` where X is the abduct, we discard everything but the abduct
AckSuccessSExp (SApp (_ : _ : _ : _ : abduct)) -> Just $ Data.String.unwords (map sExpToString abduct)
_ -> Nothing
-- get first abduct using the get-abduct command
abd1 <- getLimitedSolverResponse "get abduct" valRsp (sessionWriter s) (SMT2.getAbduct nm p)
if (n > 1) then do
let rest = n - 1
replicateM_ rest $ writeGetAbductNext (sessionWriter s)
-- get the rest of the abducts using the get-abduct-next command
abdRest <- forM [1..rest] $ \_ -> getLimitedSolverResponse "get abduct next" valRsp (sessionWriter s) (SMT2.getAbduct nm p)
return (abd1:abdRest)
else return [abd1]
else return []

-- | This function runs a check sat command
runCheckSat :: forall b t a.
SMTLib2Tweaks b
Expand Down Expand Up @@ -938,6 +978,20 @@ instance SMTLib2Tweaks a => SMTReadWriter (Writer a) where
cmd = getUnsatCoreCommand p
in getLimitedSolverResponse "unsat core" unsatCoreRsp s cmd

smtAbductResult p s nm t =
let abductRsp = \case
AckSuccessSExp (SApp (_ : _ : _ : _ : abduct)) -> Just $ Data.String.unwords (map sExpToString abduct)
_ -> Nothing
cmd = getAbductCommand p nm t
in getLimitedSolverResponse "get abduct" abductRsp s cmd

smtAbductNextResult p s =
let abductRsp = \case
AckSuccessSExp (SApp (_ : _ : _ : _ : abduct)) -> Just $ Data.String.unwords (map sExpToString abduct)
_ -> Nothing
cmd = getAbductNextCommand p
in getLimitedSolverResponse "get abduct next" abductRsp s cmd


smtAckResult :: AcknowledgementAction t (Writer a)
smtAckResult = AckAction $ getLimitedSolverResponse "get ack" $ \case
Expand Down
2 changes: 1 addition & 1 deletion what4/src/What4/Protocol/SMTLib2/Parse.hs
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ module What4.Protocol.SMTLib2.Parse
( -- * CheckSatResponse
CheckSatResponse(..)
, readCheckSatResponse
-- * GetModelResonse
-- * GetModelResponse
, GetModelResponse
, readGetModelResponse
, ModelResponse(..)
Expand Down
14 changes: 12 additions & 2 deletions what4/src/What4/Protocol/SMTLib2/Syntax.hs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,8 @@ module What4.Protocol.SMTLib2.Syntax
, assertNamed
, getUnsatAssumptions
, getUnsatCore
, getAbduct
, getAbductNext
-- * Logic
, Logic(..)
, qf_bv
Expand Down Expand Up @@ -296,7 +298,7 @@ or [] = true
or [x] = x
or l = term_app "or" l

-- | Disjunction of all terms
-- | Xor of all terms
xor :: [Term] -> Term
xor l@(_:_:_) = term_app "xor" l
xor _ = error "xor expects two or more arguments."
Expand All @@ -305,7 +307,7 @@ xor _ = error "xor expects two or more arguments."
eq :: [Term] -> Term
eq = chain_app "="

-- | Construct a chainable term with the givne relation
-- | Construct a chainable term with the given relation
--
-- @pairwise_app p [x1, x2, ..., xn]@ is equivalent to
-- \forall_{i,j} p x_i x_j@.
Expand Down Expand Up @@ -826,6 +828,14 @@ getUnsatAssumptions = Cmd "(get-unsat-assumptions)"
getUnsatCore :: Command
getUnsatCore = Cmd "(get-unsat-core)"

-- | Get an abduct that entails the formula, and bind it to the name
getAbduct :: Text -> Term -> Command
getAbduct nm p = Cmd $ "(get-abduct " <> Builder.fromText nm <> " " <> renderTerm p <> ")"

-- | Get the next command, called after a get-abduct command
getAbductNext :: Command
getAbductNext = Cmd "(get-abduct-next)"

-- | Get the values associated with the terms from the last call to @check-sat@.
getValue :: [Term] -> Command
getValue values = Cmd $ app "get-value" [builder_list (renderTerm <$> values)]
Expand Down
22 changes: 20 additions & 2 deletions what4/src/What4/Protocol/SMTWriter.hs
Original file line number Diff line number Diff line change
Expand Up @@ -867,12 +867,18 @@ class (SupportTermOps (Term h)) => SMTWriter h where
-- later reporting unsatisfiable cores).
assertNamedCommand :: f h -> Term h -> Text -> Command h

-- | Push 1 new scope
-- | Generates command `(push 1)` that opens the corresponding assertion frame
pushCommand :: f h -> Command h

-- | Pop 1 existing scope
-- | Generates command `(pop 1)` that closes the corresponding assertion frame
popCommand :: f h -> Command h

-- | Generates command `(push 2)` that opens the corresponding assertion frame
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
push2Command :: f h -> Command h

-- | Generates command `(pop 2)` that closes the corresponding assertion frame
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
pop2Command :: f h -> Command h
arjunvish marked this conversation as resolved.
Show resolved Hide resolved

-- | Pop several scopes.
popManyCommands :: f h -> Int -> [Command h]
popManyCommands w n = replicate n (popCommand w)
Expand All @@ -898,6 +904,12 @@ class (SupportTermOps (Term h)) => SMTWriter h where
-- `checkCommand`.
getUnsatCoreCommand :: f h -> Command h

-- | Ask the solver to return an abduct
getAbductCommand :: f h -> Text -> Term h -> Command h

-- | Ask the solver for the next abduct, used after a get-abduct command
getAbductNextCommand :: f h -> Command h

-- | Set an option/parameter.
setOptCommand :: f h -> Text -> Text -> Command h

Expand Down Expand Up @@ -2982,6 +2994,12 @@ class SMTWriter h => SMTReadWriter h where
-- These correspond to previously-named assertions.
smtUnsatCoreResult :: f h -> WriterConn t h -> IO [Text]

-- | Parse an abduct returned by the get-abduct command
smtAbductResult :: f h -> WriterConn t h -> Text -> Term h -> IO String

-- | Parse an abduct returned by the get-abduct-next command
smtAbductNextResult :: f h -> WriterConn t h -> IO String

-- | Parse a list of names of assumptions that form an unsatisfiable core.
-- The boolean indicates the polarity of the atom: true for an ordinary
-- atom, false for a negated atom.
Expand Down
13 changes: 11 additions & 2 deletions what4/src/What4/Solver/CVC5.hs
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,7 @@ cvc5Features = useComputableReals
.|. useDefinedFunctions
.|. useBitvectors
.|. useQuantifiers
.|. useProduceAbducts

writeMultiAsmpCVC5SMT2File
:: ExprBuilder t st fs
Expand Down Expand Up @@ -190,6 +191,8 @@ instance SMT2.SMTLib2GenericSolver CVC5 where
SMT2.setLogic writer Syntax.allLogic
-- Tell cvc5 to produce models
SMT2.setProduceModels writer True
-- Tell cvc5 to produce abducts
SMT2.setOption writer "produce-abducts" "true"

runCVC5InOverride
:: ExprBuilder t st fs
Expand Down Expand Up @@ -224,9 +227,15 @@ setInteractiveLogicAndOptions writer = do
SMT2.setOption writer "produce-models" "true"
-- Tell cvc5 to make declarations global, so they are not removed by 'pop' commands
SMT2.setOption writer "global-declarations" "true"
-- FIXME: in cvc5-1.0.1, unsat-cores and produce-abducts are incompatible, but
-- cvc5-1.0.2 will fix this, and we can uncomment this once we that is released.
-- Closing issue #214 will fix this.
arjunvish marked this conversation as resolved.
Show resolved Hide resolved
-- Tell cvc5 to compute UNSAT cores, if that feature is enabled
when (supportedFeatures writer `hasProblemFeature` useUnsatCores) $ do
SMT2.setOption writer "produce-unsat-cores" "true"
-- when (supportedFeatures writer `hasProblemFeature` useUnsatCores) $ do
-- SMT2.setOption writer "produce-unsat-cores" "true"
-- Tell cvc5 to produce abducts, if that feature is enabled
when (supportedFeatures writer `hasProblemFeature` useProduceAbducts) $ do
SMT2.setOption writer "produce-abducts" "true"
-- Tell cvc5 to use all supported logics.
SMT2.setLogic writer Syntax.allLogic

Expand Down
Loading