| author | wenzelm | 
| Fri, 05 Oct 2007 20:10:33 +0200 | |
| changeset 24857 | 2dde4189a055 | 
| parent 23222 | daca4731942f | 
| child 25473 | 812db0f215b3 | 
| permissions | -rw-r--r-- | 
| 12784 | 1  | 
(* Title: Pure/pattern.ML  | 
| 0 | 2  | 
ID: $Id$  | 
| 12784 | 3  | 
Author: Tobias Nipkow, Christine Heinzelmann, and Stefan Berghofer  | 
| 0 | 4  | 
|
5  | 
Unification of Higher-Order Patterns.  | 
|
6  | 
||
7  | 
See also:  | 
|
8  | 
Tobias Nipkow. Functional Unification of Higher-Order Patterns.  | 
|
9  | 
In Proceedings of the 8th IEEE Symposium Logic in Computer Science, 1993.  | 
|
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
10  | 
|
| 
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
11  | 
TODO: optimize red by special-casing it  | 
| 0 | 12  | 
*)  | 
13  | 
||
| 2751 | 14  | 
infix aeconv;  | 
15  | 
||
| 0 | 16  | 
signature PATTERN =  | 
| 14787 | 17  | 
sig  | 
| 17203 | 18  | 
val trace_unify_fail: bool ref  | 
19  | 
val aeconv: term * term -> bool  | 
|
20  | 
val eta_long: typ list -> term -> term  | 
|
21  | 
val eta_contract_atom: term -> term  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
22  | 
val match: theory -> term * term -> Type.tyenv * Envir.tenv -> Type.tyenv * Envir.tenv  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
23  | 
val first_order_match: theory -> term * term  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
24  | 
-> Type.tyenv * Envir.tenv -> Type.tyenv * Envir.tenv  | 
| 17203 | 25  | 
val matches: theory -> term * term -> bool  | 
| 19880 | 26  | 
val equiv: theory -> term * term -> bool  | 
| 17203 | 27  | 
val matches_subterm: theory -> term * term -> bool  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
28  | 
val unify: theory -> term * term -> Envir.env -> Envir.env  | 
| 17203 | 29  | 
val first_order: term -> bool  | 
30  | 
val pattern: term -> bool  | 
|
31  | 
val rewrite_term: theory -> (term * term) list -> (term -> term option) list -> term -> term  | 
|
| 0 | 32  | 
exception Unif  | 
33  | 
exception MATCH  | 
|
34  | 
exception Pattern  | 
|
| 14787 | 35  | 
end;  | 
| 0 | 36  | 
|
| 17203 | 37  | 
structure Pattern: PATTERN =  | 
| 0 | 38  | 
struct  | 
39  | 
||
40  | 
exception Unif;  | 
|
41  | 
exception Pattern;  | 
|
42  | 
||
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
43  | 
val trace_unify_fail = ref false;  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
44  | 
|
| 17203 | 45  | 
fun string_of_term thy env binders t = Sign.string_of_term thy  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
46  | 
(Envir.norm_term env (subst_bounds(map Free binders,t)));  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
47  | 
|
| 
18011
 
685d95c793ff
cleaned up nth, nth_update, nth_map and nth_string functions
 
haftmann 
parents: 
17756 
diff
changeset
 | 
48  | 
fun bname binders i = fst (nth binders i);  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
49  | 
fun bnames binders is = space_implode " " (map (bname binders) is);  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
50  | 
|
| 17203 | 51  | 
fun typ_clash thy (tye,T,U) =  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
52  | 
if !trace_unify_fail  | 
| 17203 | 53  | 
then let val t = Sign.string_of_typ thy (Envir.norm_type tye T)  | 
54  | 
and u = Sign.string_of_typ thy (Envir.norm_type tye U)  | 
|
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
55  | 
       in tracing("The following types do not unify:\n" ^ t ^ "\n" ^ u) end
 | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
56  | 
else ()  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
57  | 
|
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
58  | 
fun clash a b =  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
59  | 
  if !trace_unify_fail then tracing("Clash: " ^ a ^ " =/= " ^ b) else ()
 | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
