Cross Reference: /hets/HasCASL/TypeRel.hs
TypeRel.hs revision 9e5c10805bd50b5baaf6bf1f6a3f085c7afb17d8
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
{- |
Module : $Header$
Description : compute subtype dependencies
Copyright : (c) Christian Maeder and Uni Bremen 2003-2005
License : GPLv2 or higher, see LICENSE.txt
Maintainer : Christian.Maeder@dfki.de
Stability : experimental
Portability : portable
compute subtype dependencies
-}
module HasCASL.TypeRel where
import HasCASL.As
import HasCASL.AsUtils
import HasCASL.ClassAna
import HasCASL.Le
import HasCASL.Builtin
import HasCASL.DataAna
import HasCASL.MinType
import Common.Id
import Common.AS_Annotation
import qualified Common.Lib.Rel as Rel
import qualified Data.Map as Map
import qualified Data.Set as Set
import Data.Maybe
typeRel :: TypeMap -> Rel.Rel Id
typeRel = Rel.transReduce . Rel.irreflex . Rel.transClosure
. Map.foldrWithKey ( \ i ti r ->
Set.fold (Rel.insert i) r $ superTypes ti) Rel.empty
getRawKind :: TypeMap -> Id -> RawKind
getRawKind tm i = typeKind $
Map.findWithDefault (error $ showId i " not found in type map") i tm
-- | make a polymorphic function from a to b
mkInjOrProjType :: Arrow -> TypeScheme
mkInjOrProjType arr =
TypeScheme [aTypeArg, bTypeArg] (mkFunArrType aType arr bType) nullRange
injType :: TypeScheme
injType = mkInjOrProjType FunArr
projType :: TypeScheme
projType = mkInjOrProjType PFunArr
mkInjOrProj :: Arrow -> Set.Set OpInfo
mkInjOrProj arr = Set.singleton OpInfo
{ opType = mkInjOrProjType arr
, opAttrs = Set.empty
, opDefn = NoOpDefn Fun }
subtRelName :: Id
subtRelName = mkId [genToken "subt"]
subtRelType :: TypeScheme
subtRelType = TypeScheme [aTypeArg, bTypeArg]
(mkFunArrType (mkProductType [aType, bType]) PFunArr unitType) nullRange
subtRel :: Set.Set OpInfo
subtRel = Set.singleton OpInfo
{ opType = subtRelType
, opAttrs = Set.empty
, opDefn = NoOpDefn Fun }
subtAxioms :: TypeMap -> [Named Sentence]
subtAxioms tm =
let tr = typeRel tm in
if Rel.null tr then [] else subtReflex : subtTrans : subtInjProj : injTrans
: map (subtAx tm) (Rel.toList tr)
nr :: Range
nr = nullRange
getKindAppl :: RawKind -> [Variance]
getKindAppl rk = case rk of
ClassKind () -> []
FunKind v _ r _ -> v : getKindAppl r
mkTypeArg :: Id -> Int -> TypeArg
mkTypeArg i c = TypeArg i NonVar (VarKind universe) rStar c Other nr
subtAx :: TypeMap -> (Id, Id) -> Named Sentence
subtAx tm (i1, i2) = let
findType i = Map.findWithDefault
(Map.findWithDefault (error "TypeRel.subtAx") i bTypes) i tm
e1 = findType i1
e2 = findType i2
txt = shows i1 "_<_" ++ show i2
l1 = getKindAppl $ typeKind e1
l2 = getKindAppl $ typeKind e2
l3 = zipWith minVariance l1 l2
l4 = foldr (\ v (((tl, vl), pl), fl) ->
let c = length pl + 1
a = simpleIdToId $ genNumVar "a" c
ta = mkTypeArg a (- (length tl + 1))
b = simpleIdToId $ genNumVar "b" c
tb = mkTypeArg b (- (length tl + 2))
x = stringToId $ 'x' : show c
y = stringToId $ 'y' : show c
tx = typeArgToType ta
ty = typeArgToType tb
vx = mkVarDecl x tx
vy = mkVarDecl y ty
in (case v of
NonVar -> ((ta : tl, vx : vl), (tx, tx) : pl)
_ -> (([ta, tb] ++ tl, [vx, vy] ++ vl), (tx, ty) : pl)
, case v of
CoVar -> mkSubtTerm tx ty vx vy : fl
ContraVar -> mkSubtTerm ty tx vy vx : fl
_ -> fl)) ((([], []), []), []) l3
(args, fs) = l4
(gens, acts) = args
(act1, act2) = unzip acts
(tyargs, vargs) = gens
t1 = mkTypeAppl (toType i1) act1
t2 = mkTypeAppl (toType i2) act2
x1 = stringToId "x"
x2 = stringToId "y"
v1 = mkVarDecl x1 t1
v2 = mkVarDecl x2 t2
in makeNamed ("ga_subt_" ++ txt) $ Formula
$ mkForall (map GenTypeVarDecl tyargs
++ map GenVarDecl (vargs ++ [v1, v2]))
$ (if null fs then id else
mkLogTerm implId nr (foldr1 (mkLogTerm andId nr) fs))
$ mkSubtTerm t1 t2 v1 v2
mkSubtTerm :: Type -> Type -> VarDecl -> VarDecl -> Term
mkSubtTerm t1 t2 v1 v2 = mkTerm subtRelName subtRelType [t1, t2] nr
$ TupleTerm [QualVar v1, QualVar v2] nr
subtReflex :: Named Sentence
subtReflex = let
v1 = mkVarDecl (stringToId "x") aType
v2 = mkVarDecl (stringToId "y") aType
in makeNamed "ga_subt_reflexive" $ Formula
$ mkForall (GenTypeVarDecl aTypeArg : map GenVarDecl [v1, v2])
$ mkSubtTerm aType aType v1 v2
subtTrans :: Named Sentence
subtTrans = let
v1 = mkVarDecl (stringToId "x") aType
v2 = mkVarDecl (stringToId "y") bType
v3 = mkVarDecl (stringToId "z") cType
in makeNamed "ga_subt_transitive" $ Formula
$ mkForall (map GenTypeVarDecl [aTypeArg, bTypeArg, cTypeArg]
++ map GenVarDecl [v1, v2, v3])
$ mkLogTerm implId nr
(mkLogTerm andId nr
(mkSubtTerm aType bType v1 v2)
$ mkSubtTerm bType cType v2 v3)
$ mkSubtTerm aType cType v1 v3
mkInjTerm :: Type -> Type -> Term -> Term
mkInjTerm t1 t2 = mkTerm injName injType [t1, t2] nr
mkInjEq :: Type -> Type -> VarDecl -> VarDecl -> Term
mkInjEq t1 t2 v1 v2 =
mkEqTerm eqId t2 nr (QualVar v2)
$ mkInjTerm t1 t2 $ QualVar v1
subtInjProj :: Named Sentence
subtInjProj = let
v1 = mkVarDecl (stringToId "x") aType
v2 = mkVarDecl (stringToId "y") bType
in makeNamed "ga_subt_inj_proj" $ Formula
$ mkForall (map GenTypeVarDecl [aTypeArg, bTypeArg]
++ map GenVarDecl [v1, v2])
$ mkLogTerm implId nr
(mkSubtTerm aType bType v1 v2)
$ mkLogTerm eqvId nr (mkInjEq aType bType v1 v2)
$ mkEqTerm eqId aType nr (QualVar v1)
$ mkTerm projName projType [bType, aType] nr
$ QualVar v2
injTrans :: Named Sentence
injTrans = let
v1 = mkVarDecl (stringToId "x") aType
v2 = mkVarDecl (stringToId "y") bType
v3 = mkVarDecl (stringToId "z") cType
in makeNamed "ga_inj_transitive" $ Formula
$ mkForall (map GenTypeVarDecl [aTypeArg, bTypeArg, cTypeArg]
++ map GenVarDecl [v1, v2, v3])
$ mkLogTerm implId nr
(mkLogTerm andId nr (mkSubtTerm aType bType v1 v2)
$ mkLogTerm andId nr (mkSubtTerm bType cType v2 v3)
$ mkInjEq aType bType v1 v2)
$ mkLogTerm eqvId nr (mkInjEq aType cType v1 v3)
$ mkInjEq bType cType v2 v3
monos :: Env -> [Named Sentence]
monos e = concatMap (makeMonos e) . Map.toList $ assumps e
makeMonos :: Env -> (Id, Set.Set OpInfo) -> [Named Sentence]
makeMonos e (i, s) = makeEquivMonos e i . map opType $ Set.toList s
makeEquivMonos :: Env -> Id -> [TypeScheme] -> [Named Sentence]
makeEquivMonos e i l = case l of
[] -> []
t : r -> mapMaybe (makeEquivMono e i t) r ++ makeEquivMonos e i r
makeEquivMono :: Env -> Id -> TypeScheme -> TypeScheme -> Maybe (Named Sentence)
makeEquivMono e i s1 s2 = case getCommonSupertype e s1 s2 of
Just (ty1, l1, ty2, l2, args, nTy) ->
Just $ makeNamed "ga_monotonicity"
$ Formula $ mkForall (map GenTypeVarDecl args)
$ mkEqTerm eqId nTy nr
(mkInjTerm ty1 nTy
$ QualOp Op (PolyId i [] nr) s1 l1 UserGiven nr)
$ mkInjTerm ty2 nTy
$ QualOp Op (PolyId i [] nr) s2 l2 UserGiven nr
Nothing -> Nothing