diff --git a/src/interpretation.ml b/src/interpretation.ml
index b3d3c44645cd6cbd0089f3ea2b4609fcdc5bb983..c92540539c2bccff2f3c72af93219c3306e88169 100644
--- a/src/interpretation.ml
+++ b/src/interpretation.ml
@@ -93,6 +93,8 @@ let const_real_of_float value =
   in
   Constant.ConstReal (Number.real_literal ~radix:10 ~neg ~int ~frac ~exp:None)
 
+let term t = CRE.Value (Term t)
+
 let builtin_caisar : caisar_env CRE.built_in_theories list =
   let error_message ls =
     Fmt.str "Invalid arguments for '%a'" Pretty.print_ls ls
@@ -127,14 +129,14 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
           ( term_of_caisar_op engine (Data (D_csv features)) ty_features,
             Term.t_int_const (BigInt.of_int (Int.of_string label)) )
         in
-        Term (Term.t_tuple [ t_features; t_label ])
+        term (Term.t_tuple [ t_features; t_label ])
       | Vector n ->
         assert (List.length tl1 = n && i <= n);
-        Term (List.nth_exn tl1 i)
+        term (List.nth_exn tl1 i)
       | Data _ | Classifier _ | Tensor _ | Index _ -> assert false)
     | [ Term t1; Term t2 ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t1; t2 ])
+      term (Term.t_app_infer ls [ t1; t2 ])
     | _ -> invalid_arg (error_message ls)
   in
   let length : _ CRE.builtin =
@@ -145,11 +147,11 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
     match vl with
     | [ Term { t_node = Tapp (ls, []); _ } ] -> (
       match caisar_op_of_ls engine ls with
-      | Dataset (DS_csv csv) -> Int (BigInt.of_int (Csv.lines csv))
+      | Dataset (DS_csv csv) -> Value (Int (BigInt.of_int (Csv.lines csv)))
       | Data _ | Classifier _ | Tensor _ | Vector _ | Index _ -> assert false)
     | [ Term t ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t ])
+      term (Term.t_app_infer ls [ t ])
     | _ -> invalid_arg (error_message ls)
   in
   let mapi : _ CRE.builtin =
@@ -172,12 +174,12 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
             let idx = Term.t_int_const (BigInt.of_int idx) in
             (Term.t_func_app_beta_l t2 [ idx; t ], Option.value_exn t.t_ty))
         in
-        Term (term_of_caisar_op ~args engine (Vector n) ty)
-      | Dataset (DS_csv csv) -> Int (BigInt.of_int (Csv.lines csv))
+        Eval (term_of_caisar_op ~args engine (Vector n) ty)
+      | Dataset (DS_csv csv) -> Value (Int (BigInt.of_int (Csv.lines csv)))
       | Data _ | Classifier _ | Tensor _ | Index _ -> assert false)
     | [ Term t1; Term t2 ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t1; t2 ])
+      term (Term.t_app_infer ls [ t1; t2 ])
     | _ -> invalid_arg (error_message ls)
   in
 
@@ -196,11 +198,11 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
       match (caisar_op_of_ls engine ls1, caisar_op_of_ls engine ls2) with
       | Tensor n, Index (I_csv i) ->
         assert (i <= n);
-        Term (List.nth_exn tl1 i)
+        term (List.nth_exn tl1 i)
       | _ -> assert false)
     | [ Term t1; Term t2 ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t1; t2 ])
+      term (Term.t_app_infer ls [ t1; t2 ])
     | _ -> invalid_arg (error_message ls)
   in
   let tminus : _ CRE.builtin =
@@ -237,11 +239,11 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
           List.map2_exn tl1 csts ~f:(fun tl c ->
             (Term.t_app_infer minus [ tl; c ], ty_cst))
         in
-        Term (term_of_caisar_op ~args engine (Tensor n) ty)
+        term (term_of_caisar_op ~args engine (Tensor n) ty)
       | _ -> assert false)
     | [ Term t1; Term t2 ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t1; t2 ])
+      term (Term.t_app_infer ls [ t1; t2 ])
     | _ -> invalid_arg (error_message ls)
   in
 
@@ -261,7 +263,7 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
         let filename = Caml.Filename.concat cwd classifier in
         Classifier filename
       in
-      Term (term_of_caisar_op engine caisar_op ty)
+      term (term_of_caisar_op engine caisar_op ty)
     | _ -> invalid_arg (error_message ls)
   in
   let apply_classifier : _ CRE.builtin =
@@ -272,7 +274,7 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
     match vl with
     | [ Term t1; Term t2 ] ->
       (* Fmt.pr "Terms: %a , %a@." Pretty.print_term t1 Pretty.print_term t2; *)
-      Term (Term.t_app_infer ls [ t1; t2 ])
+      term (Term.t_app_infer ls [ t1; t2 ])
     | _ -> invalid_arg (error_message ls)
   in
 
@@ -293,7 +295,7 @@ let builtin_caisar : caisar_env CRE.built_in_theories list =
         let dataset = DS_csv (Csv.load filename) in
         Dataset dataset
       in
-      Term (term_of_caisar_op engine caisar_op ty)
+      term (term_of_caisar_op engine caisar_op ty)
     | _ -> invalid_arg (error_message ls)
   in
   [
diff --git a/src/reduction_engine.ml b/src/reduction_engine.ml
index 6d30617057aaa65b729bdcec368739681c76894d..e184a8180d7b4eb322717309a459248d9782c04e 100644
--- a/src/reduction_engine.ml
+++ b/src/reduction_engine.ml
@@ -39,7 +39,11 @@ type bounded_quant_result = {
   substitutions : term list;
 }
 
-type 'a builtin = 'a engine -> lsymbol -> value list -> Ty.ty option -> value
+type builtin_value =
+  | Eval of Why3.Term.term
+  | Value of value
+
+type 'a builtin = 'a engine -> lsymbol -> value list -> Ty.ty option -> builtin_value
 
 and 'a bounded_quant =
   'a engine -> vsymbol -> cond:term -> bounded_quant_result option
@@ -113,65 +117,65 @@ let is_one v =
 let eval_int_op op simpl _ ls l ty =
   match l with
   | [ t1; t2 ] -> (
-    try
-      let n1 = big_int_of_value t1 in
-      let n2 = big_int_of_value t2 in
-      Int (op n1 n2)
-    with NotNum | Division_by_zero -> simpl ls t1 t2 ty)
+      try
+        let n1 = big_int_of_value t1 in
+        let n2 = big_int_of_value t2 in
+        Value (Int (op n1 n2))
+      with NotNum | Division_by_zero -> simpl ls t1 t2 ty)
   | _ -> assert false
 
 let simpl_add _ls t1 t2 _ty =
-  if is_zero t1 then t2 else if is_zero t2 then t1 else raise Undetermined
+  Value (if is_zero t1 then t2 else if is_zero t2 then t1 else raise Undetermined)
 
-let simpl_sub _ls t1 t2 _ty = if is_zero t2 then t1 else raise Undetermined
+let simpl_sub _ls t1 t2 _ty = Value (if is_zero t2 then t1 else raise Undetermined)
 
 let simpl_mul _ls t1 t2 _ty =
-  if is_zero t1
-  then t1
-  else if is_zero t2
-  then t2
-  else if is_one t1
-  then t2
-  else if is_one t2
-  then t1
-  else raise Undetermined
+  Value (if is_zero t1
+         then t1
+         else if is_zero t2
+         then t2
+         else if is_one t1
+         then t2
+         else if is_one t2
+         then t1
+         else raise Undetermined)
 
 let simpl_div _ls t1 t2 _ty =
-  if is_zero t2 then raise Undetermined;
-  if is_zero t1 then t1 else if is_one t2 then t1 else raise Undetermined
+  Value (if is_zero t2 then raise Undetermined;
+         if is_zero t1 then t1 else if is_one t2 then t1 else raise Undetermined)
 
 let simpl_mod _ls t1 t2 _ty =
-  if is_zero t2 then raise Undetermined;
-  if is_zero t1
-  then t1
-  else if is_one t2
-  then Int BigInt.zero
-  else raise Undetermined
+  Value (if is_zero t2 then raise Undetermined;
+         if is_zero t1
+         then t1
+         else if is_one t2
+         then Int BigInt.zero
+         else raise Undetermined)
 
 let simpl_minmax _ls v1 v2 _ty =
   match (v1, v2) with
