Skip to content

Commit

Permalink
Skip further processing of all type declarations in a scope
Browse files Browse the repository at this point in the history
if any of them form a recursive group.

Fixes #1040
  • Loading branch information
robdockins committed Feb 9, 2021
1 parent 3c50959 commit a33814d
Show file tree
Hide file tree
Showing 5 changed files with 23 additions and 11 deletions.
16 changes: 6 additions & 10 deletions src/Cryptol/TypeCheck/Depends.hs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import qualified Cryptol.Parser.AST as P
import Cryptol.Parser.Position(Range, Located(..), thing)
import Cryptol.Parser.Names (namesB, tnamesT, tnamesC,
boundNamesSet, boundNames)
import Cryptol.TypeCheck.Monad( InferM, recordError, getTVars )
import Cryptol.TypeCheck.Monad( InferM, getTVars )
import Cryptol.TypeCheck.Error(Error(..))
import Cryptol.Utils.Panic(panic)
import Cryptol.Utils.RecordMap(recordElements)
Expand All @@ -29,6 +29,7 @@ import Data.Map (Map)
import qualified Data.Map as Map
import qualified Data.Set as Set
import Data.Text (Text)
import MonadLib (ExceptionT, runExceptionT, raise)

data TyDecl =
TS (P.TySyn Name) (Maybe Text) -- ^ Type synonym
Expand All @@ -49,13 +50,13 @@ setDocString x d =

-- | Check for duplicate and recursive type synonyms.
-- Returns the type-synonyms in dependency order.
orderTyDecls :: [TyDecl] -> InferM [TyDecl]
orderTyDecls :: [TyDecl] -> InferM (Either Error [TyDecl])
orderTyDecls ts =
do vs <- getTVars
ds <- combine $ map (toMap vs) ts
let ordered = mkScc [ (t,[x],deps)
| (x,(t,deps)) <- Map.toList (Map.map thing ds) ]
concat `fmap` mapM check ordered
runExceptionT (concat `fmap` mapM check ordered)

where
toMap vs ty@(PT p _) =
Expand Down Expand Up @@ -113,17 +114,12 @@ orderTyDecls ts =
getN (AT x _) = thing (P.ptName x)
getN (PT x _) = thing (P.primTName x)

check :: SCC TyDecl -> ExceptionT Error InferM [TyDecl]
check (AcyclicSCC x) = return [x]

-- We don't support any recursion, for now.
-- We could support recursion between newtypes, or newtypes and tysysn.
check (CyclicSCC xs) =
do recordError (RecursiveTypeDecls (map getN xs))
return [] -- XXX: This is likely to cause fake errors for missing
-- type synonyms. We could avoid this by, for example, checking
-- for recursive synonym errors, when looking up tycons.


check (CyclicSCC xs) = raise (RecursiveTypeDecls (map getN xs))

-- | Associate type signatures with bindings and order bindings by dependency.
orderBinds :: [P.Bind Name] -> [SCC (P.Bind Name)]
Expand Down
4 changes: 3 additions & 1 deletion src/Cryptol/TypeCheck/Infer.hs
Original file line number Diff line number Diff line change
Expand Up @@ -946,8 +946,10 @@ checkSigB b (Forall as asmps0 t0, validSchema) = case thing (P.bDef b) of
}

inferDs :: FromDecl d => [d] -> ([DeclGroup] -> InferM a) -> InferM a
inferDs ds continue = checkTyDecls =<< orderTyDecls (mapMaybe toTyDecl ds)
inferDs ds continue = either onErr checkTyDecls =<< orderTyDecls (mapMaybe toTyDecl ds)
where
onErr err = recordError err >> continue []

isTopLevel = isTopDecl (head ds)

checkTyDecls (AT t mbD : ts) =
Expand Down
6 changes: 6 additions & 0 deletions tests/issues/issue1040.cry
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
module binarytree where

newtype Tree = { inTree : Tree }

build : Tree
build = Tree { inTree = undefined }
1 change: 1 addition & 0 deletions tests/issues/issue1040.icry
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
:l issue1040.cry
7 changes: 7 additions & 0 deletions tests/issues/issue1040.icry.stdout
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
Loading module Cryptol
Loading module Cryptol
Loading module binarytree

[error] at issue1040.cry:1:1--6:36:
Recursive type declarations:
`binarytree::Tree`

0 comments on commit a33814d

Please sign in to comment.