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

Revert #1169 #1182

Merged
merged 1 commit into from
Mar 12, 2024
Merged
Show file tree
Hide file tree
Changes from all 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
5 changes: 0 additions & 5 deletions crucible/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,8 +1,3 @@
# next -- TBA

* Rename `Lang.Crucible.Backend.popFrame` to `popFrameOrPanic`,
provide helpers such as `popFrame` to manage assumptions without `panic`ing.

# 0.7 -- 2024-02-05

* Add `TypedOverride`, `SomeTypedOverride`, and `runTypedOverride` to
Expand Down
126 changes: 49 additions & 77 deletions crucible/src/Lang/Crucible/Backend/AssumptionStack.hs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ solvers.
-}
{-# LANGUAGE ConstraintKinds #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE MultiParamTypeClasses #-}
Expand All @@ -33,15 +32,13 @@ module Lang.Crucible.Backend.AssumptionStack
, FrameIdentifier
, AssumptionFrame(..)
, AssumptionFrames(..)
, AssumptionStack
, AssumptionStack(..)
-- ** Manipulating assumption stacks
, initAssumptionStack
, saveAssumptionStack
, restoreAssumptionStack
, pushFrame
, PopFrameError(..)
, popFrame
, popFrameOrPanic
, popFrameAndGoals
, popFramesUntil
, resetStack
Expand All @@ -57,13 +54,10 @@ module Lang.Crucible.Backend.AssumptionStack
) where

import Control.Exception (bracketOnError)
import Data.Functor ((<&>))
import qualified Data.Foldable as F
import Data.IORef
import Data.Parameterized.Nonce

import qualified Prettyprinter as PP

import Lang.Crucible.Backend.ProofGoals
import Lang.Crucible.Panic (panic)