-  | Term t1, Term t2 -> if t_equal t1 t2 then v1 else raise Undetermined
+  | Term t1, Term t2 -> if t_equal t1 t2 then Value v1 else raise Undetermined
   | _ -> raise Undetermined
 
 let eval_int_rel op _ _ls l _ty =
   match l with
   | [ t1; t2 ] -> (
-    try
-      let n1 = big_int_of_value t1 in
-      let n2 = big_int_of_value t2 in
-      Term (to_bool (op n1 n2))
-    with NotNum | Division_by_zero ->
-      raise Undetermined (* t_app_value ls l ty *))
+      try
+        let n1 = big_int_of_value t1 in
+        let n2 = big_int_of_value t2 in
+        Value (Term (to_bool (op n1 n2)))
+      with NotNum | Division_by_zero ->
+        raise Undetermined (* t_app_value ls l ty *))
   | _ -> assert false
 
 let eval_int_uop op _ _ls l _ty =
   match l with
   | [ t1 ] -> (
-    try
-      let n1 = big_int_of_value t1 in
-      Int (op n1)
-    with NotNum | Division_by_zero ->
-      raise Undetermined (* t_app_value ls l ty *))
+      try
+        let n1 = big_int_of_value t1 in
+        Value (Int (op n1))
+      with NotNum | Division_by_zero ->
+        raise Undetermined (* t_app_value ls l ty *))
   | _ -> assert false
 
 let real_align ~pow2 ~pow5 r =
@@ -184,43 +188,43 @@ let real_align ~pow2 ~pow5 r =
 let eval_real_op op simpl _ ls l ty =
   match l with
   | [ t1; t2 ] -> (
-    try
-      let n1 = real_of_value t1 in
-      let n2 = real_of_value t2 in
-      Real (op n1 n2)
-    with NotNum -> simpl ls t1 t2 ty)
+      try
+        let n1 = real_of_value t1 in
+        let n2 = real_of_value t2 in
+        Value (Real (op n1 n2))
+      with NotNum -> simpl ls t1 t2 ty)
   | _ -> assert false
 
 let eval_real_uop op _ _ls l _ty =
   match l with
   | [ t1 ] -> (
-    try
-      let n1 = real_of_value t1 in
-      Real (op n1)
-    with NotNum -> raise Undetermined)
+      try
+        let n1 = real_of_value t1 in
+        Value (Real (op n1))
+      with NotNum -> raise Undetermined)
   | _ -> assert false
 
 let eval_real_rel op _ _ls l _ty =
   let open Number in
   match l with
   | [ t1; t2 ] -> (
-    try
-      let n1 = real_of_value t1 in
-      let n2 = real_of_value t2 in
-      let s1, s2 =
-        let { rv_sig = s1; rv_pow2 = p21; rv_pow5 = p51 } = n1 in
-        let { rv_sig = s2; rv_pow2 = p22; rv_pow5 = p52 } = n2 in
-        if BigInt.sign s1 <> BigInt.sign s2
-        then (s1, s2)
-        else
-          let pow2 = BigInt.min p21 p22 in
-          let pow5 = BigInt.min p51 p52 in
-          let s1 = real_align ~pow2 ~pow5 n1 in
-          let s2 = real_align ~pow2 ~pow5 n2 in
-          (s1, s2)
-      in
-      Term (to_bool (op s1 s2))
-    with NotNum -> raise Undetermined (* t_app_value ls l ty *))
+      try
+        let n1 = real_of_value t1 in
+        let n2 = real_of_value t2 in
+        let s1, s2 =
+          let { rv_sig = s1; rv_pow2 = p21; rv_pow5 = p51 } = n1 in
+          let { rv_sig = s2; rv_pow2 = p22; rv_pow5 = p52 } = n2 in
+          if BigInt.sign s1 <> BigInt.sign s2
+          then (s1, s2)
+          else
+            let pow2 = BigInt.min p21 p22 in
+            let pow5 = BigInt.min p51 p52 in
+            let s1 = real_align ~pow2 ~pow5 n1 in
+            let s2 = real_align ~pow2 ~pow5 n2 in
+            (s1, s2)
+        in
+        Value (Term (to_bool (op s1 s2)))
+      with NotNum -> raise Undetermined (* t_app_value ls l ty *))
   | _ -> assert false
 
 let real_0 = Number.real_value BigInt.zero
@@ -274,25 +278,25 @@ let real_mul r1 r2 =
   real_value ~pow2 ~pow5 s
 
 let simpl_real_add _ls t1 t2 _ty =
-  if is_real t1 real_0
-  then t2
-  else if is_real t2 real_0
-  then t1
-  else raise Undetermined
+  Value (if is_real t1 real_0
+         then t2
+         else if is_real t2 real_0
+         then t1
+         else raise Undetermined)
 
 let simpl_real_sub _ls t1 t2 _ty =
-  if is_real t2 real_0 then t1 else raise Undetermined
+  if is_real t2 real_0 then Value t1 else raise Undetermined
 
 let simpl_real_mul _ls t1 t2 _ty =
-  if is_real t1 real_0
-  then t1
-  else if is_real t2 real_0
-  then t2
-  else if is_real t1 real_1
-  then t2
-  else if is_real t2 real_1
-  then t1
-  else raise Undetermined
+  Value (if is_real t1 real_0
+         then t1
+         else if is_real t2 real_0
+         then t2
+         else if is_real t1 real_1
+         then t2
+         else if is_real t2 real_1
+         then t1
+         else raise Undetermined)
 
 let real_pow r1 r2 =
   let open Number in
@@ -318,15 +322,15 @@ let real_pow r1 r2 =
     Number.real_value ~pow2 ~pow5 s
 
 let simpl_real_pow _ls t1 _t2 _ty =
-  if is_real t1 real_1 then t1 else raise Undetermined
+  if is_real t1 real_1 then Value t1 else raise Undetermined
 
 let real_from_int _ _ls l _ty =
   match l with
   | [ t ] -> (
-    try
-      let n = big_int_of_value t in
-      Real (Number.real_value n)
-    with NotNum -> raise Undetermined)
+      try
+        let n = big_int_of_value t in
+        Value (Real (Number.real_value n))
+      with NotNum -> raise Undetermined)
   | _ -> assert false
 
 type 'a built_in_theories =
@@ -336,7 +340,7 @@ type 'a built_in_theories =
   * (string * lsymbol ref option * 'a builtin) list
 
 let built_in_theories : unit -> 'a built_in_theories list =
- fun () ->
+  fun () ->
   [
     (* ["bool"],"Bool", [], [ "True", None, eval_true ; "False", None,
        eval_false ; ] ; *)
@@ -401,14 +405,14 @@ let add_builtin_th env ((l, n, t, d) : 'a built_in_theories) =
   let th = Env.read_theory env.env l n in
   List.iter
     (fun (id, r) ->
-      let ts = Theory.ns_find_ts th.Theory.th_export [ id ] in
-      r ts)
+       let ts = Theory.ns_find_ts th.Theory.th_export [ id ] in
+       r ts)
     t;
   List.iter
     (fun (id, r, f) ->
-      let ls = Theory.ns_find_ls th.Theory.th_export [ id ] in
-      Hls.add env.builtins ls f;
-      match r with None -> () | Some r -> r := ls)
+       let ls = Theory.ns_find_ls th.Theory.th_export [ id ] in
+       Hls.add env.builtins ls f;
+       match r with None -> () | Some r -> r := ls)
     d
 
 let get_builtins env built_in_user =
@@ -461,7 +465,7 @@ type cont =
 type config = {
   value_stack : value list;
   cont_stack : (cont * term) list;
-    (* second term is the original term, for attribute and loc copy *)
+  (* second term is the original term, for attribute and loc copy *)
 }
 
 (* This global variable is used to approximate a count of the elementary
@@ -476,22 +480,22 @@ let rec pattern_renaming (bound_vars, mt) p1 p2 =
   match (p1.pat_node, p2.pat_node) with
   | Pwild, Pwild -> (bound_vars, mt)
   | Pvar v1, Pvar v2 -> (
-    try
-      let mt = Ty.ty_match mt v1.vs_ty v2.vs_ty in
-      let bound_vars = Mvs.add v2 v1 bound_vars in
-      (bound_vars, mt)
-    with Ty.TypeMismatch _ -> raise (NoMatchpat (Some (p1, p2))))
+      try
+        let mt = Ty.ty_match mt v1.vs_ty v2.vs_ty in
+        let bound_vars = Mvs.add v2 v1 bound_vars in
+        (bound_vars, mt)
+      with Ty.TypeMismatch _ -> raise (NoMatchpat (Some (p1, p2))))
   | Papp (ls1, tl1), Papp (ls2, tl2) when ls_equal ls1 ls2 ->
     List.fold_left2 pattern_renaming (bound_vars, mt) tl1 tl2
   | Por (p1a, p1b), Por (p2a, p2b) ->
     let bound_vars, mt = pattern_renaming (bound_vars, mt) p1a p2a in
     pattern_renaming (bound_vars, mt) p1b p2b
   | Pas (p1, v1), Pas (p2, v2) -> (
-    try
-      let mt = Ty.ty_match mt v1.vs_ty v2.vs_ty in
-      let bound_vars = Mvs.add v2 v1 bound_vars in
-      pattern_renaming (bound_vars, mt) p1 p2
-    with Ty.TypeMismatch _ -> raise (NoMatchpat (Some (p1, p2))))
+      try
+        let mt = Ty.ty_match mt v1.vs_ty v2.vs_ty in
+        let bound_vars = Mvs.add v2 v1 bound_vars in
+        pattern_renaming (bound_vars, mt) p1 p2
+      with Ty.TypeMismatch _ -> raise (NoMatchpat (Some (p1, p2))))
   | _ -> raise (NoMatchpat (Some (p1, p2)))
 
 let first_order_matching (vars : Svs.t) (largs : term list) (args : term list) :
