MkExternalCore.lhs 7.65 KB
Newer Older
1

apt's avatar
apt committed
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
% (c) The University of Glasgow 2001
%
\begin{code}

module MkExternalCore (
	emitExternalCore
) where

#include "HsVersions.h"

import qualified ExternalCore as C
import Char
import Module
import CoreSyn
import HscTypes	
import TyCon
import TypeRep
import Type
20
import PprExternalCore	-- Instances
21
import DataCon	( DataCon, dataConExTyVars, dataConRepArgTys, 
22
		  dataConName, dataConTyCon )
apt's avatar
apt committed
23
24
25
26
27
import CoreSyn
import Var
import IdInfo
import Literal
import Name
28
29
import NameSet ( NameSet, emptyNameSet )
import UniqSet ( elementOfUniqSet )
apt's avatar
apt committed
30
31
import Outputable
import ForeignCall
32
33
import DynFlags	( DynFlags(..) )
import StaticFlags	( opt_EmitExternalCore )
apt's avatar
apt committed
34
import IO
35
import FastString
apt's avatar
apt committed
36

37
38
emitExternalCore :: DynFlags -> NameSet -> CgGuts -> IO ()
emitExternalCore dflags exports cg_guts
apt's avatar
apt committed
39
40
 | opt_EmitExternalCore 
 = (do handle <- openFile corename WriteMode
41
       hPutStrLn handle (show (mkExternalCore exports cg_guts))      
apt's avatar
apt committed
42
43
44
45
       hClose handle)
   `catch` (\err -> pprPanic "Failed to open or write external core output file" 
	                     (text corename))
   where corename = extCoreName dflags
46
emitExternalCore _ _ _
apt's avatar
apt committed
47
48
49
50
 | otherwise
 = return ()


51
mkExternalCore :: NameSet -> CgGuts -> C.Module
52
53
54
55
-- The ModGuts has been tidied, but the implicit bindings have
-- not been injected, so we have to add them manually here
-- We don't include the strange data-con *workers* because they are
-- implicit in the data type declaration itself
56
57
mkExternalCore exports (CgGuts {cg_module=this_mod, cg_tycons = tycons, cg_binds = binds})
  = C.Module mname tdefs (map (make_vdef exports) binds)
apt's avatar
apt committed
58
  where
59
60
    mname  = make_mid this_mod
    tdefs  = foldr collect_tdefs [] tycons
61

apt's avatar
apt committed
62
collect_tdefs :: TyCon -> [C.Tdef] -> [C.Tdef]
63
collect_tdefs tcon tdefs 
apt's avatar
apt committed
64
  | isAlgTyCon tcon = tdef: tdefs
65
  where
apt's avatar
apt committed
66
    tdef | isNewTyCon tcon = 
apt's avatar
apt committed
67
                C.Newtype (make_con_qid (tyConName tcon)) (map make_tbind tyvars) repclause 
68
69
-- 20060420 GHC handles empty data types just fine. ExtCore should too! jds
--         | null (tyConDataCons tcon) = error "MkExternalCore died: can't handle datatype declarations with no data constructors"
apt's avatar
apt committed
70
         | otherwise = 
apt's avatar
apt committed
71
                C.Data (make_con_qid (tyConName tcon)) (map make_tbind tyvars) (map make_cdef (tyConDataCons tcon)) 
apt's avatar
apt committed
72
73
74
75
         where repclause | isRecursiveTyCon tcon = Nothing
		         | otherwise = Just (make_ty rep)
                                           where (_, rep) = newTyConRep tcon
    tyvars = tyConTyVars tcon
76

apt's avatar
apt committed
77
78
79
80
81
82
collect_tdefs _ tdefs = tdefs


make_cdef :: DataCon -> C.Cdef
make_cdef dcon =  C.Constr dcon_name existentials tys
  where 
83
    dcon_name    = make_var_id (dataConName dcon)
apt's avatar
apt committed
84
    existentials = map make_tbind ex_tyvars
85
    ex_tyvars    = dataConExTyVars dcon
86
    tys 	 = map make_ty (dataConRepArgTys dcon)
