2024-04-30 11:44:28 +02:00
open Parser_entry
2024-04-09 11:30:52 +02:00
open Affichage
2024-04-30 11:44:28 +02:00
open Proof
open Types
2024-05-16 12:38:45 +02:00
open Hlam
2024-04-09 11:09:33 +02:00
2024-04-16 11:40:08 +02:00
type entry =
2024-05-14 11:43:57 +02:00
Simple of ( unit -> cmd )
2024-04-25 04:20:12 +02:00
| Reduce of Lam . lam
2024-04-16 11:40:08 +02:00
| AlphaEquiv of Lam . lam * Lam . lam
2024-05-13 18:05:28 +02:00
type interactive_state = ( hlam * ty ) option * proof
2024-04-16 11:40:08 +02:00
2024-04-30 11:44:28 +02:00
let parse_lam t =
match Parser . main Lexer . token t with
| Lam l -> l
| Cmd _ -> failwith " entry must be a lam "
let parse_cmd t =
match Parser . main Lexer . token t with
| Cmd c -> c
| Lam _ -> failwith " entry must be a cmd "
2024-05-13 18:05:28 +02:00
let show_beta_reduction e =
2024-04-25 04:20:12 +02:00
let rec aux = function
Some e ->
print_expr e ;
print_newline () ;
aux ( Lam . betastep e ) ;
| None -> ()
in print_expr e ;
print_newline () ;
2024-04-25 04:33:42 +02:00
let e = Lam . alpha_convert ~ readable : true e in
2024-04-25 04:20:12 +02:00
print_expr e ;
print_newline () ;
aux ( Lam . betastep e )
2024-05-13 18:05:28 +02:00
let rec beta_reduce ( l : Lam . lam ) =
match Lam . betastep l with
| None -> l
| Some l' -> beta_reduce l'
let clean_state ( ( s , p ) : interactive_state ) =
let assoc , new_p = clean_proof p in
match s with
| None -> None , new_p
| Some ( hl , ty ) -> Some ( clean_hlam assoc hl , ty ) , new_p
2024-04-25 04:20:12 +02:00
let alpha_get_lam where_from =
let input_str = In_channel . input_all where_from in
match Str . split ( Str . regexp " & " ) input_str with
[ s1 ; s2 ] -> AlphaEquiv (
2024-04-30 11:44:28 +02:00
parse_lam ( Lexing . from_string ( s1 ^ " \n " ) ) ,
parse_lam ( Lexing . from_string s2 )
2024-04-25 04:20:12 +02:00
)
| _ -> failwith " Alpha-equivalence: nombre de delimiteurs incorrect "
2024-05-13 18:05:28 +02:00
(* * Interactive loop
- cg : current top goal : type and reference to lambda - term
- g , gs : next goals
- sq : previous states of the interactive loop
* )
2024-05-14 11:43:57 +02:00
let rec interactive ( get_cmd : unit -> cmd ) ( sl : ( interactive_state ) list ) : proof =
2024-05-14 10:49:24 +02:00
let ( cg , ( g , gs ) ) , sq = match sl with
[] -> ( None , ( None , [] ) ) , []
| s :: sq -> s , sq
in
2024-04-30 11:44:28 +02:00
begin
let _ = match g with
2024-05-01 11:07:40 +02:00
None -> print_string " \n \027 [1mNo more goals. \027 [0m \n "
2024-05-01 10:44:36 +02:00
| Some g' -> print_newline () ; print_goal g'
2024-04-30 11:44:28 +02:00
in
2024-05-01 10:35:12 +02:00
try
2024-05-14 11:43:57 +02:00
match get_cmd () with
2024-05-13 18:05:28 +02:00
Goal ty ->
let rh = Ref ( ref Hole ) in
2024-05-14 11:43:57 +02:00
[ Some ( rh , ty ) , ( Some ( rh , ty , [] ) , [] ) ] | > interactive get_cmd
| Undo -> interactive get_cmd sq
2024-05-13 18:05:28 +02:00
| Qed -> begin match cg with
None ->
print_error " No current goal " " " ;
2024-05-14 11:43:57 +02:00
( cg , ( g , gs ) ) :: sq | > interactive get_cmd
2024-05-13 18:05:28 +02:00
| Some ( h , t ) ->
let l = lam_of_hlam h
| > beta_reduce in
if Typing . typecheck [] l t then begin
print_string " Ok " ;
2024-05-14 11:43:57 +02:00
[ None , ( None , [] ) ] | > interactive get_cmd
2024-05-13 18:05:28 +02:00
end else begin
print_error " Typing failed " " " ;
2024-05-16 12:31:06 +02:00
print_expr l ;
2024-05-14 10:49:24 +02:00
print_newline () ;
2024-05-16 12:31:06 +02:00
print_ty t ;
2024-05-14 11:43:57 +02:00
( cg , ( g , gs ) ) :: sq | > interactive get_cmd
2024-05-13 18:05:28 +02:00
end
end
| Tact t ->
2024-05-14 11:43:57 +02:00
( cg , ( apply_tactic ( g , gs ) t ) ) :: ( clean_state ( cg , ( g , gs ) ) ) :: sq | > interactive get_cmd
2024-05-01 10:35:12 +02:00
with
Parser . Error ->
2024-05-01 11:07:40 +02:00
print_error " Invalid input " " " ;
2024-05-14 11:43:57 +02:00
( cg , ( g , gs ) ) :: sq | > interactive get_cmd
2024-05-11 11:44:43 +02:00
| TacticFailed arg ->
print_error " Tactic failed " arg ;
2024-05-14 11:43:57 +02:00
( cg , ( g , gs ) ) :: sq | > interactive get_cmd
| End_of_file | Lexer . Eof ->
2024-05-01 10:35:12 +02:00
print_string " Bye! \n " ;
( g , gs )
2024-04-30 11:44:28 +02:00
end
2024-04-16 11:40:08 +02:00
let nom_fichier = ref " "
2024-04-25 04:20:12 +02:00
let reduce = ref false
2024-04-16 14:24:06 +02:00
let alpha = ref false
2024-04-25 04:20:12 +02:00
let equiv_fichier = ref " "
2024-04-16 11:40:08 +02:00
let recupere_entree () =
let optlist = [
2024-04-25 04:20:12 +02:00
( " -alpha " ,
Arg . Set alpha ,
" Vérifie l'alpha équivalence de deux termes séparés par & " ) ;
( " -reduce " ,
Arg . Set reduce ,
" Affiche les réductions successives du lambda-terme " )
2024-04-16 11:40:08 +02:00
] in
let usage = " Bienvenue à bord. " in (* message d'accueil, option -help *)
Arg . parse (* ci-dessous les 3 arguments de Arg.parse : *)
optlist (* la liste des options definie plus haut *)
( fun s -> nom_fichier := s ) (* la fonction a declencher lorsqu'on recupere un string qui n'est pas une option : ici c'est le nom du fichier, et on stocke cette information dans la reference nom_fichier *)
usage ; (* le message d'accueil *)
2024-04-16 14:24:06 +02:00
try
2024-04-16 11:40:08 +02:00
let where_from = match ! nom_fichier with
| " " -> stdin
| s -> open_in s in
2024-04-25 04:20:12 +02:00
if ! alpha
then alpha_get_lam where_from
else if ! reduce
2024-05-14 11:43:57 +02:00
then Reduce ( Lexing . from_channel where_from | > parse_lam )
else Simple begin
let cmd_buff = ref [] in
if ! nom_fichier = " " then
(
fun () ->
match ! cmd_buff with
| [] ->
begin match ( read_line () ) ^ " \n "
| > Lexing . from_string
| > parse_cmd with
[] -> raise Parser . Error
| e :: q -> cmd_buff := q ; e
end
| e :: q -> cmd_buff := q ; e
)
else
(
fun () ->
match ! cmd_buff with
| [] ->
begin match ( input_line where_from ) ^ " \n "
| > Lexing . from_string
| > parse_cmd with
[] -> raise End_of_file
| e :: q -> cmd_buff := q ; e
end
| e :: q -> cmd_buff := q ; e
)
end
2024-05-01 11:07:40 +02:00
with e -> ( print_error " problème de saisie " " " ; raise e )
2024-04-09 11:09:33 +02:00
2024-04-16 11:40:08 +02:00
(* la fonction principale *)
let run () =
try
match recupere_entree () with
2024-05-14 11:43:57 +02:00
Simple get_cmd ->
let _ = interactive get_cmd [ None , ( None , [] ) ] in ()
| Reduce l ->
let _ = show_beta_reduction l in ()
2024-04-16 11:40:08 +02:00
| AlphaEquiv ( l1 , l2 ) -> begin
if ( ( Lam . ( = ~ ) ) l1 l2 ) then
print_string " true \n "
else
print_string " false \n "
end ;
flush stdout
with e -> raise e
2024-04-09 11:09:33 +02:00
2024-04-16 11:40:08 +02:00
let _ = run ()
2024-04-09 11:09:33 +02:00