diff options
author | Manuel M T Chakravarty <chak@cse.unsw.edu.au> | 2006-09-20 18:36:46 +0000 |
---|---|---|
committer | Manuel M T Chakravarty <chak@cse.unsw.edu.au> | 2006-09-20 18:36:46 +0000 |
commit | 909d2dd885f5eebaf7c12cf15d5ac153d646566e (patch) | |
tree | 21f2e2710b8238135e3b8fb488b7fc2b8dfb0247 /compiler/iface/BuildTyCl.lhs | |
parent | 275dde6de685153db621b11f2f404aa78d9183e2 (diff) | |
download | haskell-909d2dd885f5eebaf7c12cf15d5ac153d646566e.tar.gz |
Introduce coercions for data instance decls
Mon Sep 18 19:07:30 EDT 2006 Manuel M T Chakravarty <chak@cse.unsw.edu.au>
* Introduce coercions for data instance decls
Tue Aug 22 20:33:46 EDT 2006 Manuel M T Chakravarty <chak@cse.unsw.edu.au>
* Introduce coercions for data instance decls
- data instance declarations implicitly generate a coercion moving between the
representation type and family instance type.
- The coercion is *implicitly* generated when type checking both source and
ifaces. Ie, we don't safe it in ifaces - this is really exactly as newtype
coercions are handled.
- The previous addition of the instance types to DataCons has been moved to
the representation TyCon. This is more efficient as it is shared between all
constructors of one representation tycon and it also gathers everything about
data instances (family tycon, instance types, and coercion) in one place: the
algTcParent field of TyCon.
- The coercion is already used in the datacon wrappers, but not yet during type
checking pattern matching of indexed data types.
- The code has only been lightly tested, but doesn't seem to break features not
related to indexed types. For indexed data types only the pattern matching
tc code (in TcPat.tcConPat) and some well-formedness checks are still
missing. And there will surely be some bugs to fix. (newtypes still require
some more work.)
** WARNING: Interface file format changed! **
** Recompile from scratch! **
Diffstat (limited to 'compiler/iface/BuildTyCl.lhs')
-rw-r--r-- | compiler/iface/BuildTyCl.lhs | 74 |
1 files changed, 51 insertions, 23 deletions
diff --git a/compiler/iface/BuildTyCl.lhs b/compiler/iface/BuildTyCl.lhs index 05f5f4bc22..5f23fd5fb4 100644 --- a/compiler/iface/BuildTyCl.lhs +++ b/compiler/iface/BuildTyCl.lhs @@ -25,7 +25,8 @@ import BasicTypes ( RecFlag, StrictnessMark(..) ) import Name ( Name ) import OccName ( mkDataConWrapperOcc, mkDataConWorkerOcc, mkClassTyConOcc, mkClassDataConOcc, - mkSuperDictSelOcc, mkNewTyCoOcc, mkLocalOcc ) + mkSuperDictSelOcc, mkNewTyCoOcc, mkInstTyTcOcc, + mkInstTyCoOcc ) import MkId ( mkDataConIds, mkRecordSelId, mkDictSelId ) import Class ( mkClass, Class( classTyCon), FunDep, DefMeth(..) ) import TyCon ( mkSynTyCon, mkAlgTyCon, visibleDataCons, @@ -41,7 +42,7 @@ import Type ( mkArrowKinds, liftedTypeKind, typeKind, TyThing(..), substTyWith, zipTopTvSubst, substTheta, mkForAllTys, mkTyConApp, mkTyVarTy ) -import Coercion ( mkNewTypeCoercion ) +import Coercion ( mkNewTypeCoercion, mkDataInstCoercion ) import Outputable import List ( nub ) @@ -68,27 +69,55 @@ buildAlgTyCon :: Name -> [TyVar] -> RecFlag -> Bool -- True <=> want generics functions -> Bool -- True <=> was declared in GADT syntax - -> Maybe TyCon -- Just family <=> instance of `family' + -> Maybe (TyCon, [Type]) -- Just (family, tys) + -- <=> instance of `family' at `tys' -> TcRnIf m n TyCon buildAlgTyCon tc_name tvs stupid_theta rhs is_rec want_generics gadt_syn mb_family - = do { -- In case of a type instance, we need to invent a new name for the - -- instance type, as `tc_name' is the family name. - ; uniq <- newUnique - ; (final_name, parent) <- - case mb_family of - Nothing -> return (tc_name, NoParentTyCon) - Just family -> - do { final_name <- newImplicitBinder tc_name (mkLocalOcc uniq) - ; return (final_name, FamilyTyCon family) - } - ; let { tycon = mkAlgTyCon final_name kind tvs stupid_theta rhs - fields parent is_rec want_generics gadt_syn - ; kind = mkArrowKinds (map tyVarKind tvs) liftedTypeKind - ; fields = mkTyConSelIds tycon rhs - } - ; return tycon } + = do { -- We need to tie a knot as the coercion of a data instance depends + -- on the instance representation tycon and vice versa. + ; tycon <- fixM (\ tycon_rec -> do + { (final_name, parent) <- maybeComputeFamilyInfo mb_family tycon_rec + ; let { tycon = mkAlgTyCon final_name kind tvs stupid_theta rhs + fields parent is_rec want_generics gadt_syn + ; kind = mkArrowKinds (map tyVarKind tvs) liftedTypeKind + ; fields = mkTyConSelIds tycon rhs + } + ; return tycon + }) + ; return tycon + } + where + -- If a family tycon with instance types is given, the current tycon is an + -- instance of that family and we have to perform three extra tasks: + -- + -- (1) The instance tycon (representing the family at a particular type + -- instance) need to get a new, derived name - we may not reuse the + -- family name. + -- (2) Create a coercion that identifies the family instance type and the + -- representation type from Step (1); ie, it is of the form + -- `Co tvs :: F ts :=: R tvs', where `Co' is the name of the coercion, + -- `F' the family tycon and `R' the (derived) representation tycon. + -- (3) Produce a `AlgTyConParent' value containing the parent and coercion + -- information. + -- + maybeComputeFamilyInfo Nothing rep_tycon = + return (tc_name, NoParentTyCon) + maybeComputeFamilyInfo (Just (family, instTys)) rep_tycon = + do { -- (1) New, derived name for the instance tycon + ; uniq <- newUnique + ; final_name <- newImplicitBinder tc_name (mkInstTyTcOcc uniq) + + -- (2) Create the coercion. + ; co_tycon_name <- newImplicitBinder tc_name (mkInstTyCoOcc uniq) + ; let co_tycon = mkDataInstCoercion co_tycon_name tvs + family instTys rep_tycon + + -- (3) Produce parent information. + ; return (final_name, FamilyTyCon family instTys co_tycon) + } + ------------------------------------------------------ mkAbstractTyConRhs :: AlgTyConRhs @@ -190,14 +219,13 @@ buildDataCon :: Name -> Bool -> ThetaType -- Does not include the "stupid theta" -- or the GADT equalities -> [Type] -> TyCon - -> Maybe [Type] -- Just ts <=> type pats of inst type -> TcRnIf m n DataCon -- A wrapper for DataCon.mkDataCon that -- a) makes the worker Id -- b) makes the wrapper Id if necessary, including -- allocating its unique (hence monadic) buildDataCon src_name declared_infix arg_stricts field_lbls - univ_tvs ex_tvs eq_spec ctxt arg_tys tycon mb_typats + univ_tvs ex_tvs eq_spec ctxt arg_tys tycon = do { wrap_name <- newImplicitBinder src_name mkDataConWrapperOcc ; work_name <- newImplicitBinder src_name mkDataConWorkerOcc -- This last one takes the name of the data constructor in the source @@ -209,7 +237,7 @@ buildDataCon src_name declared_infix arg_stricts field_lbls data_con = mkDataCon src_name declared_infix arg_stricts field_lbls univ_tvs ex_tvs eq_spec ctxt - arg_tys tycon mb_typats + arg_tys tycon stupid_ctxt dc_ids dc_ids = mkDataConIds wrap_name work_name data_con @@ -286,7 +314,7 @@ buildClass class_name tvs sc_theta fds ats sig_stuff tc_isrec tvs [{- no existentials -}] [{- No equalities -}] [{-No context-}] dict_component_tys - rec_tycon Nothing + rec_tycon ; rhs <- case dict_component_tys of [rep_ty] -> mkNewTyConRhs tycon_name rec_tycon dict_con |