apt's avatar
apt committed
87
88
89
90
91

make_tbind :: TyVar -> C.Tbind
make_tbind tv = (make_var_id (tyVarName tv), make_kind (tyVarKind tv))
    
make_vbind :: Var -> C.Vbind
92
make_vbind v = (make_var_id  (Var.varName v), make_ty (idType v))
apt's avatar
apt committed
93

94
95
make_vdef :: NameSet -> CoreBind -> C.Vdefg
make_vdef exports b = 
apt's avatar
apt committed
96
  case b of
apt's avatar
apt committed
97
98
    NonRec v e -> C.Nonrec (f (v,e))
    Rec ves -> C.Rec (map f ves)
99
100
101
  where
  f (v,e) = (local, make_var_id (Var.varName v), make_ty (idType v),make_exp e)
  	where local = not $ elementOfUniqSet (Var.varName v) exports
102
	-- Top level bindings are unqualified now
apt's avatar
apt committed
103
104
105
106

make_exp :: CoreExpr -> C.Exp
make_exp (Var v) =  
  case globalIdDetails v of
107
108
     -- a DataConId represents the Id of a worker, which is a varName. -- sof 4/02
--    DataConId _ -> C.Dcon (make_con_qid (Var.varName v))
109
    FCallId (CCall (CCallSpec (StaticTarget nm) _ _)) -> C.External (unpackFS nm) (make_ty (idType v))
apt's avatar
apt committed
110
    FCallId _ -> error "MkExternalCore died: can't handle non-static-C foreign call"
apt's avatar
apt committed
111
    _ -> C.Var (make_var_qid (Var.varName v))
krc's avatar
krc committed
112
make_exp (Lit (l@(MachLabel s _))) = error "MkExternalCore died: can't handle \"foreign label\" declarations"
apt's avatar
apt committed
113
114
115
116
117
make_exp (Lit l) = C.Lit (make_lit l)
make_exp (App e (Type t)) = C.Appt (make_exp e) (make_ty t)
make_exp (App e1 e2) = C.App (make_exp e1) (make_exp e2)
make_exp (Lam v e) | isTyVar v = C.Lam (C.Tb (make_tbind v)) (make_exp e)
make_exp (Lam v e) | otherwise = C.Lam (C.Vb (make_vbind v)) (make_exp e)
118
make_exp (Let b e) = C.Let (make_vdef b) (make_exp e)
119
120
-- gaw 2004
make_exp (Case e v ty alts) = C.Case (make_exp e) (make_vbind v) (make_ty ty) (map make_alt alts)
apt's avatar
apt committed
121
make_exp (Note (SCC cc) e) = C.Note "SCC"  (make_exp e) -- temporary
122
make_exp (Note (CoreNote s) e) = C.Note s (make_exp e)  -- hdaume: core annotations
apt's avatar
apt committed
123
124
125
126
127
make_exp (Note InlineMe e) = C.Note "InlineMe" (make_exp e)
make_exp _ = error "MkExternalCore died: make_exp"

make_alt :: CoreAlt -> C.Alt
make_alt (DataAlt dcon, vs, e) = 
128
129
130
131
    C.Acon (make_con_qid (dataConName dcon))
           (map make_tbind tbs)
           (map make_vbind vbs)
	   (make_exp e)    
apt's avatar
apt committed
132
133
134
135
136
137
138
	where (tbs,vbs) = span isTyVar vs
make_alt (LitAlt l,_,e) = C.Alit (make_lit l) (make_exp e)
make_alt (DEFAULT,[],e) = C.Adefault (make_exp e)

make_lit :: Literal -> C.Lit
make_lit l = 
  case l of
139
    MachChar i -> C.Lchar i t
140
    MachStr s -> C.Lstring (unpackFS s) t
sof's avatar
sof committed
141
    MachNullAddr -> C.Lint 0 t