60  | 
|
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
61  | 
fun boundVar binders i =  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
62  | 
"bound variable " ^ bname binders i ^ " (depth " ^ string_of_int i ^ ")";  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
63  | 
|
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
64  | 
fun clashBB binders i j =  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
65  | 
if !trace_unify_fail then clash (boundVar binders i) (boundVar binders j)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
66  | 
else ()  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
67  | 
|
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
68  | 
fun clashB binders i s =  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
69  | 
if !trace_unify_fail then clash (boundVar binders i) s  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
70  | 
else ()  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
71  | 
|
| 17203 | 72  | 
fun proj_fail thy (env,binders,F,_,is,t) =  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
73  | 
if !trace_unify_fail  | 
| 22678 | 74  | 
then let val f = Term.string_of_vname F  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
75  | 
val xs = bnames binders is  | 
| 17203 | 76  | 
val u = string_of_term thy env binders t  | 
| 19300 | 77  | 
val ys = bnames binders (subtract (op =) is (loose_bnos t))  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
78  | 
       in tracing("Cannot unify variable " ^ f ^
 | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
79  | 
" (depending on bound variables " ^ xs ^ ")\nwith term " ^ u ^  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
80  | 
"\nTerm contains additional bound variable(s) " ^ ys)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
81  | 
end  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
82  | 
else ()  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
83  | 
|
| 17203 | 84  | 
fun ocheck_fail thy (F,t,binders,env) =  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
85  | 
if !trace_unify_fail  | 
| 22678 | 86  | 
then let val f = Term.string_of_vname F  | 
| 17203 | 87  | 
val u = string_of_term thy env binders t  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
88  | 
       in tracing("Variable " ^ f ^ " occurs in term\n" ^ u ^
 | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
89  | 
"\nCannot unify!\n")  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
90  | 
end  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
91  | 
else ()  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
92  | 
|
| 12784 | 93  | 
fun occurs(F,t,env) =  | 
| 15797 | 94  | 
let fun occ(Var (G, T)) = (case Envir.lookup (env, (G, T)) of  | 
| 15531 | 95  | 
SOME(t) => occ t  | 
96  | 
| NONE => F=G)  | 
|
| 0 | 97  | 
| occ(t1$t2) = occ t1 orelse occ t2  | 
98  | 
| occ(Abs(_,_,t)) = occ t  | 
|
99  | 
| occ _ = false  | 
|
100  | 
in occ t end;  | 
|
101  | 
||
102  | 
||
103  | 
fun mapbnd f =  | 
|
104  | 
let fun mpb d (Bound(i)) = if i < d then Bound(i) else Bound(f(i-d)+d)  | 
|
105  | 
| mpb d (Abs(s,T,t)) = Abs(s,T,mpb(d+1) t)  | 
|
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
106  | 
| mpb d ((u1 $ u2)) = (mpb d u1)$(mpb d u2)  | 
| 
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
107  | 
| mpb _ atom = atom  | 
| 0 | 108  | 
in mpb 0 end;  | 
109  | 
||
| 
14861
 
ca5cae7fb65a
Removed ~10000 hack in function idx that can lead to inconsistencies
 
berghofe 
parents: 
14787 
diff
changeset
 | 
110  | 
fun idx [] j = raise Unif  | 
| 16668 | 111  | 
| idx(i::is) j = if (i:int) =j then length is else idx is j;  | 
| 0 | 112  | 
|
113  | 
fun mkabs (binders,is,t) =  | 
|
| 
18011
 
685d95c793ff
cleaned up nth, nth_update, nth_map and nth_string functions
 
haftmann 
parents: 
17756 
diff
changeset
 | 
114  | 
let fun mk(i::is) = let val (x,T) = nth binders i  | 
| 12784 | 115  | 
in Abs(x,T,mk is) end  | 
| 0 | 116  | 
| mk [] = t  | 
117  | 
in mk is end;  | 
|
118  | 
||
119  | 
val incr = mapbnd (fn i => i+1);  | 
|
120  | 
||
121  | 
fun ints_of [] = []  | 
|
| 12784 | 122  | 
| ints_of (Bound i ::bs) =  | 
| 0 | 123  | 
let val is = ints_of bs  | 
| 20672 | 124  | 
in if member (op =) is i then raise Pattern else i::is end  | 
| 0 | 125  | 
| ints_of _ = raise Pattern;  | 
126  | 
||
| 12232 | 127  | 
fun ints_of' env ts = ints_of (map (Envir.head_norm env) ts);  | 
128  | 
||
| 0 | 129  | 
|
130  | 
fun app (s,(i::is)) = app (s$Bound(i),is)  | 
|
131  | 
| app (s,[]) = s;  | 
|
132  | 
||
133  | 
fun red (Abs(_,_,s)) (i::is) js = red s is (i::js)  | 
|
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
134  | 
| red t [] [] = t  | 
| 
18011
 
685d95c793ff
cleaned up nth, nth_update, nth_map and nth_string functions
 
haftmann 
parents: 
17756 
diff
changeset
 | 
135  | 
| red t is jn = app (mapbnd (nth jn) t,is);  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
136  | 
|
| 0 | 137  | 
|
138  | 
(* split_type ([T1,....,Tn]---> T,n,[]) = ([Tn,...,T1],T) *)  | 
|
139  | 
fun split_type (T,0,Ts) = (Ts,T)  | 
|
140  | 
  | split_type (Type ("fun",[T1,T2]),n,Ts) = split_type (T2,n-1,T1::Ts)
 | 
|
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
141  | 
  | split_type _                           = error("split_type");
 | 
| 0 | 142  | 
|
| 
13891
 
ae9a2a433388
type_of_G now applies type substitution before decomposing type.
 
berghofe 
parents: 
13645 
diff
changeset
 | 
143  | 
fun type_of_G (env as Envir.Envir {iTs, ...}) (T,n,is) =
 | 
| 
 
ae9a2a433388
type_of_G now applies type substitution before decomposing type.
 
berghofe 
parents: 
13645 
diff
changeset
 | 