@@ -500,121 +504,121 @@ let first_order_matching (vars : Svs.t) (largs : term list) (args : term list) :
     match (largs, args) with
     | [], [] -> sigma
     | t1 :: r1, t2 :: r2 -> (
-      (* Format.eprintf "matching terms %a and %a...@." Pretty.print_term t1
-         Pretty.print_term t2; *)
-      match t1.t_node with
-      | Tvar vs when Svs.mem vs vars -> (
-        try
-          let t = Mvs.find vs mv in
-          if t_equal t t2
-          then loop bound_vars sigma r1 r2
-          else raise (NoMatch (Some (t1, t2, Some t)))
-        with Not_found -> (
-          try
-            let ts = Ty.ty_match mt vs.vs_ty (t_type t2) in
-            let fv2 = t_vars t2 in
-            if Mvs.is_empty (Mvs.set_inter bound_vars fv2)
-            then loop bound_vars (ts, Mvs.add vs t2 mv) r1 r2
-            else raise (NoMatch (Some (t1, t2, None)))
-          with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))))
-      | Tapp (ls1, args1) -> (
-        match t2.t_node with
-        | Tapp (ls2, args2) when ls_equal ls1 ls2 -> (
-          let mt, mv =
-            loop bound_vars sigma (List.rev_append args1 r1)
-              (List.rev_append args2 r2)
-          in
-          try (Ty.oty_match mt t1.t_ty t2.t_ty, mv)
-          with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None))))
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tquant (q1, bv1) -> (
-        match t2.t_node with
-        | Tquant (q2, bv2) when q1 = q2 ->
-          let vl1, _tl1, term1 = t_open_quant bv1 in
-          let vl2, _tl2, term2 = t_open_quant bv2 in
-          let bound_vars, term1, mt =
+        (* Format.eprintf "matching terms %a and %a...@." Pretty.print_term t1
+           Pretty.print_term t2; *)
+        match t1.t_node with
+        | Tvar vs when Svs.mem vs vars -> (
             try
-              List.fold_left2
-                (fun (bound_vars, term1, mt) e1 e2 ->
-                  let mt = Ty.ty_match mt e1.vs_ty e2.vs_ty in
-                  let bound_vars = Mvs.add e2 e1 bound_vars in
-                  (bound_vars, term1, mt))
-                (bound_vars, term1, mt) vl1 vl2
-            with Invalid_argument _ | Ty.TypeMismatch _ ->
-              raise (NoMatch (Some (t1, t2, None)))
-          in
-          loop bound_vars (mt, mv) (term1 :: r1) (term2 :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tbinop (b1, t1_l, t1_r) -> (
-        match t2.t_node with
-        | Tbinop (b2, t2_l, t2_r) when b1 = b2 ->
-          loop bound_vars (mt, mv) (t1_l :: t1_r :: r1) (t2_l :: t2_r :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tif (t11, t12, t13) -> (
-        match t2.t_node with
-        | Tif (t21, t22, t23) ->
-          loop bound_vars (mt, mv) (t11 :: t12 :: t13 :: r1)
-            (t21 :: t22 :: t23 :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tlet (td1, tb1) -> (
-        match t2.t_node with
-        | Tlet (td2, tb2) ->
-          let v1, tl1 = t_open_bound tb1 in
-          let v2, tl2 = t_open_bound tb2 in
-          let mt =
-            try Ty.ty_match mt v1.vs_ty v2.vs_ty
-            with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))
-          in
-          let bound_vars = Mvs.add v2 v1 bound_vars in
-          loop bound_vars (mt, mv) (td1 :: tl1 :: r1) (td2 :: tl2 :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tcase (ts1, tbl1) -> (
-        match t2.t_node with
-        | Tcase (ts2, tbl2) -> (
-          try
-            let bound_vars, mt, l1, l2 =
-              List.fold_left2
-                (fun (bound_vars, mt, l1, l2) tb1 tb2 ->
-                  let p1, tb1 = t_open_branch tb1 in
-                  let p2, tb2 = t_open_branch tb2 in
-                  let bound_vars, mt =
-                    pattern_renaming (bound_vars, mt) p1 p2
+              let t = Mvs.find vs mv in
+              if t_equal t t2
+              then loop bound_vars sigma r1 r2
+              else raise (NoMatch (Some (t1, t2, Some t)))
+            with Not_found -> (
+                try
+                  let ts = Ty.ty_match mt vs.vs_ty (t_type t2) in
+                  let fv2 = t_vars t2 in
+                  if Mvs.is_empty (Mvs.set_inter bound_vars fv2)
+                  then loop bound_vars (ts, Mvs.add vs t2 mv) r1 r2
+                  else raise (NoMatch (Some (t1, t2, None)))
+                with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))))
+        | Tapp (ls1, args1) -> (
+            match t2.t_node with
+            | Tapp (ls2, args2) when ls_equal ls1 ls2 -> (
+                let mt, mv =
+                  loop bound_vars sigma (List.rev_append args1 r1)
+                    (List.rev_append args2 r2)
+                in
+                try (Ty.oty_match mt t1.t_ty t2.t_ty, mv)
+                with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None))))
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tquant (q1, bv1) -> (
+            match t2.t_node with
+            | Tquant (q2, bv2) when q1 = q2 ->
+              let vl1, _tl1, term1 = t_open_quant bv1 in
+              let vl2, _tl2, term2 = t_open_quant bv2 in
+              let bound_vars, term1, mt =
+                try
+                  List.fold_left2
+                    (fun (bound_vars, term1, mt) e1 e2 ->
+                       let mt = Ty.ty_match mt e1.vs_ty e2.vs_ty in
+                       let bound_vars = Mvs.add e2 e1 bound_vars in
+                       (bound_vars, term1, mt))
+                    (bound_vars, term1, mt) vl1 vl2
+                with Invalid_argument _ | Ty.TypeMismatch _ ->
+                  raise (NoMatch (Some (t1, t2, None)))
+              in
+              loop bound_vars (mt, mv) (term1 :: r1) (term2 :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tbinop (b1, t1_l, t1_r) -> (
+            match t2.t_node with
+            | Tbinop (b2, t2_l, t2_r) when b1 = b2 ->
+              loop bound_vars (mt, mv) (t1_l :: t1_r :: r1) (t2_l :: t2_r :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tif (t11, t12, t13) -> (
+            match t2.t_node with
+            | Tif (t21, t22, t23) ->
+              loop bound_vars (mt, mv) (t11 :: t12 :: t13 :: r1)
+                (t21 :: t22 :: t23 :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tlet (td1, tb1) -> (
+            match t2.t_node with
+            | Tlet (td2, tb2) ->
+              let v1, tl1 = t_open_bound tb1 in
+              let v2, tl2 = t_open_bound tb2 in
+              let mt =
+                try Ty.ty_match mt v1.vs_ty v2.vs_ty
+                with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))
+              in
+              let bound_vars = Mvs.add v2 v1 bound_vars in
+              loop bound_vars (mt, mv) (td1 :: tl1 :: r1) (td2 :: tl2 :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tcase (ts1, tbl1) -> (
+            match t2.t_node with
+            | Tcase (ts2, tbl2) -> (
+                try
+                  let bound_vars, mt, l1, l2 =
+                    List.fold_left2
+                      (fun (bound_vars, mt, l1, l2) tb1 tb2 ->
+                         let p1, tb1 = t_open_branch tb1 in
+                         let p2, tb2 = t_open_branch tb2 in
+                         let bound_vars, mt =
+                           pattern_renaming (bound_vars, mt) p1 p2
+                         in
+                         (bound_vars, mt, tb1 :: l1, tb2 :: l2))
+                      (bound_vars, mt, ts1 :: r1, ts2 :: r2)
+                      tbl1 tbl2
                   in
-                  (bound_vars, mt, tb1 :: l1, tb2 :: l2))
-                (bound_vars, mt, ts1 :: r1, ts2 :: r2)
-                tbl1 tbl2
-            in
-            loop bound_vars (mt, mv) l1 l2
-          with Invalid_argument _ -> raise (NoMatch (Some (t1, t2, None))))
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Teps tb1 -> (
-        match t2.t_node with
-        | Teps tb2 ->
-          let v1, td1 = t_open_bound tb1 in
-          let v2, td2 = t_open_bound tb2 in
-          let mt =
-            try Ty.ty_match mt v1.vs_ty v2.vs_ty
-            with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))
-          in
-          let bound_vars = Mvs.add v2 v1 bound_vars in
-          loop bound_vars (mt, mv) (td1 :: r1) (td2 :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tnot t1 -> (
-        match t2.t_node with
-        | Tnot t2 -> loop bound_vars sigma (t1 :: r1) (t2 :: r2)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | Tvar v1 -> (
-        match t2.t_node with
-        | Tvar v2 -> (
-          try
-            if vs_equal v1 (Mvs.find v2 bound_vars)
-            then loop bound_vars sigma r1 r2
-            else raise (NoMatch (Some (t1, t2, None)))
-          with Not_found -> assert false)
-        | _ -> raise (NoMatch (Some (t1, t2, None))))
-      | (Tconst _ | Ttrue | Tfalse) when t_equal t1 t2 ->
-        loop bound_vars sigma r1 r2
-      | Tconst _ | Ttrue | Tfalse -> raise (NoMatch (Some (t1, t2, None))))
+                  loop bound_vars (mt, mv) l1 l2
+                with Invalid_argument _ -> raise (NoMatch (Some (t1, t2, None))))
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Teps tb1 -> (
+            match t2.t_node with
+            | Teps tb2 ->
+              let v1, td1 = t_open_bound tb1 in
+              let v2, td2 = t_open_bound tb2 in
+              let mt =
+                try Ty.ty_match mt v1.vs_ty v2.vs_ty
+                with Ty.TypeMismatch _ -> raise (NoMatch (Some (t1, t2, None)))
+              in
+              let bound_vars = Mvs.add v2 v1 bound_vars in
+              loop bound_vars (mt, mv) (td1 :: r1) (td2 :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tnot t1 -> (
+            match t2.t_node with
+            | Tnot t2 -> loop bound_vars sigma (t1 :: r1) (t2 :: r2)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | Tvar v1 -> (
+            match t2.t_node with
+            | Tvar v2 -> (
+                try
+                  if vs_equal v1 (Mvs.find v2 bound_vars)
+                  then loop bound_vars sigma r1 r2
+                  else raise (NoMatch (Some (t1, t2, None)))
+                with Not_found -> assert false)
+            | _ -> raise (NoMatch (Some (t1, t2, None))))
+        | (Tconst _ | Ttrue | Tfalse) when t_equal t1 t2 ->
+          loop bound_vars sigma r1 r2
+        | Tconst _ | Ttrue | Tfalse -> raise (NoMatch (Some (t1, t2, None))))
     | _ -> raise (NoMatch None)
   in
   loop Mvs.empty (Ty.Mtv.empty, Mvs.empty) largs args
@@ -628,10 +632,10 @@ let one_step_reduce engine ls args =
       match rules with
       | [] -> raise Irreducible
       | (vars, largs, rhs) :: rem -> (
-        try
-          let sigma = first_order_matching vars largs args in
-          (sigma, rhs)
-        with NoMatch _ -> loop rem)
+          try
+            let sigma = first_order_matching vars largs args in
+            (sigma, rhs)
+          with NoMatch _ -> loop rem)
     in
     loop rules
   with Not_found -> raise Irreducible
@@ -641,17 +645,17 @@ let rec matching ((mt, mv) as sigma) t p =
   | Pwild -> sigma
   | Pvar v -> (mt, Mvs.add v t mv)
   | Por (p1, p2) -> (
-    try matching sigma t p1 with NoMatch _ -> matching sigma t p2)
+      try matching sigma t p1 with NoMatch _ -> matching sigma t p2)
   | Pas (p, v) -> matching (mt, Mvs.add v t mv) t p
   | Papp (ls1, pl) -> (
-    match t.t_node with
-    | Tapp (ls2, tl) ->
-      if ls_equal ls1 ls2
-      then List.fold_left2 matching sigma tl pl
-      else if ls2.ls_constr > 0
-      then raise (NoMatch None)
-      else raise Undetermined
-    | _ -> raise Undetermined)
+      match t.t_node with
+      | Tapp (ls2, tl) ->
+        if ls_equal ls1 ls2
+        then List.fold_left2 matching sigma tl pl
+        else if ls2.ls_constr > 0
+        then raise (NoMatch None)
+        else raise Undetermined
+      | _ -> raise Undetermined)
 
 let rec extract_first n acc l =
   if n = 0
@@ -668,26 +672,26 @@ let bounds ls_lt t1 t2 =
   let lt order = function
     (* match [i < vs] or [vs < i], return [i, vs] *)
     | { t_node = Tapp (ls, [ t1; t2 ]) } when ls_equal ls ls_lt -> (
-      assert (Ty.(ty_equal (t_type t1) ty_int && ty_equal (t_type t2) ty_int));
-      match order t1 t2 with
-      | ( { t_node = Tconst (Constant.ConstInt { Number.il_int = i }) },
-          { t_node = Tvar vs } ) ->
-        (i, vs)
-      | _ -> raise Exit)
+        assert (Ty.(ty_equal (t_type t1) ty_int && ty_equal (t_type t2) ty_int));
+        match order t1 t2 with
+        | ( { t_node = Tconst (Constant.ConstInt { Number.il_int = i }) },
+            { t_node = Tvar vs } ) ->
+          (i, vs)
+        | _ -> raise Exit)
     | _ -> raise Exit
   in
   let eq order = function
     (* match [i = vs] or [vs = i], return [i, vs] *)
     | { t_node = Tapp (ls, [ t1; t2 ]) }
       when ls_equal ls Term.ps_equ
-           && Ty.ty_equal (t_type t1) Ty.ty_int
-           && Ty.ty_equal (t_type t2) Ty.ty_int -> (
-      match order t1 t2 with
-      | ( { t_node = Tconst (Constant.ConstInt { Number.il_int = i }) },
-          { t_node = Tvar vs } )
-        when Ty.ty_equal vs.vs_ty Ty.ty_int ->
-        (i, vs)
-      | _ -> raise Exit)
+        && Ty.ty_equal (t_type t1) Ty.ty_int
+        && Ty.ty_equal (t_type t2) Ty.ty_int -> (
+        match order t1 t2 with
+        | ( { t_node = Tconst (Constant.ConstInt { Number.il_int = i }) },
+            { t_node = Tvar vs } )
+          when Ty.ty_equal vs.vs_ty Ty.ty_int ->
+          (i, vs)
+        | _ -> raise Exit)
     | _ -> raise Exit
   in
   let le order = function
@@ -723,9 +727,9 @@ let bounds ls_lt t1 t2 =
 
       TODO:
 
-      - expand to bounded quantifications on range values
-      - compatiblity with reverse direction (forall i. b > i > a -> t)
-      - detect SPARK-style [forall i. if a < i /\ i < b then t else true] *)
+    - expand to bounded quantifications on range values
+    - compatiblity with reverse direction (forall i. b > i > a -> t)
+    - detect SPARK-style [forall i. if a < i /\ i < b then t else true] *)
 let reduce_bounded_quant engine ls_lt limit t sigma st rem =
   match (st, rem) with
   (* st = a < vs < b :: _; rem = -> :: forall vs :: _ *)
@@ -734,7 +738,7 @@ let reduce_bounded_quant engine ls_lt limit t sigma st rem =
         :: (Kquant ((Tforall as quant), [ vs ], _), _)
         :: rem
       | (Kbinop Tand, _) :: (Kquant ((Texists as quant), [ vs ], _), _) :: rem
-        ) )
+      ) )
     when Ty.ty_equal vs.vs_ty Ty.ty_int ->
     let t_empty, binop =
       match quant with Tforall -> (t_true, Tand) | Texists -> (t_false, Tor)
@@ -763,75 +767,75 @@ let reduce_bounded_quant engine ls_lt limit t sigma st rem =
       (Kbinop Timplies, _)
       :: (Kquant ((Tforall as quant), [ vs ], _), t_orig)
       :: rem ) -> (
-    match engine.bounded_quant engine vs ~cond with
-    | None -> raise Exit
-    | Some res -> (
-      let t_empty, binop =
-        match quant with Tforall -> (t_true, Tand) | Texists -> (t_false, Tor)
-      in
-      let rem =
-        match res.new_quant with
-        | [] -> rem
-        | _ -> (Kquant (quant, res.new_quant, []), t_orig) :: rem
-      in
-      let rec loop rem = function
-        | [] -> rem
-        | t_i :: l ->
+      match engine.bounded_quant engine vs ~cond with
+      | None -> raise Exit
+      | Some res -> (
+          let t_empty, binop =
+            match quant with Tforall -> (t_true, Tand) | Texists -> (t_false, Tor)
+          in
           let rem =
-            match l with
+            match res.new_quant with
             | [] -> rem
-            | _ ->
-              (* conjunction *)
-              (Kbinop binop, t_true) :: rem
+            | _ -> (Kquant (quant, res.new_quant, []), t_orig) :: rem
           in
-          let rem = (Keval (t, Mvs.add vs t_i sigma), t_true) :: rem in
-          loop rem l
-      in
-      match res.substitutions with
-      | [] -> { value_stack = Term t_empty :: st; cont_stack = rem }
-      | _ -> { value_stack = st; cont_stack = loop rem res.substitutions }))
+          let rec loop rem = function
+            | [] -> rem
+            | t_i :: l ->
+              let rem =
+                match l with
+                | [] -> rem
+                | _ ->
+                  (* conjunction *)
+                  (Kbinop binop, t_true) :: rem
+              in
+              let rem = (Keval (t, Mvs.add vs t_i sigma), t_true) :: rem in
+              loop rem l
+          in
+          match res.substitutions with
+          | [] -> { value_stack = Term t_empty :: st; cont_stack = rem }
+          | _ -> { value_stack = st; cont_stack = loop rem res.substitutions }))
   | _ -> raise Exit
 
 let rec reduce engine c =
   match (c.value_stack, c.cont_stack) with
   | _, [] -> assert false
   | st, (Keval (t, sigma), orig) :: rem -> (
-    let limit = engine.params.compute_max_quantifier_domain in
-    try reduce_bounded_quant engine engine.ls_lt limit t sigma st rem
-    with Exit -> reduce_eval engine st t ~orig sigma rem)
+      let limit = engine.params.compute_max_quantifier_domain in
+      try reduce_bounded_quant engine engine.ls_lt limit t sigma st rem
+      with Exit -> reduce_eval engine st t ~orig sigma rem)
   | [], (Kif _, _) :: _ -> assert false
   | v :: st, (Kif (t2, t3, sigma), orig) :: rem -> (
-    match v with
-    | Term { t_node = Ttrue } ->
-      incr rec_step_limit;
-      {
-        value_stack = st;
-        cont_stack = (Keval (t2, sigma), t_attr_copy orig t2) :: rem;
-      }
-    | Term { t_node = Tfalse } ->
-      incr rec_step_limit;
-      {
-        value_stack = st;
-        cont_stack = (Keval (t3, sigma), t_attr_copy orig t3) :: rem;
-      }
-    | Term t1 -> (
-      match (t1.t_node, t2.t_node, t3.t_node) with
-      | ( Tapp (ls, [ b0; { t_node = Tapp (ls1, _) } ]),
-          Tapp (ls2, _),
-          Tapp (ls3, _) )
-        when ls_equal ls ps_equ && ls_equal ls1 fs_bool_true
-             && ls_equal ls2 fs_bool_true && ls_equal ls3 fs_bool_false ->
+      match v with
+      | Term { t_node = Ttrue } ->
         incr rec_step_limit;
-        { value_stack = Term (t_attr_copy orig b0) :: st; cont_stack = rem }
-      | _ ->
         {
-          value_stack =
-            Term
-              (t_attr_copy orig (t_if t1 (t_subst sigma t2) (t_subst sigma t3)))
-            :: st;
-          cont_stack = rem;
-        })
-    | Int _ | Real _ -> assert false (* would be ill-typed *))
+          value_stack = st;
+          cont_stack = (Keval (t2, sigma), t_attr_copy orig t2) :: rem;
+        }
+      | Term { t_node = Tfalse } ->
+        incr rec_step_limit;
+        {
+          value_stack = st;
+          cont_stack = (Keval (t3, sigma), t_attr_copy orig t3) :: rem;
+        }
+      | Term t1 -> (
+          match (t1.t_node, t2.t_node, t3.t_node) with
+          | ( Tapp (ls, [ b0; { t_node = Tapp (ls1, _) } ]),
+              Tapp (ls2, _),
+              Tapp (ls3, _) )
+            when ls_equal ls ps_equ && ls_equal ls1 fs_bool_true
+                 && ls_equal ls2 fs_bool_true && ls_equal ls3 fs_bool_false ->
+            incr rec_step_limit;
+            { value_stack = Term (t_attr_copy orig b0) :: st; cont_stack = rem }
+          | _ ->
+            {
+              value_stack =
+                Term
+                  (t_attr_copy orig (t_if t1 (t_subst sigma t2) (t_subst sigma t3)))
+                :: st;
+              cont_stack = rem;
+            })
+      | Int _ | Real _ -> assert false (* would be ill-typed *))
   | [], (Klet _, _) :: _ -> assert false
   | t1 :: st, (Klet (v, t2, sigma), orig) :: rem ->
     incr rec_step_limit;