Expand Down Expand Up @@ -179,10 +173,12 @@ popFramesUntil ident stk = atomicModifyIORef' (proofObligations stk) (go 1)
where
go n gc =
case gcPop gc of
Left frm
| ident == poppedFrameId frm -> (gc',n)
Left (ident', _assumes, mg, gc1)
| ident == ident' -> (gc',n)
| otherwise -> go (n+1) gc'
where gc' = collectPoppedGoals frm
where gc' = case mg of
Nothing -> gc1
Just g -> gcAddGoals g gc1
Right _ ->
panic "AssumptionStack.popFrameUntil"
[ "Frame not found in stack."
Expand All @@ -191,84 +187,60 @@ popFramesUntil ident stk = atomicModifyIORef' (proofObligations stk) (go 1)

showFrameId (FrameIdentifier x) = show x

-- | Call 'gcPop' on the 'proofObligations' of this 'AssumptionStack'
popFrameUnchecked ::
Monoid asmp =>
AssumptionStack asmp ast ->
IO (Either (PoppedFrame asmp ast) (Maybe (Goals asmp ast)))
popFrameUnchecked stk =
atomicModifyIORef' (proofObligations stk) $ \gc ->
case gcPop gc of
Left frm -> (collectPoppedGoals frm, Left frm)
Right mgs -> (gc, Right mgs)

data PopFrameError
= NoFrames
-- | Expected, actual
| WrongFrame FrameIdentifier FrameIdentifier

instance PP.Pretty PopFrameError where
pretty =
\case
NoFrames -> PP.pretty "Pop with no push in goal collector."
WrongFrame expected actual ->
PP.hsep
[ PP.pretty "Mismatch in assumption stack frames."
, PP.pretty "Expected ident:"
, PP.viaShow expected
, PP.pretty "Current frame:"
, PP.viaShow actual
]

instance Show PopFrameError where
show = show . PP.pretty

-- | Pop a previously-pushed assumption frame from the stack.
-- All assumptions in that frame will be forgotten. The
-- assumptions contained in the popped frame are returned.
--
-- Returns 'Left' if there are no frames on the stack, or if the top frame
-- doesn\'t have the provided 'FrameIdentifier'.
popFrame ::
Monoid asmp =>
FrameIdentifier ->
AssumptionStack asmp ast ->
IO (Either PopFrameError (PoppedFrame asmp ast))
popFrame :: Monoid asmp => FrameIdentifier -> AssumptionStack asmp ast -> IO asmp
popFrame ident stk =
popFrameUnchecked stk <&>
\case
Left frm
| ident == poppedFrameId frm -> Right frm
| otherwise -> Left (WrongFrame ident (poppedFrameId frm))
Right _ -> Left NoFrames
atomicModifyIORef' (proofObligations stk) $ \gc ->
case gcPop gc of
Left (ident', assumes, mg, gc1)
| ident == ident' ->
let gc' = case mg of
Nothing -> gc1
Just g -> gcAddGoals g gc1
in (gc', assumes)
| otherwise ->
panic "AssumptionStack.popFrame"
[ "Push/pop mismatch in assumption stack!"
, "*** Current frame: " ++ showFrameId ident
, "*** Expected ident: " ++ showFrameId ident'
]
Right _ ->
panic "AssumptionStack.popFrame"
[ "Pop with no push in goal collector."
, "*** Current frame: " ++ showFrameId ident
]

where
showFrameId (FrameIdentifier x) = show x

-- | Pop a previously-pushed assumption frame from the stack.
-- All assumptions in that frame will be forgotten. The
-- assumptions contained in the popped frame are returned.
--
-- Panics if there are no frames on the stack, or if the top frame doesn\'t
-- have the provided 'FrameIdentifier'.
popFrameOrPanic :: Monoid asmp => FrameIdentifier -> AssumptionStack asmp ast -> IO asmp
popFrameOrPanic ident stk =
popFrame ident stk <&>
\case
Left err -> panic "AssumptionStack.popFrame" [show err]
Right frm -> poppedAssumptions frm

-- | Pop the assumptions and goals from the top frame.
--
-- Panics if there are no frames on the stack, or if the top frame doesn\'t
-- have the provided 'FrameIdentifier'.
popFrameAndGoals ::
Monoid asmp =>
FrameIdentifier ->
AssumptionStack asmp ast ->
IO (asmp, Maybe (Goals asmp ast))
popFrameAndGoals ident stk =
popFrame ident stk <&>
\case
Left err -> panic "AssumptionStack.popFrameAndGoals" [show err]
Right frm -> (poppedAssumptions frm, poppedGoals frm)
atomicModifyIORef' (proofObligations stk) $ \gc ->
case gcPop gc of
Left (ident', assumes, mg, gc1)
| ident == ident' -> (gc1, (assumes, mg))
| otherwise ->
panic "AssumptionStack.popFrameAndGoals"
[ "Push/pop mismatch in assumption stack!"
, "*** Current frame: " ++ showFrameId ident
, "*** Expected ident: " ++ showFrameId ident'
]
Right _ ->
panic "AssumptionStack.popFrameAndGoals"
[ "Pop with no push in goal collector."
, "*** Current frame: " ++ showFrameId ident
]

where
showFrameId (FrameIdentifier x) = show x


-- | Run an action in the scope of a fresh assumption frame.
-- The frame will be popped and returned on successful
Expand All @@ -280,5 +252,5 @@ inFreshFrame stk action =
(pushFrame stk)
(\ident -> popFrame ident stk)
(\ident -> do x <- action
frm <- popFrameOrPanic ident stk
frm <- popFrame ident stk
return (frm, x))
2 changes: 1 addition & 1 deletion crucible/src/Lang/Crucible/Backend/Online.hs
Original file line number Diff line number Diff line change
Expand Up @@ -579,7 +579,7 @@ instance (IsSymInterface (B.ExprBuilder scope st fs), OnlineSolver solver) =>

popAssumptionFrame bak ident =
-- NB, pop the frame whether or not the solver pop succeeds
do frm <- popFrameOrPanic ident (assumptionStack bak)
do frm <- popFrame ident (assumptionStack bak)
withSolverProcess bak (pure ()) pop
return frm

Expand Down
46 changes: 16 additions & 30 deletions crucible/src/Lang/Crucible/Backend/ProofGoals.hs
Original file line number Diff line number Diff line change
Expand Up @@ -28,8 +28,6 @@ module Lang.Crucible.Backend.ProofGoals
, traverseGoalCollectorWithAssumptions

-- ** Context management
, PoppedFrame(..)
, collectPoppedGoals
, gcAddAssumes, gcProve
, gcPush, gcPop, gcAddGoals,

Expand Down Expand Up @@ -267,35 +265,23 @@ gcAddAssumes :: Monoid asmp => asmp -> GoalCollector asmp goal -> GoalCollector
gcAddAssumes as' (CollectingAssumptions as gls) = CollectingAssumptions (as <> as') gls
gcAddAssumes as' gls = CollectingAssumptions as' gls

-- | A frame, popped by 'gcPop'
data PoppedFrame asmp goal
= PoppedFrame
{ -- | The frame identifier of the popped frame
poppedFrameId :: !FrameIdentifier
-- | The assumptions that were forgotten by the pop
, poppedAssumptions :: asmp
-- | Any proof goals that were generated since the frame push
, poppedGoals :: Maybe (Goals asmp goal)
-- | The state of the collector before the push.
, poppedCollector :: GoalCollector asmp goal
}

-- | Retrieve the 'GoalCollector' from a 'PoppedFrame', adding the
-- 'poppedGoals' (if there are any) via 'gcAddGoals'.
collectPoppedGoals :: PoppedFrame asmp goal -> GoalCollector asmp goal
collectPoppedGoals frm =
case poppedGoals frm of
Nothing -> poppedCollector frm
Just goals -> gcAddGoals goals (poppedCollector frm)

{- | Pop to the last push, or all the way to the top, if there were no more pushes.
If the result is 'Left', then we popped until an explicitly marked push.
If the result is 'Left', then we popped until an explicitly marked push;
in that case we return:

1. the frame identifier of the popped frame,
2. the assumptions that were forgotten,
3. any proof goals that were generated since the frame push, and
4. the state of the collector before the push.

If the result is 'Right', then we popped all the way to the top, and the
result is the goal tree, or 'Nothing' if there were no goals. -}

gcPop ::
Monoid asmp =>
GoalCollector asmp goal ->
Either (PoppedFrame asmp goal) (Maybe (Goals asmp goal))
Either (FrameIdentifier, asmp, Maybe (Goals asmp goal), GoalCollector asmp goal)
(Maybe (Goals asmp goal))
gcPop = go Nothing mempty
where

Expand All @@ -307,7 +293,7 @@ gcPop = go Nothing mempty
Right (goalsConj (proveAll gs) hole)

go hole as (CollectorFrame fid gc) =
Left (PoppedFrame fid as hole gc)
Left (fid, as, hole, gc)

go hole as (CollectingAssumptions as' gc) =
go (assuming as' <$> hole) (as' <> as) gc
Expand All @@ -317,10 +303,10 @@ gcPop = go Nothing mempty

-- | Get all currently collected goals.
gcFinish :: Monoid asmp => GoalCollector asmp goal -> Maybe (Goals asmp goal)
gcFinish gc =
case gcPop gc of
Left poppedFrame -> gcFinish (collectPoppedGoals poppedFrame)
Right a -> a
gcFinish gc = case gcPop gc of
Left (_,_,Just g,gc1) -> gcFinish (gcAddGoals g gc1)
Left (_,_,Nothing,gc1) -> gcFinish gc1
Right a -> a

-- | Reset the goal collector to the empty assumption state; but first
-- collect all the pending proof goals and stash them.
Expand Down
2 changes: 1 addition & 1 deletion crucible/src/Lang/Crucible/Backend/Simple.hs
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,7 @@ instance IsSymInterface (B.ExprBuilder t st fs) =>
AS.pushFrame (sbAssumptionStack bak)

popAssumptionFrame bak ident = do
AS.popFrameOrPanic ident (sbAssumptionStack bak)
AS.popFrame ident (sbAssumptionStack bak)

popAssumptionFrameAndObligations bak ident = do
AS.popFrameAndGoals ident (sbAssumptionStack bak)
Expand Down
Loading