144  | 
let val (Ts, U) = split_type (Envir.norm_type iTs T, n, [])  | 
| 
18011
 
685d95c793ff
cleaned up nth, nth_update, nth_map and nth_string functions
 
haftmann 
parents: 
17756 
diff
changeset
 | 
145  | 
in map (nth Ts) is ---> U end;  | 
| 0 | 146  | 
|
147  | 
fun mkhnf (binders,is,G,js) = mkabs (binders, is, app(G,js));  | 
|
148  | 
||
149  | 
fun mknewhnf(env,binders,is,F as (a,_),T,js) =  | 
|
| 
13891
 
ae9a2a433388
type_of_G now applies type substitution before decomposing type.
 
berghofe 
parents: 
13645 
diff
changeset
 | 
150  | 
let val (env',G) = Envir.genvar a (env,type_of_G env (T,length is,js))  | 
| 15797 | 151  | 
in Envir.update (((F, T), mkhnf (binders, is, G, js)), env') end;  | 
| 0 | 152  | 
|
153  | 
||
| 23222 | 154  | 
(*predicate: downto0 (is, n) <=> is = [n, n - 1, ..., 0]*)  | 
155  | 
fun downto0 (i :: is, n) = i = n andalso downto0 (is, n - 1)  | 
|
156  | 
| downto0 ([], n) = n = ~1;  | 
|
157  | 
||
158  | 
(*mk_proj_list(is) = [ |is| - k | 1 <= k <= |is| and is[k] >= 0 ]*)  | 
|
| 0 | 159  | 
fun mk_proj_list is =  | 
| 19502 | 160  | 
let fun mk(i::is,j) = if is_some i then j :: mk(is,j-1) else mk(is,j-1)  | 
| 0 | 161  | 
| mk([],_) = []  | 
162  | 
in mk(is,length is - 1) end;  | 
|
163  | 
||
164  | 
fun proj(s,env,binders,is) =  | 
|
165  | 
let fun trans d i = if i<d then i else (idx is (i-d))+d;  | 
|
| 12232 | 166  | 
fun pr(s,env,d,binders) = (case Envir.head_norm env s of  | 
| 0 | 167  | 
Abs(a,T,t) => let val (t',env') = pr(t,env,d+1,((a,T)::binders))  | 
168  | 
in (Abs(a,T,t'),env') end  | 
|
169  | 
| t => (case strip_comb t of  | 
|
170  | 
(c as Const _,ts) =>  | 
|
171  | 
let val (ts',env') = prs(ts,env,d,binders)  | 
|
172  | 
in (list_comb(c,ts'),env') end  | 
|
173  | 
| (f as Free _,ts) =>  | 
|
174  | 
let val (ts',env') = prs(ts,env,d,binders)  | 
|
175  | 
in (list_comb(f,ts'),env') end  | 
|
176  | 
| (Bound(i),ts) =>  | 
|
177  | 
let val j = trans d i  | 
|
| 
14861
 
ca5cae7fb65a
Removed ~10000 hack in function idx that can lead to inconsistencies
 
berghofe 
parents: 
14787 
diff
changeset
 | 
178  | 
val (ts',env') = prs(ts,env,d,binders)  | 
| 
 
ca5cae7fb65a
Removed ~10000 hack in function idx that can lead to inconsistencies
 
berghofe 
parents: 
14787 
diff
changeset
 | 
179  | 
in (list_comb(Bound j,ts'),env') end  | 
| 0 | 180  | 
| (Var(F as (a,_),Fty),ts) =>  | 
| 12232 | 181  | 
let val js = ints_of' env ts;  | 
| 
14861
 
ca5cae7fb65a
Removed ~10000 hack in function idx that can lead to inconsistencies
 
berghofe 
parents: 
14787 
diff
changeset
 | 
182  | 
val js' = map (try (trans d)) js;  | 
| 0 | 183  | 
val ks = mk_proj_list js';  | 
| 
19482
 
9f11af8f7ef9
tuned basic list operators (flat, maps, map_filter);
 
wenzelm 
parents: 
19300 
diff
changeset
 | 
184  | 
val ls = map_filter I js'  | 
| 
13891
 
ae9a2a433388
type_of_G now applies type substitution before decomposing type.
 
berghofe 
parents: 
13645 
diff
changeset
 | 
185  | 
val Hty = type_of_G env (Fty,length js,ks)  | 
| 0 | 186  | 
val (env',H) = Envir.genvar a (env,Hty)  | 
187  | 
val env'' =  | 
|
| 15797 | 188  | 
Envir.update (((F, Fty), mkhnf (binders, js, H, ks)), env')  | 
| 0 | 189  | 
in (app(H,ls),env'') end  | 
190  | 
| _ => raise Pattern))  | 
|
191  | 
and prs(s::ss,env,d,binders) =  | 
|
192  | 
let val (s',env1) = pr(s,env,d,binders)  | 
|
193  | 
val (ss',env2) = prs(ss,env1,d,binders)  | 
|
194  | 
in (s'::ss',env2) end  | 
|
195  | 
| prs([],env,_,_) = ([],env)  | 
|
196  | 
in if downto0(is,length binders - 1) then (s,env)  | 
|
197  | 
else pr(s,env,0,binders)  | 
|
198  | 
end;  | 
|
199  | 
||
200  | 
||
201  | 
(* mk_ff_list(is,js) = [ length(is) - k | 1 <= k <= |is| and is[k] = js[k] ] *)  | 
|
| 12784 | 202  | 
fun mk_ff_list(is,js) =  | 
203  | 
let fun mk([],[],_) = []  | 
|
| 16668 | 204  | 
| mk(i::is,j::js, k) = if (i:int) = j then k :: mk(is,js,k-1)  | 
| 0 | 205  | 
else mk(is,js,k-1)  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
206  | 
| mk _ = error"mk_ff_list"  | 
| 0 | 207  | 
in mk(is,js,length is-1) end;  | 
208  | 
||
209  | 
fun flexflex1(env,binders,F,Fty,is,js) =  | 
|
210  | 
if is=js then env  | 
|
211  | 
else let val ks = mk_ff_list(is,js)  | 
|
212  | 
in mknewhnf(env,binders,is,F,Fty,ks) end;  | 
|
213  | 
||
214  | 
fun flexflex2(env,binders,F,Fty,is,G,Gty,js) =  | 
|
215  | 
let fun ff(F,Fty,is,G as (a,_),Gty,js) =  | 
|
| 
1576
 
af8f43f742a0
Added some optimized versions of functions dealing with sets
 
berghofe 
parents: 
1501 
diff
changeset
 | 
216  | 
if js subset_int is  | 
| 0 | 217  | 
then let val t= mkabs(binders,is,app(Var(G,Gty),map (idx is) js))  | 
| 15797 | 218  | 
in Envir.update (((F, Fty), t), env) end  | 
| 
1576
 
af8f43f742a0
Added some optimized versions of functions dealing with sets
 
berghofe 
parents: 
1501 
diff
changeset
 | 
219  | 
else let val ks = is inter_int js  | 
| 
13891
 
ae9a2a433388
type_of_G now applies type substitution before decomposing type.
 
berghofe 
parents: 
13645 
diff
changeset
 | 
220  | 
val Hty = type_of_G env (Fty,length is,map (idx is) ks)  | 
| 0 | 221  | 
val (env',H) = Envir.genvar a (env,Hty)  | 
222  | 
fun lam(is) = mkabs(binders,is,app(H,map (idx is) ks));  | 
|
| 15797 | 223  | 
in Envir.update (((G, Gty), lam js), Envir.update (((F, Fty), lam is), env'))  | 
| 0 | 224  | 
end;  | 
| 20098 | 225  | 
in if Term.indexname_ord (G,F) = LESS then ff(F,Fty,is,G,Gty,js) else ff(G,Gty,js,F,Fty,is) end  | 
| 0 | 226  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
227  | 
fun unify_types thy (T,U) (env as Envir.Envir{asol,iTs,maxidx}) =
 | 
| 0 | 228  | 
if T=U then env  | 
| 17203 | 229  | 
else let val (iTs',maxidx') = Sign.typ_unify thy (U, T) (iTs, maxidx)  | 
| 
1435
 
aefcd255ed4a
Removed bug in type unification. Negative indexes are not used any longer.
 
nipkow 
parents: 
1029 
diff
changeset
 | 
230  | 
       in Envir.Envir{asol=asol,maxidx=maxidx',iTs=iTs'} end
 | 
| 17203 | 231  | 
handle Type.TUNIFY => (typ_clash thy (iTs,T,U); raise Unif);  | 
| 0 | 232  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
233  | 
fun unif thy binders (s,t) env = case (Envir.head_norm env s, Envir.head_norm env t) of  | 
| 0 | 234  | 
(Abs(ns,Ts,ts),Abs(nt,Tt,tt)) =>  | 
235  | 
let val name = if ns = "" then nt else ns  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
236  | 
in unif thy ((name,Ts)::binders) (ts,tt) env end  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
237  | 
| (Abs(ns,Ts,ts),t) => unif thy ((ns,Ts)::binders) (ts,(incr t)$Bound(0)) env  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
238  | 
| (t,Abs(nt,Tt,tt)) => unif thy ((nt,Tt)::binders) ((incr t)$Bound(0),tt) env  | 
| 17203 | 239  | 
| p => cases thy (binders,env,p)  | 
| 0 | 240  | 
|
| 17203 | 241  | 
and cases thy (binders,env,(s,t)) = case (strip_comb s,strip_comb t) of  | 
| 12784 | 242  | 
((Var(F,Fty),ss),(Var(G,Gty),ts)) =>  | 
| 12232 | 243  | 
if F = G then flexflex1(env,binders,F,Fty,ints_of' env ss,ints_of' env ts)  | 
244  | 
else flexflex2(env,binders,F,Fty,ints_of' env ss,G,Gty,ints_of' env ts)  | 
|
| 17203 | 245  | 
| ((Var(F,Fty),ss),_) => flexrigid thy (env,binders,F,Fty,ints_of' env ss,t)  | 
246  | 
| (_,(Var(F,Fty),ts)) => flexrigid thy (env,binders,F,Fty,ints_of' env ts,s)  | 
|
247  | 
| ((Const c,ss),(Const d,ts)) => rigidrigid thy (env,binders,c,d,ss,ts)  | 
|
248  | 
| ((Free(f),ss),(Free(g),ts)) => rigidrigid thy (env,binders,f,g,ss,ts)  | 
|
249  | 
| ((Bound(i),ss),(Bound(j),ts)) => rigidrigidB thy (env,binders,i,j,ss,ts)  | 
|
| 0 | 250  | 
| ((Abs(_),_),_) => raise Pattern  | 
251  | 
| (_,(Abs(_),_)) => raise Pattern  | 
|
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
252  | 
| ((Const(c,_),_),(Free(f,_),_)) => (clash c f; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
253  | 
| ((Const(c,_),_),(Bound i,_)) => (clashB binders i c; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
254  | 
| ((Free(f,_),_),(Const(c,_),_)) => (clash f c; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
255  | 
| ((Free(f,_),_),(Bound i,_)) => (clashB binders i f; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
256  | 
| ((Bound i,_),(Const(c,_),_)) => (clashB binders i c; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
257  | 
| ((Bound i,_),(Free(f,_),_)) => (clashB binders i f; raise Unif)  | 
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
258  | 
|
| 0 | 259  | 
|
| 17203 | 260  | 
and rigidrigid thy (env,binders,(a,Ta),(b,Tb),ss,ts) =  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
261  | 
if a<>b then (clash a b; raise Unif)  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
262  | 
else env |> unify_types thy (Ta,Tb) |> fold (unif thy binders) (ss~~ts)  | 
| 0 | 263  | 
|
| 17203 | 264  | 
and rigidrigidB thy (env,binders,i,j,ss,ts) =  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
265  | 
if i <> j then (clashBB binders i j; raise Unif)  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
266  | 
else fold (unif thy binders) (ss~~ts) env  | 
| 0 | 267  | 
|
| 17203 | 268  | 
and flexrigid thy (params as (env,binders,F,Fty,is,t)) =  | 
269  | 
if occurs(F,t,env) then (ocheck_fail thy (F,t,binders,env); raise Unif)  | 
|
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
270  | 
else (let val (u,env') = proj(t,env,binders,is)  | 
| 15797 | 271  | 
in Envir.update (((F, Fty), mkabs (binders, is, u)), env') end  | 
| 17203 | 272  | 
handle Unif => (proj_fail thy params; raise Unif));  | 
| 0 | 273  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
274  | 
fun unify thy = unif thy [];  | 
| 0 | 275  | 
|
276  | 
||
| 
6539
 
2e7d2fba9f6c
Eta contraction is now performed all the time during rewriting.
 
nipkow 
parents: 
4820 
diff
changeset
 | 
277  | 
(*Eta-contract a term from outside: just enough to reduce it to an atom  | 
| 
 
2e7d2fba9f6c
Eta contraction is now performed all the time during rewriting.
 
nipkow 
parents: 
4820 
diff
changeset
 | 
278  | 
DOESN'T QUITE WORK!  | 
| 
 
2e7d2fba9f6c
Eta contraction is now performed all the time during rewriting.
 
nipkow 
parents: 
4820 
diff
changeset
 | 
279  | 
*)  | 
| 12784 | 280  | 
fun eta_contract_atom (t0 as Abs(a, T, body)) =  | 
| 
2725
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
281  | 
(case eta_contract2 body of  | 
| 12784 | 282  | 
body' as (f $ Bound 0) =>  | 
283  | 
if loose_bvar1(f,0) then Abs(a,T,body')  | 
|
284  | 
else eta_contract_atom (incr_boundvars ~1 f)  | 
|
| 
2725
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
285  | 
| _ => t0)  | 
| 
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
286  | 
| eta_contract_atom t = t  | 
| 
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
287  | 
and eta_contract2 (f$t) = f $ eta_contract_atom t  | 
| 
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
288  | 
| eta_contract2 t = eta_contract_atom t;  | 
| 
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
289  | 
|
| 13998 | 290  | 
(* put a term into eta long beta normal form *)  | 
291  | 
fun eta_long Ts (Abs (s, T, t)) = Abs (s, T, eta_long (T :: Ts) t)  | 
|
292  | 
| eta_long Ts t = (case strip_comb t of  | 
|
293  | 
(Abs _, _) => eta_long Ts (Envir.beta_norm t)  | 
|
| 14787 | 294  | 
| (u, ts) =>  | 
| 13998 | 295  | 
let  | 
296  | 
val Us = binder_types (fastype_of1 (Ts, t));  | 
|
297  | 
val i = length Us  | 
|
298  | 
in list_abs (map (pair "x") Us,  | 
|
299  | 
list_comb (incr_boundvars i u, map (eta_long (rev Us @ Ts))  | 
|
300  | 
(map (incr_boundvars i) ts @ map Bound (i - 1 downto 0))))  | 
|
301  | 
end);  | 
|
302  | 
||
| 
2725
 
9453616d4b80
Declares eta_contract_atom; fixed comment; some tidying
 
paulson 
parents: 
2616 
diff
changeset
 | 
303  | 
|
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
304  | 
(*Tests whether 2 terms are alpha/eta-convertible and have same type.  | 
| 
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
305  | 
Note that Consts and Vars may have more than one type.*)  | 
| 2751 | 306  | 
fun t aeconv u = aconv_aux (eta_contract_atom t, eta_contract_atom u)  | 
307  | 
and aconv_aux (Const(a,T), Const(b,U)) = a=b andalso T=U  | 
|
308  | 
| aconv_aux (Free(a,T), Free(b,U)) = a=b andalso T=U  | 
|
309  | 
| aconv_aux (Var(v,T), Var(w,U)) = eq_ix(v,w) andalso T=U  | 
|
| 12784 | 310  | 
| aconv_aux (Bound i, Bound j) = i=j  | 
| 2751 | 311  | 
| aconv_aux (Abs(_,T,t), Abs(_,U,u)) = (t aeconv u) andalso T=U  | 
| 12784 | 312  | 
| aconv_aux (f$t, g$u) = (f aeconv g) andalso (t aeconv u)  | 
| 2751 | 313  | 
| aconv_aux _ = false;  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
314  | 
|
| 
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
315  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
316  | 
(*** Matching ***)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
317  | 
|
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
318  | 
exception MATCH;  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
319  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
320  | 
fun typ_match thy TU tyenv = Sign.typ_match thy TU tyenv  | 
| 16939 | 321  | 
handle Type.TYPE_MATCH => raise MATCH;  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
322  | 
|
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
323  | 
(*First-order matching;  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
324  | 
The pattern and object may have variables in common.  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
325  | 
Instantiation does not affect the object, so matching ?a with ?a+1 works.  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
326  | 
Object is eta-contracted on the fly (by eta-expanding the pattern).  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
327  | 
Precondition: the pattern is already eta-contracted!  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
328  | 
Types are matched on the fly*)  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
329  | 
fun first_order_match thy =  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
330  | 
let  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
331  | 
fun mtch (instsp as (tyinsts,insts)) = fn  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
332  | 
(Var(ixn,T), t) =>  | 
| 12784 | 333  | 
if loose_bvar(t,0) then raise MATCH  | 
| 16651 | 334  | 
else (case Envir.lookup' (insts, (ixn, T)) of  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
335  | 
NONE => (typ_match thy (T, fastype_of t) tyinsts,  | 
| 17412 | 336  | 
Vartab.update_new (ixn, (T, t)) insts)  | 
| 15531 | 337  | 
| SOME u => if t aeconv u then instsp else raise MATCH)  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
338  | 
| (Free (a,T), Free (b,U)) =>  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
339  | 
if a=b then (typ_match thy (T,U) tyinsts, insts) else raise MATCH  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
340  | 
| (Const (a,T), Const (b,U)) =>  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
341  | 
if a=b then (typ_match thy (T,U) tyinsts, insts) else raise MATCH  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
342  | 
| (Bound i, Bound j) => if i=j then instsp else raise MATCH  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
343  | 
| (Abs(_,T,t), Abs(_,U,u)) =>  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
344  | 
mtch (typ_match thy (T,U) tyinsts, insts) (t,u)  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
345  | 
| (f$t, g$u) => mtch (mtch instsp (f,g)) (t, u)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
346  | 
| (t, Abs(_,U,u)) => mtch instsp ((incr t)$(Bound 0), u)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
347  | 
| _ => raise MATCH  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
348  | 
in fn tu => fn env => mtch env tu end;  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
349  | 
|
| 
8406
 
a217b0cd304d
Type.unify and Type.typ_match now use Vartab instead of association lists.
 
berghofe 
parents: 
6539 
diff
changeset
 | 
350  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
351  | 
(* Matching of higher-order patterns *)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
352  | 
|
| 15797 | 353  | 
fun match_bind(itms,binders,ixn,T,is,t) =  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
354  | 
let val js = loose_bnos t  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
355  | 
in if null is  | 
| 17412 | 356  | 
then if null js then Vartab.update_new (ixn, (T, t)) itms else raise MATCH  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
357  | 
else if js subset_int is  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
358  | 
then let val t' = if downto0(is,length binders - 1) then t  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
359  | 
else mapbnd (idx is) t  | 
| 17412 | 360  | 
in Vartab.update_new (ixn, (T, mkabs (binders, is, t'))) itms end  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
361  | 
else raise MATCH  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
362  | 
end;  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
363  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
364  | 
fun match thy (po as (pat,obj)) envir =  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
365  | 
let  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
366  | 
(* Pre: pat and obj have same type *)  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
367  | 
fun mtch binders (pat,obj) (env as (iTs,itms)) =  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
368  | 
case pat of  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
369  | 
Abs(ns,Ts,ts) =>  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
370  | 
(case obj of  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
371  | 
Abs(nt,Tt,tt) => mtch ((nt,Tt)::binders) (ts,tt) env  | 
| 15797 | 372  | 
| _ => let val Tt = Envir.typ_subst_TVars iTs Ts  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
373  | 
in mtch((ns,Tt)::binders) (ts,(incr obj)$Bound(0)) env end)  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
374  | 
| _ => (case obj of  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
375  | 
Abs(nt,Tt,tt) =>  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
376  | 
mtch((nt,Tt)::binders) ((incr pat)$Bound(0),tt) env  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
377  | 
| _ => cases(binders,env,pat,obj))  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
378  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
379  | 
and cases(binders,env as (iTs,itms),pat,obj) =  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
380  | 
let val (ph,pargs) = strip_comb pat  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
381  | 
fun rigrig1(iTs,oargs) = fold (mtch binders) (pargs~~oargs) (iTs,itms)  | 
| 16668 | 382  | 
fun rigrig2((a:string,Ta),(b,Tb),oargs) =  | 
383  | 
if a <> b then raise MATCH  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
384  | 
else rigrig1(typ_match thy (Ta,Tb) iTs, oargs)  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
385  | 
in case ph of  | 
| 15797 | 386  | 
Var(ixn,T) =>  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
387  | 
let val is = ints_of pargs  | 
| 16651 | 388  | 
in case Envir.lookup' (itms, (ixn, T)) of  | 
| 15797 | 389  | 
NONE => (iTs,match_bind(itms,binders,ixn,T,is,obj))  | 
| 15531 | 390  | 
| SOME u => if obj aeconv (red u is []) then env  | 
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
391  | 
else raise MATCH  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
392  | 
end  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
393  | 
| _ =>  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
394  | 
let val (oh,oargs) = strip_comb obj  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
395  | 
in case (ph,oh) of  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
396  | 
(Const c,Const d) => rigrig2(c,d,oargs)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
397  | 
| (Free f,Free g) => rigrig2(f,g,oargs)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
398  | 
| (Bound i,Bound j) => if i<>j then raise MATCH  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
399  | 
else rigrig1(iTs,oargs)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
400  | 
| (Abs _, _) => raise Pattern  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
401  | 
| (_, Abs _) => raise Pattern  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
402  | 
| _ => raise MATCH  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
403  | 
end  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
404  | 
end;  | 
| 
678
 
6151b7f3b606
Modified pattern.ML to perform proper matching of Higher-Order Patterns.
 
nipkow 
parents: 
63 
diff
changeset
 | 
405  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
406  | 
val pT = fastype_of pat  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
407  | 
and oT = fastype_of obj  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
408  | 
val envir' = apfst (typ_match thy (pT, oT)) envir;  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
409  | 
in mtch [] po envir' handle Pattern => first_order_match thy po envir' end;  | 
| 0 | 410  | 
|
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
411  | 
fun matches thy po = (match thy po (Vartab.empty, Vartab.empty); true) handle MATCH => false;  | 
| 
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
412  | 
|
| 19880 | 413  | 
fun equiv thy (t, u) = matches thy (t, u) andalso matches thy (u, t);  | 
414  | 
||
| 0 | 415  | 
|
| 
4667
 
6328d427a339
Tried to reorganize rewriter a little. More to be done.
 
nipkow 
parents: 
2792 
diff
changeset
 | 
416  | 
(* Does pat match a subterm of obj? *)  | 
| 22255 | 417  | 
fun matches_subterm thy (pat, obj) =  | 
418  | 
let  | 
|
419  | 
fun msub bounds obj = matches thy (pat, obj) orelse  | 
|
420  | 
(case obj of  | 
|
421  | 
Abs (x, T, t) => msub (bounds + 1) (snd (Term.dest_abs (Name.bound bounds, T, t)))  | 
|
422  | 
| t $ u => msub bounds t orelse msub bounds u  | 
|
423  | 
| _ => false)  | 
|
424  | 
in msub 0 obj end;  | 
|
| 
4667
 
6328d427a339
Tried to reorganize rewriter a little. More to be done.
 
nipkow 
parents: 
2792 
diff
changeset
 | 
425  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
426  | 
fun first_order(Abs(_,_,t)) = first_order t  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
427  | 
| first_order(t $ u) = first_order t andalso first_order u andalso  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
428  | 
not(is_Var t)  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
429  | 
| first_order _ = true;  | 
| 
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
430  | 
|
| 20672 | 431  | 
fun pattern (Abs (_, _, t)) = pattern t  | 
432  | 
| pattern t =  | 
|
433  | 
let val (head, args) = strip_comb t in  | 
|
434  | 
if is_Var head then  | 
|
435  | 
forall is_Bound args andalso not (has_duplicates (op aconv) args)  | 
|
436  | 
else forall pattern args  | 
|
437  | 
end;  | 
|
| 
4820
 
8f6dbbd8d497
Tried to speed up the rewriter by eta-contracting all patterns beforehand and
 
nipkow 
parents: 
4667 
diff
changeset
 | 
438  | 
|
| 12784 | 439  | 
|
440  | 
(* rewriting -- simple but fast *)  | 
|
441  | 
||
| 17203 | 442  | 
fun rewrite_term thy rules procs tm =  | 
| 12784 | 443  | 
let  | 
| 13195 | 444  | 
val skel0 = Bound 0;  | 
445  | 
||
| 16939 | 446  | 
fun variant_absfree bounds (x, T, t) =  | 
| 12797 | 447  | 
let  | 
| 
20079
 
ec5c8584487c
replaced Term.variant(list) by Name.variant(_list);
 
wenzelm 
parents: 
19880 
diff
changeset
 | 
448  | 
val (x', t') = Term.dest_abs (Name.bound bounds, T, t);  | 
| 16939 | 449  | 
fun abs u = Abs (x, T, abstract_over (Free (x', T), u));  | 
450  | 
in (abs, t') end;  | 
|
| 12797 | 451  | 
|
| 12784 | 452  | 
fun match_rew tm (tm1, tm2) =  | 
| 18940 | 453  | 
let val rtm = the_default tm2 (Term.rename_abs tm1 tm tm2) in  | 
| 
18182
 
786d83044780
tuned interfaces to support incremental match/unify (cf. versions in type.ML);
 
wenzelm 
parents: 
18011 
diff
changeset
 | 
454  | 
SOME (Envir.subst_vars (match thy (tm1, tm) (Vartab.empty, Vartab.empty)) rtm, rtm)  | 
| 16939 | 455  | 
handle MATCH => NONE  | 
| 13195 | 456  | 
end;  | 
| 12784 | 457  | 
|
| 15531 | 458  | 
fun rew (Abs (_, _, body) $ t) = SOME (subst_bound (t, body), skel0)  | 
| 13195 | 459  | 
| rew tm = (case get_first (match_rew tm) rules of  | 
| 15570 | 460  | 
NONE => Option.map (rpair skel0) (get_first (fn p => p tm) procs)  | 
| 13195 | 461  | 
| x => x);  | 
462  | 
||
| 16939 | 463  | 
fun rew1 bounds (Var _) _ = NONE  | 
464  | 
| rew1 bounds skel tm = (case rew2 bounds skel tm of  | 
|
| 15531 | 465  | 
SOME tm1 => (case rew tm1 of  | 
| 18940 | 466  | 
SOME (tm2, skel') => SOME (the_default tm2 (rew1 bounds skel' tm2))  | 
| 15531 | 467  | 
| NONE => SOME tm1)  | 
468  | 
| NONE => (case rew tm of  | 
|
| 18940 | 469  | 
SOME (tm1, skel') => SOME (the_default tm1 (rew1 bounds skel' tm1))  | 
| 15531 | 470  | 
| NONE => NONE))  | 
| 12784 | 471  | 
|
| 16939 | 472  | 
and rew2 bounds skel (tm1 $ tm2) = (case tm1 of  | 
| 12784 | 473  | 
Abs (_, _, body) =>  | 
474  | 
let val tm' = subst_bound (tm2, body)  | 
|
| 18940 | 475  | 
in SOME (the_default tm' (rew2 bounds skel0 tm')) end  | 
| 14787 | 476  | 
| _ =>  | 
| 13195 | 477  | 
let val (skel1, skel2) = (case skel of  | 
478  | 
skel1 $ skel2 => (skel1, skel2)  | 
|
479  | 
| _ => (skel0, skel0))  | 
|
| 16939 | 480  | 
in case rew1 bounds skel1 tm1 of  | 
481  | 
SOME tm1' => (case rew1 bounds skel2 tm2 of  | 
|
| 15531 | 482  | 
SOME tm2' => SOME (tm1' $ tm2')  | 
483  | 
| NONE => SOME (tm1' $ tm2))  | 
|
| 16939 | 484  | 
| NONE => (case rew1 bounds skel2 tm2 of  | 
| 15531 | 485  | 
SOME tm2' => SOME (tm1 $ tm2')  | 
486  | 
| NONE => NONE)  | 
|
| 13195 | 487  | 
end)  | 
| 16939 | 488  | 
| rew2 bounds skel (Abs body) =  | 
| 13195 | 489  | 
let  | 
| 16939 | 490  | 
val (abs, tm') = variant_absfree bounds body;  | 
| 13195 | 491  | 
val skel' = (case skel of Abs (_, _, skel') => skel' | _ => skel0)  | 
| 16939 | 492  | 
in case rew1 (bounds + 1) skel' tm' of  | 
| 15531 | 493  | 
SOME tm'' => SOME (abs tm'')  | 
494  | 
| NONE => NONE  | 
|
| 12797 | 495  | 
end  | 
| 16939 | 496  | 
| rew2 _ _ _ = NONE;  | 
| 12784 | 497  | 
|
| 18940 | 498  | 
in the_default tm (rew1 0 skel0 tm) end;  | 
| 12784 | 499  | 
|
| 0 | 500  | 
end;  | 
| 
13642
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
501  | 
|
| 
 
a3d97348ceb6
added failure trace information to pattern unification
 
nipkow 
parents: 
13195 
diff
changeset
 | 
502  | 
val trace_unify_fail = Pattern.trace_unify_fail;  |