X-Git-Url: http://git.megacz.com/?p=ghc-hetmet.git;a=blobdiff_plain;f=compiler%2Fiface%2FBuildTyCl.lhs;h=bf71ca843ce8fa5158f27b326197aa2aa42341cb;hp=e4c392b6a55fe6d2e46b9be604b772275d7ce285;hb=bb106f283663e9c16a4c72ec9ca57109ae57a0ed;hpb=3e0b6b2542d8464bfba365b97a6e4b95c3885f10 diff --git a/compiler/iface/BuildTyCl.lhs b/compiler/iface/BuildTyCl.lhs index e4c392b..bf71ca8 100644 --- a/compiler/iface/BuildTyCl.lhs +++ b/compiler/iface/BuildTyCl.lhs @@ -6,7 +6,8 @@ module BuildTyCl ( buildSynTyCon, buildAlgTyCon, buildDataCon, buildClass, - mkAbstractTyConRhs, mkNewTyConRhs, mkDataTyConRhs + mkAbstractTyConRhs, mkOpenDataTyConRhs, mkOpenNewTyConRhs, + mkNewTyConRhs, mkDataTyConRhs ) where #include "HsVersions.h" @@ -26,14 +27,17 @@ import OccName ( mkDataConWrapperOcc, mkDataConWorkerOcc, mkClassTyConOcc, mkClassDataConOcc, mkSuperDictSelOcc, mkNewTyCoOcc ) import MkId ( mkDataConIds, mkRecordSelId, mkDictSelId ) import Class ( mkClass, Class( classTyCon), FunDep, DefMeth(..) ) -import TyCon ( mkSynTyCon, mkAlgTyCon, visibleDataCons, tyConStupidTheta, - tyConDataCons, isNewTyCon, mkClassTyCon, TyCon( tyConTyVars ), - isRecursiveTyCon, tyConArity, - AlgTyConRhs(..), newTyConRhs ) +import TyCon ( mkSynTyCon, mkAlgTyCon, visibleDataCons, + tyConStupidTheta, tyConDataCons, isNewTyCon, + mkClassTyCon, TyCon( tyConTyVars ), + isRecursiveTyCon, tyConArity, AlgTyConRhs(..), + SynTyConRhs(..), newTyConRhs ) import Type ( mkArrowKinds, liftedTypeKind, typeKind, tyVarsOfType, tyVarsOfTypes, tyVarsOfPred, - splitTyConApp_maybe, splitAppTy_maybe, getTyVar_maybe, - mkPredTys, mkTyVarTys, ThetaType, Type, + splitTyConApp_maybe, splitAppTy_maybe, + getTyVar_maybe, + mkPredTys, mkTyVarTys, ThetaType, Type, Kind, + TyThing(..), substTyWith, zipTopTvSubst, substTheta, mkForAllTys, mkTyConApp, mkTyVarTy ) import Coercion ( mkNewTypeCoercion ) @@ -45,8 +49,13 @@ import List ( nub ) \begin{code} ------------------------------------------------------ -buildSynTyCon name tvs rhs_ty - = mkSynTyCon name kind tvs rhs_ty +buildSynTyCon :: Name -> [TyVar] -> SynTyConRhs -> TyCon +buildSynTyCon name tvs rhs@(OpenSynTyCon rhs_ki) + = mkSynTyCon name kind tvs rhs + where + kind = mkArrowKinds (map tyVarKind tvs) rhs_ki +buildSynTyCon name tvs rhs@(SynonymTyCon rhs_ty) + = mkSynTyCon name kind tvs rhs where kind = mkArrowKinds (map tyVarKind tvs) (typeKind rhs_ty) @@ -72,6 +81,12 @@ buildAlgTyCon tc_name tvs stupid_theta rhs is_rec want_generics gadt_syn mkAbstractTyConRhs :: AlgTyConRhs mkAbstractTyConRhs = AbstractTyCon +mkOpenDataTyConRhs :: AlgTyConRhs +mkOpenDataTyConRhs = OpenDataTyCon + +mkOpenNewTyConRhs :: AlgTyConRhs +mkOpenNewTyConRhs = OpenNewTyCon + mkDataTyConRhs :: [DataCon] -> AlgTyConRhs mkDataTyConRhs cons = DataTyCon { data_cons = cons, is_enum = all isNullarySrcDataCon cons } @@ -82,13 +97,24 @@ mkNewTyConRhs :: Name -> TyCon -> DataCon -> TcRnIf m n AlgTyConRhs -- because the latter is part of a knot, whereas the former is not. mkNewTyConRhs tycon_name tycon con = do { co_tycon_name <- newImplicitBinder tycon_name mkNewTyCoOcc - ; let co_tycon = mkNewTypeCoercion co_tycon_name tycon tvs rhs_ty + ; let co_tycon = mkNewTypeCoercion co_tycon_name tycon tvs rhs_ty + cocon_maybe + | all_coercions || isRecursiveTyCon tycon + = Just co_tycon + | otherwise + = Nothing ; return (NewTyCon { data_con = con, - nt_co = co_tycon, + nt_co = cocon_maybe, + -- Coreview looks through newtypes with a Nothing + -- for nt_co, or uses explicit coercions otherwise nt_rhs = rhs_ty, nt_etad_rhs = eta_reduce tvs rhs_ty, nt_rep = mkNewTyConRep tycon rhs_ty }) } where + -- if all_coercions is True then we use coercions for all newtypes + -- otherwise we use coercions for recursive newtypes and look through + -- non-recursive newtypes + all_coercions = True tvs = tyConTyVars tycon rhs_ty = head (dataConInstOrigArgTys con (mkTyVarTys tvs)) -- Instantiate the data con with the @@ -116,9 +142,8 @@ mkNewTyConRep :: TyCon -- The original type constructor -- Remember that the representation type is the *ultimate* representation -- type, looking through other newtypes. -- --- The non-recursive newtypes are easy, because they look transparent --- to splitTyConApp_maybe, but recursive ones really are represented as --- TyConApps (see TypeRep). +-- splitTyConApp_maybe no longer looks through newtypes, so we must +-- deal explicitly with this case -- -- The trick is to to deal correctly with recursive newtypes -- such as newtype T = MkT T @@ -133,10 +158,11 @@ mkNewTyConRep tc rhs_ty = case splitTyConApp_maybe rep_ty of Just (tc, tys) | tc `elem` tcs -> unitTy -- Recursive loop - | isNewTyCon tc -> ASSERT( isRecursiveTyCon tc ) - -- Non-recursive ones have been - -- dealt with by splitTyConApp_maybe - go (tc:tcs) (substTyWith tvs tys rhs_ty) + | isNewTyCon tc -> + if isRecursiveTyCon tc then + go (tc:tcs) (substTyWith tvs tys rhs_ty) + else + substTyWith tvs tys rhs_ty where (tvs, rhs_ty) = newTyConRhs tc @@ -206,11 +232,12 @@ mkTyConSelIds tycon rhs \begin{code} buildClass :: Name -> [TyVar] -> ThetaType -> [FunDep TyVar] -- Functional dependencies + -> [TyThing] -- Associated types -> [(Name, DefMeth, Type)] -- Method info -> RecFlag -- Info for type constructor -> TcRnIf m n Class -buildClass class_name tvs sc_theta fds sig_stuff tc_isrec +buildClass class_name tvs sc_theta fds ats sig_stuff tc_isrec = do { tycon_name <- newImplicitBinder class_name mkClassTyConOcc ; datacon_name <- newImplicitBinder class_name mkClassDataConOcc -- The class name is the 'parent' for this datacon, not its tycon, @@ -260,10 +287,12 @@ buildClass class_name tvs sc_theta fds sig_stuff tc_isrec -- Because C has only one operation, it is represented by -- a newtype, and it should be a *recursive* newtype. -- [If we don't make it a recursive newtype, we'll expand the - -- newtype like a synonym, but that will lead to an infinite type] + -- newtype like a synonym, but that will lead to an infinite + -- type] + ; atTyCons = [tycon | ATyCon tycon <- ats] } - ; return (mkClass class_name tvs fds - sc_theta sc_sel_ids op_items + ; return (mkClass class_name tvs fds + sc_theta sc_sel_ids atTyCons op_items tycon) })} \end{code}