@@ -882,26 +886,26 @@ and reduce_match st u ~orig tbl sigma cont =
     match tbl with
     | [] -> assert false (* pattern matching not exhaustive *)
     | b :: rem -> (
-      let p, t = t_open_branch b in
-      try
-        let mt', mv' = matching (Ty.Mtv.empty, sigma) u p in
-        (* Format.eprintf "Pattern-matching succeeded:@\nmt' = @["; Ty.Mtv.iter
-           (fun tv ty -> Format.eprintf "%a -> %a," Pretty.print_tv tv
-           Pretty.print_ty ty) mt'; Format.eprintf "@]@\n"; Format.eprintf "mv'
-           = @["; Mvs.iter (fun v t -> Format.eprintf "%a -> %a,"
-           Pretty.print_vs v Pretty.print_term t) mv'; Format.eprintf "@]@.";
-           Format.eprintf "branch before inst: %a@." Pretty.print_term t; *)
-        let mv'', t = t_subst_types mt' mv' t in
-        (* Format.eprintf "branch after types inst: %a@." Pretty.print_term t;
-           Format.eprintf "mv'' = @["; Mvs.iter (fun v t -> Format.eprintf "%a
-           -> %a," Pretty.print_vs v Pretty.print_term t) mv''; Format.eprintf
-           "@]@."; *)
-        incr rec_step_limit;
-        {
-          value_stack = st;
-          cont_stack = (Keval (t, mv''), t_attr_copy orig t) :: cont;
-        }
-      with NoMatch _ -> iter rem)
+        let p, t = t_open_branch b in
+        try
+          let mt', mv' = matching (Ty.Mtv.empty, sigma) u p in
+          (* Format.eprintf "Pattern-matching succeeded:@\nmt' = @["; Ty.Mtv.iter
+             (fun tv ty -> Format.eprintf "%a -> %a," Pretty.print_tv tv
+             Pretty.print_ty ty) mt'; Format.eprintf "@]@\n"; Format.eprintf "mv'
+             = @["; Mvs.iter (fun v t -> Format.eprintf "%a -> %a,"
+             Pretty.print_vs v Pretty.print_term t) mv'; Format.eprintf "@]@.";
+             Format.eprintf "branch before inst: %a@." Pretty.print_term t; *)
+          let mv'', t = t_subst_types mt' mv' t in
+          (* Format.eprintf "branch after types inst: %a@." Pretty.print_term t;
+             Format.eprintf "mv'' = @["; Mvs.iter (fun v t -> Format.eprintf "%a
+             -> %a," Pretty.print_vs v Pretty.print_term t) mv''; Format.eprintf
+             "@]@."; *)
+          incr rec_step_limit;
+          {
+            value_stack = st;
+            cont_stack = (Keval (t, mv''), t_attr_copy orig t) :: cont;
+          }
+        with NoMatch _ -> iter rem)
   in
   try iter tbl
   with Undetermined ->
@@ -920,43 +924,43 @@ and reduce_eval eng st t ~orig sigma rem =
   let orig = t_attr_copy orig t in
   match t.t_node with
   | Tvar v -> (
-    match Ident.Mid.find v.vs_name eng.known_map with
-    | Decl.{ d_node = Dlogic dl } ->
-      incr rec_step_limit;
-      let _, ls_defn =
-        List.find
-          (fun (ls, _) ->
-            String.equal ls.ls_name.Ident.id_string v.vs_name.Ident.id_string)
-          dl
-      in
-      let vs, t = Decl.open_ls_defn ls_defn in
-      let aux vs t =
-        (* Create ε fc. λ vs. fc @ vs = t to make the value from known_map
-           compatible to reduce_func_app *)
-        let ty = Opt.get t.t_ty in
-        let app_ty = Ty.ty_func vs.vs_ty ty in
-        let fc = create_vsymbol (Ident.id_fresh "fc") app_ty in
-        t_eps
-          (t_close_bound fc
-             (t_quant Tforall
-                (t_close_quant [ vs ] []
-                   (t_app ps_equ
-                      [ t_app fs_func_app [ t_var fc; t_var vs ] (Some ty); t ]
-                      None))))
-      in
-      let t = List.fold_right aux vs t in
-      { value_stack = Term t :: st; cont_stack = rem }
-    | _ -> assert false
-    | exception Not_found -> (
-      match Mvs.find v sigma with
-      | t ->
+      match Ident.Mid.find v.vs_name eng.known_map with
+      | Decl.{ d_node = Dlogic dl } ->
         incr rec_step_limit;
-        { value_stack = Term (t_attr_copy orig t) :: st; cont_stack = rem }
-      | exception Not_found ->
-        (* this may happen, e.g when computing below a quantified formula *)
-        (* Format.eprintf "Tvar not found: %a@." Pretty.print_vs v; assert
-           false *)
-        { value_stack = Term orig :: st; cont_stack = rem }))
+        let _, ls_defn =
+          List.find
+            (fun (ls, _) ->
+               String.equal ls.ls_name.Ident.id_string v.vs_name.Ident.id_string)
+            dl
+        in
+        let vs, t = Decl.open_ls_defn ls_defn in
+        let aux vs t =
+          (* Create ε fc. λ vs. fc @ vs = t to make the value from known_map
+             compatible to reduce_func_app *)
+          let ty = Opt.get t.t_ty in
+          let app_ty = Ty.ty_func vs.vs_ty ty in
+          let fc = create_vsymbol (Ident.id_fresh "fc") app_ty in
+          t_eps
+            (t_close_bound fc
+               (t_quant Tforall
+                  (t_close_quant [ vs ] []
+                     (t_app ps_equ
+                        [ t_app fs_func_app [ t_var fc; t_var vs ] (Some ty); t ]
+                        None))))
+        in
+        let t = List.fold_right aux vs t in
+        { value_stack = Term t :: st; cont_stack = rem }
+      | _ -> assert false
+      | exception Not_found -> (
+          match Mvs.find v sigma with
+          | t ->
+            incr rec_step_limit;
+            { value_stack = Term (t_attr_copy orig t) :: st; cont_stack = rem }
+          | exception Not_found ->
+            (* this may happen, e.g when computing below a quantified formula *)
+            (* Format.eprintf "Tvar not found: %a@." Pretty.print_vs v; assert
+               false *)
+            { value_stack = Term orig :: st; cont_stack = rem }))
   | Tif (t1, t2, t3) ->
     {
       value_stack = st;
@@ -1012,15 +1016,15 @@ and reduce_app engine st ls ~orig ty rem_cont =
   then
     match st with
     | t2 :: t1 :: rem_st -> (
-      try reduce_equ ~orig rem_st t1 t2 rem_cont
-      with Undetermined -> reduce_app_no_equ engine st ls ~orig ty rem_cont)
+        try reduce_equ ~orig rem_st t1 t2 rem_cont
+        with Undetermined -> reduce_app_no_equ engine st ls ~orig ty rem_cont)
     | _ -> assert false
   else if ls_equal ls fs_func_app
   then
     match st with
     | t2 :: t1 :: rem_st -> (
-      try reduce_func_app ~orig ty rem_st t1 t2 rem_cont
-      with Undetermined -> reduce_app_no_equ engine st ls ~orig ty rem_cont)
+        try reduce_func_app ~orig ty rem_st t1 t2 rem_cont
+        with Undetermined -> reduce_app_no_equ engine st ls ~orig ty rem_cont)
     | _ -> assert false
   else reduce_app_no_equ engine st ls ~orig ty rem_cont
 
