1
1
mirror of https://github.com/github/semantic.git synced 2024-12-29 18:06:14 +03:00

Define looping as primitive on MonadValue.

This commit is contained in:
Rob Rix 2018-03-14 15:27:53 -04:00
parent 222ce40be3
commit 2eec625fbb

View File

@ -67,6 +67,8 @@ class (MonadAnalysis term value m, Show value) => MonadValue term value m where
-- | Evaluate an application (like a function call).
apply :: value -> [m value] -> m value
loop :: (m value -> m value) -> m value
-- | Attempt to extract a 'Prelude.Bool' from a given value.
toBool :: MonadValue term value m => value -> m Bool
toBool v = ifthenelse v (pure True) (pure False)
@ -95,7 +97,7 @@ while :: ( MonadAddressable (LocationFor value) value m
=> m value
-> m value
-> m value
while cond body = loop $ do
while cond body = loop $ \ continue -> do
this <- cond
ifthenelse this (body *> continue) unit
@ -108,28 +110,10 @@ doWhile :: ( MonadAddressable (LocationFor value) value m
=> m value
-> m value
-> m value
doWhile body cond = loop $ body *> do
doWhile body cond = loop $ \ continue -> body *> do
this <- cond
ifthenelse this continue unit
loop :: ( MonadAddressable (LocationFor value) value m
, MonadEnvironment value m
, MonadStore value m
)
=> m value
-> m value
loop = fmap fst . letrec (name "loop")
continue :: ( MonadAddressable (LocationFor value) value m
, MonadEnvironment value m
, MonadFail m
, MonadStore value m
)
=> m value
continue = do
env <- askLocalEnv
maybe (fail "free loop variable") deref (envLookup (name "loop") env)
-- | Construct a 'Value' wrapping the value arguments (if any).
instance ( FreeVariables term
, MonadAddressable location (Value location term) m
@ -205,6 +189,8 @@ instance ( FreeVariables term
envInsert name a <$> rest) (pure env) (zip names params)
localEnv (mappend bindings) (evaluateTerm body)
loop = fix
-- | Discard the value arguments (if any), constructing a 'Type.Type' instead.
instance (Alternative m, MonadAnalysis term (Type f) m, MonadFresh m) => MonadValue term (Type f) m where
abstract names (Subterm _ body) = do
@ -249,3 +235,5 @@ instance (Alternative m, MonadAnalysis term (Type f) m, MonadFresh m) => MonadVa
paramTypes <- sequenceA params
_ :-> ret <- op `unify` (Product paramTypes :-> Var tvar)
pure ret
loop f = f empty