pieuvre/proof.ml

236 lines
6.8 KiB
OCaml
Raw Normal View History

2024-05-16 12:38:45 +02:00
open Hlam
open Lam
2024-05-16 12:38:45 +02:00
open Types
type context = (id * id * hlam * Types.ty) list
type goal = hlam * Types.ty * context
type proof = goal option * goal list
2024-04-30 11:44:28 +02:00
type tactic =
Exact_term of lam
| Exact_proof of id
2024-04-30 11:44:28 +02:00
| Assumption
2024-05-01 10:44:36 +02:00
| Intros
2024-04-30 11:44:28 +02:00
| Intro
| Cut of ty
| Apply of id
2024-05-06 10:09:40 +02:00
| Split
2024-05-05 20:45:55 +02:00
| Right
| Left
2024-04-30 11:44:28 +02:00
2024-05-11 11:44:43 +02:00
let hyp_count = ref 0
let get_fresh_hyp () =
2024-05-11 11:44:43 +02:00
let n = string_of_int !hyp_count in
incr hyp_count;
let hyp_id = "H" ^ n in
let var_id = "x_" ^ n in
(hyp_id, var_id)
2024-05-11 11:44:43 +02:00
(** replace ref's in a proof *)
2024-05-13 18:05:28 +02:00
let rec clean_context assoc (c : context) : context = match c with
[] -> []
| (s1, s2, h, t)::q -> (s1, s2, clean_hlam assoc h, t)::(clean_context assoc q)
let clean_goal assoc ((h, t, c) : goal) : goal =
(clean_hlam assoc h, t, clean_context assoc c)
let clean_proof ((g, gs) : proof) : (hlam ref * hlam ref) list ref * proof =
2024-05-11 11:44:43 +02:00
let assoc = ref [] in
let g' = match g with
2024-05-13 18:05:28 +02:00
| Some g -> Some (clean_goal assoc g)
2024-05-11 11:44:43 +02:00
| None -> None
2024-05-13 18:05:28 +02:00
in assoc, (g', List.map (clean_goal assoc) gs)
2024-05-11 11:44:43 +02:00
let goal_is_over ((g, _) : proof) : bool =
match g with
None -> true
| Some _ -> false
let typecheck (e : lam) (expected_t : Types.ty) (cs : context) : bool =
let gam_of_ctx : context -> Types.gam =
let f = fun (_, var_id, _, ty) -> (var_id, ty) in
List.map f
in
let g = gam_of_ctx cs in
try Typing.typecheck g e expected_t
with _ -> raise (TacticFailed "unable to type")
let rec get_term_by_id (hyp : id) : context -> hlam option =
function
[] -> None
| (hyp',_, h, _) :: _ when hyp' = hyp -> Some h
| _ :: cs -> get_term_by_id hyp cs
let rec get_term_by_type (ty : Types.ty) : context -> hlam option =
function
[] -> None
| (_, _, h, ty') :: _ when ty' = ty -> Some h
| _ :: cs -> get_term_by_type ty cs
let next_goal (gs : goal list) : (goal option * goal list) =
match gs with
[] -> None, []
| g :: gs -> Some g, gs
2024-04-30 11:48:10 +02:00
let tact_exact_term ((g, gs) : proof) (e : lam) : proof =
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (h, expected_t, cs) ->
if typecheck e expected_t cs
then
begin
fill h (hlam_of_lam e);
next_goal gs
end
else raise (TacticFailed "type mismatch")
let tact_exact_proof ((g, gs) : proof) (hyp : id) : proof =
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (h, expected_t, cs) ->
match get_term_by_id hyp cs with
Some h' ->
if typecheck (lam_of_hlam h') expected_t cs
then
begin
fill h h';
next_goal gs
end
else raise (TacticFailed "type mismatch")
2024-05-01 11:07:40 +02:00
| None -> raise (TacticFailed "")
2024-04-30 11:48:10 +02:00
let tact_assumption ((g, gs) : proof) : proof =
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
match get_term_by_type goal_ty cs with
None -> (* failwith "assumption failed" *) (g, gs)
| Some h' ->
fill h h';
next_goal gs
let tact_intro ((g, gs) : proof) : proof =
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
match goal_ty with
Arr (t1, t2) ->
let (hyp_id, var_id) = get_fresh_hyp () in
let cs = (hyp_id, var_id, HVar var_id, t1) :: cs in
let new_h = Ref (ref Hole) in
fill h (HFun ((var_id, t1), new_h));
Some (new_h, t2, cs), gs
2024-04-30 11:48:10 +02:00
| _ -> (* failwith "expected function" *) (* (g, gs) *)
2024-05-01 11:07:40 +02:00
raise (TacticFailed "expected function")
let tact_cut ((g, gs) : proof) (new_t : Types.ty) : proof =
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
(* subgoal 2 : new_t -> goal_ty *)
let arrow_h = Ref (ref Hole) in
let arrow_goal = (arrow_h, Arr (new_t, goal_ty), cs) in
let gs = arrow_goal :: gs in
(* subgoal 1 (main goal) : new_t *)
let new_h = Ref (ref Hole) in
fill h (HApp (arrow_h, new_h));
Some (new_h, new_t, cs), gs
let tact_apply ((g, gs) : proof) (hyp_id : id) : proof =
2024-05-10 18:40:33 +02:00
let rec is_implied (goal_ty : ty) (t : ty) : bool =
match t with
t when t = goal_ty -> true
| Arr (_, t2) -> is_implied goal_ty t2
| _ -> false
in
let rec generate_goals (goal_ty : ty) (impl_ty : ty) (impl_h : hlam)
(goal_h : hlam) (h : hlam) (cs : context) (gs : goal list) : proof =
match impl_ty with
2024-05-10 18:40:33 +02:00
Arr (t1, t2) when t2 = goal_ty ->
let sub_h = Ref (ref Hole) in
let _ = fill h sub_h in
let _ = fill goal_h impl_h in
Some (sub_h, t1, cs), gs
2024-05-10 18:40:33 +02:00
| Arr (t1, t2) ->
let sub_h = Ref (ref Hole) in
let new_h = Ref (ref Hole) in
let _ = fill h sub_h in
let impl_h = HApp (impl_h, new_h) in
let gs = (sub_h, t1, cs) :: gs in
generate_goals goal_ty t2 impl_h goal_h new_h cs gs
| _ -> failwith "impossible"
2024-05-10 18:40:33 +02:00
in
let rec get_hyp : context -> (hlam * ty) = function
2024-05-01 11:07:40 +02:00
[] -> raise (TacticFailed "no such hypothesis in context")
| (hyp_id', _, h', t') :: _ when hyp_id = hyp_id' -> (h', t')
| _ :: cs -> get_hyp cs
in
match g with
2024-05-01 11:07:40 +02:00
None -> raise (TacticFailed "no current goal")
| Some (goal_h, goal_ty, cs) ->
let impl_h, impl_ty = get_hyp cs in
let new_h = Ref (ref Hole) in
let impl_h_2 = HApp (impl_h, new_h) in
if is_implied goal_ty impl_ty
then generate_goals goal_ty impl_ty impl_h_2 goal_h new_h cs gs
2024-05-10 18:40:33 +02:00
else raise (TacticFailed "not an implication")
2024-04-30 11:48:10 +02:00
let tact_intros : proof -> proof =
let rec push (p : proof) =
try
let p = tact_intro p in
push p
2024-05-01 11:07:40 +02:00
with TacticFailed _ -> p
2024-04-30 11:48:10 +02:00
in push
2024-05-06 10:09:40 +02:00
let tact_split ((g, gs) : proof) : proof =
match g with
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
match goal_ty with
| And(t1, t2) ->
2024-05-14 15:23:44 +02:00
let h1 = Ref (ref Hole) in
let h2 = Ref (ref Hole) in
fill h (HPair (h1, h2));
Some (h1, t1, cs), (h2, t2, cs)::gs
2024-05-06 10:09:40 +02:00
| _ -> raise (TacticFailed "Not a conjonction")
2024-05-05 20:45:55 +02:00
let tact_right ((g, gs) : proof) : proof =
match g with
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
match goal_ty with
2024-05-17 08:10:26 +02:00
| Or(_, t_r) as t ->
2024-05-14 15:23:44 +02:00
let new_h = Ref (ref Hole) in
2024-05-17 08:10:26 +02:00
fill h (HRight (new_h, t));
Some (new_h, t_r, cs), gs
2024-05-05 20:45:55 +02:00
| _ -> raise (TacticFailed "Not a disjunction")
let tact_left ((g, gs) : proof) : proof =
match g with
None -> raise (TacticFailed "no current goal")
| Some (h, goal_ty, cs) ->
match goal_ty with
2024-05-17 08:10:26 +02:00
| Or(t_l, _) as t->
2024-05-14 15:23:44 +02:00
let new_h = Ref (ref Hole) in
2024-05-17 08:10:26 +02:00
fill h (HLeft (new_h, t));
Some (new_h, t_l, cs), gs
2024-05-05 20:45:55 +02:00
| _ -> raise (TacticFailed "Not a disjunction")
2024-05-01 10:44:36 +02:00
let apply_tactic (p : proof) (t : tactic) : proof =
2024-04-30 11:48:10 +02:00
match t with
Exact_term e -> tact_exact_term p e
| Exact_proof hyp -> tact_exact_proof p hyp
2024-05-01 10:44:36 +02:00
| Intros -> tact_intros p
2024-04-30 11:48:10 +02:00
| Intro -> tact_intro p
| Assumption -> tact_assumption p
| Cut t -> tact_cut p t
| Apply h -> tact_apply p h
2024-05-06 10:09:40 +02:00
| Split -> tact_split p
2024-05-05 20:45:55 +02:00
| Right -> tact_right p
| Left -> tact_left p
2024-05-01 10:44:36 +02:00
let tact_try (p : proof) (t : tactic) : proof =
try apply_tactic p t
2024-05-01 11:07:40 +02:00
with TacticFailed _ -> p