@@ -1029,80 +1033,80 @@ and reduce_func_app ~orig _ty rem_st t1 t2 rem_cont =
      body) that is equivalent to \x.body *)
   match t1 with
   | Term { t_node = Teps tb } -> (
-    let fc, t = Term.t_open_bound tb in
-    match t.t_node with
-    | Tquant (Tforall, tq) -> (
-      let vl, trig, t = t_open_quant tq in
-      let process lhs body equ elim =
-        let rvl = List.rev vl in
-        let rec remove_var lhs rvh rvt =
-          match lhs.t_node with
-          | Tapp (ls2, [ lhs1; ({ t_node = Tvar v1 } as arg) ])
-            when ls_equal ls2 fs_func_app && vs_equal v1 rvh -> (
-            match (rvt, lhs1) with
-            | rvh :: rvt, _ ->
-              let lhs1, fc2 = remove_var lhs1 rvh rvt in
-              let lhs2 = t_app ls2 [ lhs1; arg ] lhs.t_ty in
-              (t_attr_copy lhs lhs2, fc2)
-            | [], { t_node = Tvar fc1 } when vs_equal fc1 fc ->
-              let fcn = fc.vs_name in
-              let fc2 = Ident.id_derive fcn.Ident.id_string fcn in
-              let fc2 = create_vsymbol fc2 (t_type lhs) in
-              (t_attr_copy lhs (t_var fc2), fc2)
-            | _ -> raise Undetermined)
-          | _ -> raise Undetermined
-        in
-        match rvl with
-        | rvh :: rvt -> (
-          let lhs, fc2 = remove_var lhs rvh rvt in
-          let vh, vl =
-            match vl with [] -> assert false | vh :: vl -> (vh, vl)
-          in
-          let t2 = term_of_value t2 in
-          match vl with
-          | [] -> elim body vh t2
-          | _ ->
-            let eq = equ lhs body in
-            let tq = t_quant Tforall (t_close_quant vl trig eq) in
-            let body = t_attr_copy t (t_eps_close fc2 tq) in
-            {
-              value_stack = rem_st;
-              cont_stack =
-                (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
-                :: rem_cont;
-            })
-        | _ -> raise Undetermined
-      in
+      let fc, t = Term.t_open_bound tb in
       match t.t_node with
-      | Tapp (ls1, [ lhs; body ]) when ls_equal ls1 ps_equ ->
-        let equ lhs body = t_attr_copy t (t_app ps_equ [ lhs; body ] None) in
-        let elim body vh t2 =
-          {
-            value_stack = rem_st;
-            cont_stack =
-              (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
-              :: rem_cont;
-          }
-        in
-        process lhs body equ elim
-      | Tbinop (Tiff, ({ t_node = Tapp (ls1, [ lhs; tr ]) } as teq), body)
-        when ls_equal ls1 ps_equ && t_equal tr t_bool_true ->
-        let equ lhs body =
-          let lhs = t_attr_copy teq (t_app ps_equ [ lhs; tr ] None) in
-          t_attr_copy t (t_binary Tiff lhs body)
-        in
-        let elim body vh t2 =
-          let body = t_if body t_bool_true t_bool_false in
-          {
-            value_stack = rem_st;
-            cont_stack =
-              (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
-              :: rem_cont;
-          }
-        in
-        process lhs body equ elim
+      | Tquant (Tforall, tq) -> (
+          let vl, trig, t = t_open_quant tq in
+          let process lhs body equ elim =
+            let rvl = List.rev vl in
+            let rec remove_var lhs rvh rvt =
+              match lhs.t_node with
+              | Tapp (ls2, [ lhs1; ({ t_node = Tvar v1 } as arg) ])
+                when ls_equal ls2 fs_func_app && vs_equal v1 rvh -> (
+                  match (rvt, lhs1) with
+                  | rvh :: rvt, _ ->
+                    let lhs1, fc2 = remove_var lhs1 rvh rvt in
+                    let lhs2 = t_app ls2 [ lhs1; arg ] lhs.t_ty in
+                    (t_attr_copy lhs lhs2, fc2)
+                  | [], { t_node = Tvar fc1 } when vs_equal fc1 fc ->
+                    let fcn = fc.vs_name in
+                    let fc2 = Ident.id_derive fcn.Ident.id_string fcn in
+                    let fc2 = create_vsymbol fc2 (t_type lhs) in
+                    (t_attr_copy lhs (t_var fc2), fc2)
+                  | _ -> raise Undetermined)
+              | _ -> raise Undetermined
+            in
+            match rvl with
+            | rvh :: rvt -> (
+                let lhs, fc2 = remove_var lhs rvh rvt in
+                let vh, vl =
+                  match vl with [] -> assert false | vh :: vl -> (vh, vl)
+                in
+                let t2 = term_of_value t2 in
+                match vl with
+                | [] -> elim body vh t2
+                | _ ->
+                  let eq = equ lhs body in
+                  let tq = t_quant Tforall (t_close_quant vl trig eq) in
+                  let body = t_attr_copy t (t_eps_close fc2 tq) in
+                  {
+                    value_stack = rem_st;
+                    cont_stack =
+                      (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
+                      :: rem_cont;
+                  })
+            | _ -> raise Undetermined
+          in
+          match t.t_node with
+          | Tapp (ls1, [ lhs; body ]) when ls_equal ls1 ps_equ ->
+            let equ lhs body = t_attr_copy t (t_app ps_equ [ lhs; body ] None) in
+            let elim body vh t2 =
+              {
+                value_stack = rem_st;
+                cont_stack =
+                  (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
+                  :: rem_cont;
+              }
+            in
+            process lhs body equ elim
+          | Tbinop (Tiff, ({ t_node = Tapp (ls1, [ lhs; tr ]) } as teq), body)
+            when ls_equal ls1 ps_equ && t_equal tr t_bool_true ->
+            let equ lhs body =
+              let lhs = t_attr_copy teq (t_app ps_equ [ lhs; tr ] None) in
+              t_attr_copy t (t_binary Tiff lhs body)
+            in
+            let elim body vh t2 =
+              let body = t_if body t_bool_true t_bool_false in
+              {
+                value_stack = rem_st;
+                cont_stack =
+                  (Keval (body, Mvs.add vh t2 Mvs.empty), t_attr_copy orig body)
+                  :: rem_cont;
+              }
+            in
+            process lhs body equ elim
+          | _ -> raise Undetermined)
       | _ -> raise Undetermined)
-    | _ -> raise Undetermined)
   | _ -> raise Undetermined
 
 and reduce_app_no_equ engine st ls ~orig ty rem_cont =
@@ -1111,111 +1115,116 @@ and reduce_app_no_equ engine st ls ~orig ty rem_cont =
   try
     let f = Hls.find engine.builtins ls in
     let v = f engine ls args ty in
-    { value_stack = v_attr_copy orig v :: rem_st; cont_stack = rem_cont }
+    begin match v with
+      | Value v ->
+        { value_stack = v_attr_copy orig v :: rem_st; cont_stack = rem_cont }
+      | Eval t ->
+        { value_stack = rem_st; cont_stack = (Keval(t,Mvs.empty),t)::rem_cont }
+    end
   with Not_found | Undetermined -> (
-    let args = List.map term_of_value args in
-    match Ident.Mid.find ls.ls_name engine.known_map with
-    | exception Not_found ->
-      Debug.dprintf debug "Reduction engine, ident not found: %s@."
-        ls.ls_name.Ident.id_string;
-      {
-        value_stack = Term (t_attr_copy orig (t_app ls args ty)) :: rem_st;
-        cont_stack = rem_cont;
-      }
-    | d -> (
-      let rewrite () =
-        (* try a rewrite rule *)
-        try
-          (* Format.eprintf "try a rewrite rule on %a@." Pretty.print_ls ls; *)
-          let (mt, mv), rhs = one_step_reduce engine ls args in
-          (* Format.eprintf "rhs = %a@." Pretty.print_term rhs; Format.eprintf
-             "sigma = "; Mvs.iter (fun v t -> Format.eprintf "%a -> %a,"
-             Pretty.print_vs v Pretty.print_term t) (snd sigma); Format.eprintf
-             "@."; Format.eprintf "try a type match: %a and %a@."
-             (Pp.print_option Pretty.print_ty) ty (Pp.print_option
-             Pretty.print_ty) rhs.t_ty; *)
-          (* let type_subst = Ty.oty_match Ty.Mtv.empty rhs.t_ty ty in
-             Format.eprintf "subst of rhs: "; Ty.Mtv.iter (fun tv ty ->
-             Format.eprintf "%a -> %a," Pretty.print_tv tv Pretty.print_ty ty)
-             type_subst; Format.eprintf "@."; let rhs = t_ty_subst type_subst
-             Mvs.empty rhs in let sigma = Mvs.map (t_ty_subst type_subst
-             Mvs.empty) sigma in Format.eprintf "rhs = %a@." Pretty.print_term
-             rhs; Format.eprintf "sigma = "; Mvs.iter (fun v t -> Format.eprintf
-             "%a -> %a," Pretty.print_vs v Pretty.print_term t) sigma;
-             Format.eprintf "@."; *)
-          let mv, rhs = t_subst_types mt mv rhs in
-          incr rec_step_limit;
-          {
-            value_stack = rem_st;
-            cont_stack = (Keval (rhs, mv), orig) :: rem_cont;
-          }
-        with Irreducible ->
-          {
-            value_stack = Term (t_attr_copy orig (t_app ls args ty)) :: rem_st;
-            cont_stack = rem_cont;
-          }
-      in
-      match d.Decl.d_node with
-      | Decl.Dtype _ | Decl.Dprop _ -> assert false
-      | Decl.Dlogic dl ->
-        (* regular definition *)
-        let d = List.assq ls dl in
-        if engine.params.compute_defs
-           || Term.Sls.mem ls engine.params.compute_def_set
-        then
-          let vl, e = Decl.open_ls_defn d in
-          let add (mt, mv) x y =
-            (Ty.ty_match mt x.vs_ty (t_type y), Mvs.add x y mv)
+      let args = List.map term_of_value args in
+      match Ident.Mid.find ls.ls_name engine.known_map with
+      | exception Not_found ->
+        Debug.dprintf debug "Reduction engine, ident not found: %s@."
+          ls.ls_name.Ident.id_string;
+        {
+          value_stack = Term (t_attr_copy orig (t_app ls args ty)) :: rem_st;
+          cont_stack = rem_cont;
+        }
+      | d -> (
+          let rewrite () =
+            (* try a rewrite rule *)
+            try
+              (* Format.eprintf "try a rewrite rule on %a@." Pretty.print_ls ls; *)
+              let (mt, mv), rhs = one_step_reduce engine ls args in
+              (* Format.eprintf "rhs = %a@." Pretty.print_term rhs; Format.eprintf
+                 "sigma = "; Mvs.iter (fun v t -> Format.eprintf "%a -> %a,"
+                 Pretty.print_vs v Pretty.print_term t) (snd sigma); Format.eprintf
+                 "@."; Format.eprintf "try a type match: %a and %a@."
+                 (Pp.print_option Pretty.print_ty) ty (Pp.print_option
+                 Pretty.print_ty) rhs.t_ty; *)
+              (* let type_subst = Ty.oty_match Ty.Mtv.empty rhs.t_ty ty in
+                 Format.eprintf "subst of rhs: "; Ty.Mtv.iter (fun tv ty ->
+                 Format.eprintf "%a -> %a," Pretty.print_tv tv Pretty.print_ty ty)
+                 type_subst; Format.eprintf "@."; let rhs = t_ty_subst type_subst
+                 Mvs.empty rhs in let sigma = Mvs.map (t_ty_subst type_subst
+                 Mvs.empty) sigma in Format.eprintf "rhs = %a@." Pretty.print_term
+                 rhs; Format.eprintf "sigma = "; Mvs.iter (fun v t -> Format.eprintf
+                 "%a -> %a," Pretty.print_vs v Pretty.print_term t) sigma;
+                 Format.eprintf "@."; *)
+              let mv, rhs = t_subst_types mt mv rhs in
+              incr rec_step_limit;
+              {
+                value_stack = rem_st;
+                cont_stack = (Keval (rhs, mv), orig) :: rem_cont;
+              }
+            with Irreducible ->
+              {
+                value_stack = Term (t_attr_copy orig (t_app ls args ty)) :: rem_st;
+                cont_stack = rem_cont;
+              }
           in
-          let mt, mv = List.fold_left2 add (Ty.Mtv.empty, Mvs.empty) vl args in
-          let mt = Ty.oty_match mt e.t_ty ty in
-          let mv, e = t_subst_types mt mv e in
-          {
-            value_stack = rem_st;
-            cont_stack = (Keval (e, mv), orig) :: rem_cont;
-          }
-        else rewrite ()
-      | Decl.Dparam _ | Decl.Dind _ -> rewrite ()
-      | Decl.Ddata dl -> (
-        (* constructor or projection *)
-        try
-          match args with
-          | [ { t_node = Tapp (ls1, tl1) } ] ->
-            (* if ls is a projection and ls1 is a constructor, we should compute
-               that projection *)
-            let rec iter dl =
-              match dl with
-              | [] -> raise Exit
-              | (_, csl) :: rem ->
-                let rec iter2 csl =
-                  match csl with
-                  | [] -> iter rem
-                  | (cs, prs) :: rem2 ->
-                    if ls_equal cs ls1
-                    then
-                      (* we found the right constructor *)
-                      let rec iter3 prs tl1 =
-                        match (prs, tl1) with
-                        | Some pr :: prs, t :: tl1 ->
-                          if ls_equal ls pr
+          match d.Decl.d_node with
+          | Decl.Dtype _ | Decl.Dprop _ -> assert false
+          | Decl.Dlogic dl ->
+            (* regular definition *)
+            let d = List.assq ls dl in
+            if engine.params.compute_defs
+            || Term.Sls.mem ls engine.params.compute_def_set
+            then
+              let vl, e = Decl.open_ls_defn d in
+              let add (mt, mv) x y =
+                (Ty.ty_match mt x.vs_ty (t_type y), Mvs.add x y mv)
+              in
+              let mt, mv = List.fold_left2 add (Ty.Mtv.empty, Mvs.empty) vl args in
+              let mt = Ty.oty_match mt e.t_ty ty in
+              let mv, e = t_subst_types mt mv e in
+              {
+                value_stack = rem_st;
+                cont_stack = (Keval (e, mv), orig) :: rem_cont;
+              }
+            else rewrite ()
+          | Decl.Dparam _ | Decl.Dind _ -> rewrite ()
+          | Decl.Ddata dl -> (
+              (* constructor or projection *)
+              try
+                match args with
+                | [ { t_node = Tapp (ls1, tl1) } ] ->
+                  (* if ls is a projection and ls1 is a constructor, we should compute
+                     that projection *)
+                  let rec iter dl =
+                    match dl with
+                    | [] -> raise Exit
+                    | (_, csl) :: rem ->
+                      let rec iter2 csl =
+                        match csl with
+                        | [] -> iter rem
+                        | (cs, prs) :: rem2 ->
+                          if ls_equal cs ls1
                           then
-                            (* projection found! *)
-                            {
-                              value_stack = Term (t_attr_copy orig t) :: rem_st;
-                              cont_stack = rem_cont;
-                            }
-                          else iter3 prs tl1
-                        | None :: prs, _ :: tl1 -> iter3 prs tl1
-                        | _ -> raise Exit
+                            (* we found the right constructor *)
+                            let rec iter3 prs tl1 =
+                              match (prs, tl1) with
+                              | Some pr :: prs, t :: tl1 ->
+                                if ls_equal ls pr
+                                then
+                                  (* projection found! *)
+                                  {
+                                    value_stack = Term (t_attr_copy orig t) :: rem_st;
+                                    cont_stack = rem_cont;
+                                  }
+                                else iter3 prs tl1
+                              | None :: prs, _ :: tl1 -> iter3 prs tl1
+                              | _ -> raise Exit
+                            in
+                            iter3 prs tl1
+                          else iter2 rem2
                       in
-                      iter3 prs tl1
-                    else iter2 rem2
-                in
-                iter2 csl
-            in
-            iter dl
-          | _ -> raise Exit
-        with Exit -> rewrite ())))
+                      iter2 csl
+                  in
+                  iter dl
+                | _ -> raise Exit
+              with Exit -> rewrite ())))
 
 and reduce_equ ~(* engine *) orig st v1 v2 cont =
   (* try *)
@@ -1229,19 +1238,19 @@ and reduce_equ ~(* engine *) orig st v1 v2 cont =
     incr rec_step_limit;
     { value_stack = Term (t_attr_copy orig b) :: st; cont_stack = cont }
   | Int n, Term { t_node = Tconst c } | Term { t_node = Tconst c }, Int n -> (
-    try
-      let n' = big_int_of_const c in
-      let b = to_bool (BigInt.eq n n') in
-      incr rec_step_limit;
-      { value_stack = Term (t_attr_copy orig b) :: st; cont_stack = cont }
-    with NotNum -> raise Undetermined)
+      try
+        let n' = big_int_of_const c in
+        let b = to_bool (BigInt.eq n n') in
+        incr rec_step_limit;
+        { value_stack = Term (t_attr_copy orig b) :: st; cont_stack = cont }
+      with NotNum -> raise Undetermined)
   | Real r, Term { t_node = Tconst c } | Term { t_node = Tconst c }, Real r -> (
-    try
-      let r' = real_of_const c in
-      let b = to_bool (Number.compare_real r r' = 0) in
-      incr rec_step_limit;
-      { value_stack = Term (t_attr_copy orig b) :: st; cont_stack = cont }
-    with NotNum -> raise Undetermined)
+      try
+        let r' = real_of_const c in
+        let b = to_bool (Number.compare_real r r' = 0) in
+        incr rec_step_limit;
+        { value_stack = Term (t_attr_copy orig b) :: st; cont_stack = cont }
+      with NotNum -> raise Undetermined)
   | Real _, Term _ | Term _, Real _ -> raise Undetermined
   | Int _, Term _ | Term _, Int _ -> raise Undetermined
   | Int _, Real _ | Real _, Int _ -> assert false
@@ -1257,15 +1266,15 @@ and reduce_term_equ ~orig st t1 t2 cont =
   else
     match (t1.t_node, t2.t_node) with
     | Tconst c1, Tconst c2 -> (
-      match (c1, c2) with
-      | Constant.ConstInt i1, Constant.ConstInt i2 ->
-        let b = BigInt.eq i1.Number.il_int i2.Number.il_int in
-        incr rec_step_limit;
-        {
-          value_stack = Term (t_attr_copy orig (to_bool b)) :: st;
-          cont_stack = cont;
-        }
-      | _ -> raise Undetermined)
+        match (c1, c2) with
+        | Constant.ConstInt i1, Constant.ConstInt i2 ->
+          let b = BigInt.eq i1.Number.il_int i2.Number.il_int in
+          incr rec_step_limit;
+          {
+            value_stack = Term (t_attr_copy orig (to_bool b)) :: st;
+            cont_stack = cont;
+          }
+        | _ -> raise Undetermined)
     | Tapp (ls1, tl1), Tapp (ls2, tl2)
       when ls1.ls_constr > 0 && ls2.ls_constr > 0 ->
       if ls_equal ls1 ls2
@@ -1342,23 +1351,23 @@ let normalize ?step_limit ~limit engine sigma t0 =
     | [ Term t ], [] -> t
     | _, [] -> assert false
     | _ -> (
-      if n = limit
-      then
-        let t1 = reconstruct c in
-        (* Loc.warning "reduction of term %a takes more than %d steps, aborted at %a.@."
-         *   Pretty.print_term t0 limit Pretty.print_term t1; *)
-        t1
-      else
-        match step_limit with
-        | None ->
-          let c = reduce engine c in
-          many_steps c (n + 1)
-        | Some step_limit ->
-          if !rec_step_limit >= step_limit
-          then reconstruct c
-          else
+        if n = limit
+        then
+          let t1 = reconstruct c in
+          (* Loc.warning "reduction of term %a takes more than %d steps, aborted at %a.@."
+           *   Pretty.print_term t0 limit Pretty.print_term t1; *)
+          t1
+        else
+          match step_limit with
+          | None ->
             let c = reduce engine c in
-            many_steps c (n + 1))
+            many_steps c (n + 1)
+          | Some step_limit ->
+            if !rec_step_limit >= step_limit
+            then reconstruct c
+            else
+              let c = reduce engine c in
+              many_steps c (n + 1))
   in
   let c = { value_stack = []; cont_stack = [ (Keval (t0, sigma), t0) ] } in
   many_steps c 0
@@ -1366,7 +1375,7 @@ let normalize ?step_limit ~limit engine sigma t0 =
 (* the rewrite engine *)
 
 let create ?(bounded_quant = fun _ _ ~cond:_ -> None) p env km user_env
-  built_in_user =
+    built_in_user =
   let th = Env.read_theory env [ "int" ] "Int" in
   let ls_lt = Theory.ns_find_ls th.Theory.th_export [ Ident.op_infix "<" ] in
   let env =
@@ -1403,9 +1412,9 @@ let extract_rule _km t =
     then raise (NotARewriteRule "lhs should contain all variables");
     (* check the same with type variables *)
     if not
-         (Ty.Stv.subset
-            (t_ty_freevars Ty.Stv.empty t2)
-            (t_ty_freevars Ty.Stv.empty t2))
+        (Ty.Stv.subset
+           (t_ty_freevars Ty.Stv.empty t2)
+           (t_ty_freevars Ty.Stv.empty t2))
     then raise (NotARewriteRule "lhs should contain all type variables")
   in
 
@@ -1415,19 +1424,19 @@ let extract_rule _km t =
       let vs, _, t = t_open_quant q in
       aux (List.fold_left (fun acc v -> Svs.add v acc) acc vs) t
     | Tbinop (Tiff, t1, t2) -> (
-      match t1.t_node with
-      | Tapp (ls, args) ->
-        (* check_ls ls; *)
-        check_vars acc t1 t2;
-        (acc, ls, args, t2)
-      | _ -> raise (NotARewriteRule "lhs of <-> should be a predicate symbol"))
+        match t1.t_node with
+        | Tapp (ls, args) ->
+          (* check_ls ls; *)
+          check_vars acc t1 t2;
+          (acc, ls, args, t2)
+        | _ -> raise (NotARewriteRule "lhs of <-> should be a predicate symbol"))
     | Tapp (ls, [ t1; t2 ]) when ls == ps_equ -> (
-      match t1.t_node with
-      | Tapp (ls, args) ->
-        (* check_ls ls; *)
-        check_vars acc t1 t2;
-        (acc, ls, args, t2)
-      | _ -> raise (NotARewriteRule "lhs of = should be a function symbol"))
+        match t1.t_node with
+        | Tapp (ls, args) ->
+          (* check_ls ls; *)
+          check_vars acc t1 t2;
+          (acc, ls, args, t2)
+        | _ -> raise (NotARewriteRule "lhs of = should be a function symbol"))
     | _ ->
       raise
         (NotARewriteRule
diff --git a/src/reduction_engine.mli b/src/reduction_engine.mli
index bcb1c4863f6441089c3fb45ad0744a5fc6c82ca5..5397657e160dd34d7c5de758f17f3442c6f8866f 100644
--- a/src/reduction_engine.mli
+++ b/src/reduction_engine.mli
@@ -85,13 +85,18 @@ type params = {
     compute builtin functions. . [compute_max_quantifier_domain]: maximum domain
     size for the reduction of bounded quantifications *)
 
-type value =
-  | Term of Why3.Term.term (* invariant: is in normal form *)
-  | Int of BigInt.t
-  | Real of Number.real_value
+  type value =
+    | Term of Why3.Term.term (* invariant: is in normal form *)
+    | Int of BigInt.t
+    | Real of Number.real_value
+  
+
+type builtin_value =
+  | Eval of Why3.Term.term
+  | Value of value
 
 type 'a builtin =
-  'a engine -> Why3.Term.lsymbol -> value list -> Ty.ty option -> value
+  'a engine -> Why3.Term.lsymbol -> value list -> Ty.ty option -> builtin_value
 
 type 'a built_in_theories =
   Env.pathname