apt's avatar
apt committed
142
143
144
145
146
147
148
149
150
151
152
    MachInt i -> C.Lint i t
    MachInt64 i -> C.Lint i t
    MachWord i -> C.Lint i t
    MachWord64 i -> C.Lint i t
    MachFloat r -> C.Lrational r t
    MachDouble r -> C.Lrational r t
    _ -> error "MkExternalCore died: make_lit"
  where 
    t = make_ty (literalType l)

make_ty :: Type -> C.Ty
153
154
155
156
157
158
make_ty (TyVarTy tv)    	 = C.Tvar (make_var_id (tyVarName tv))
make_ty (AppTy t1 t2) 		 = C.Tapp (make_ty t1) (make_ty t2)
make_ty (FunTy t1 t2) 		 = make_ty (TyConApp funTyCon [t1,t2])
make_ty (ForAllTy tv t) 	 = C.Tforall (make_tbind tv) (make_ty t)
make_ty (TyConApp tc ts) 	 = foldl C.Tapp (C.Tcon (make_con_qid (tyConName tc))) 
					 (map make_ty ts)
159
-- Newtypes are treated just like any other type constructor; not expanded
160
-- Reason: predTypeRep does substitution and, while substitution deals
161
162
163
164
165
166
167
168
169
170
-- 	   correctly with name capture, it's only correct if you see the uniques!
--	   If you just see occurrence names, name capture may occur.
-- Example: newtype A a = A (forall b. b -> a)
--	    test :: forall q b. q -> A b
--	    test _ = undefined
-- 	Here the 'a' gets substituted by 'b', which is captured.
-- Another solution would be to expand newtypes before tidying; but that would
-- expose the representation in interface files, which definitely isn't right.
-- Maybe CoreTidy should know whether to expand newtypes or not?

171
172
make_ty (PredTy p)	= make_ty (predTypeRep p)
make_ty (NoteTy _ t) 	= make_ty t
173

apt's avatar
apt committed
174
175
176


make_kind :: Kind -> C.Kind
177
178
179
180
181
182
make_kind (FunTy k1 k2)  = C.Karrow (make_kind k1) (make_kind k2)
make_kind k
  | isLiftedTypeKind k   = C.Klifted
  | isUnliftedTypeKind k = C.Kunlifted
--   | isUnboxedTypeKind k  = C.Kunboxed	Fix me
  | isOpenTypeKind k     = C.Kopen
apt's avatar
apt committed
183
184
185
186
make_kind _ = error "MkExternalCore died: make_kind"

{- Id generation. -}

apt's avatar
apt committed
187
{- Use encoded strings.
apt's avatar
apt committed
188
189
   Also, adjust casing to work around some badly-chosen internal names. -}
make_id :: Bool -> Name -> C.Id
190
191
192
make_id is_var nm = (occNameString . nameOccName) nm

{-	SIMON thinks this stuff isn't necessary
apt's avatar
apt committed
193
194
make_id is_var nm = 
  case n of
apt's avatar
apt committed
195
196
197
198
199
    'Z':cs | is_var -> 'z':cs 
    'z':cs | not is_var -> 'Z':cs 
    c:cs | isUpper c && is_var -> 'z':'d':n
    c:cs | isLower c && (not is_var) -> 'Z':'d':n
    _ -> n
apt's avatar
apt committed
200
  where n = (occNameString . nameOccName) nm
201
-}
apt's avatar
apt committed
202
203
204
205
206

make_var_id :: Name -> C.Id
make_var_id = make_id True

make_mid :: Module -> C.Id
Simon Marlow's avatar
Simon Marlow committed
207
make_mid = showSDoc . pprModule
apt's avatar
apt committed
208
209
210
211
212
213

make_qid :: Bool -> Name -> C.Qual C.Id
make_qid is_var n = (mname,make_id is_var n)
    where mname = 
           case nameModule_maybe n of
            Just m -> make_mid m
apt's avatar
apt committed
214
            Nothing -> "" 
apt's avatar
apt committed
215
216
217
218
219
220
221
222
223
224
225
226

make_var_qid :: Name -> C.Qual C.Id
make_var_qid = make_qid True

make_con_qid :: Name -> C.Qual C.Id
make_con_qid = make_qid False

\end{code}