Skip to content

Commit

Permalink
PR: Irreducible solve hints and print hint
Browse files Browse the repository at this point in the history
This PR introduces two new features:

- Irreducible hints:
This features allows the introduction of the "irreducible" keyword
between "hint" and "exact" or "solve" (e.g. "hint irreducible exact") in
order to disable reduction of the hinted expression when auto-solving
goals.

- Print hint:
This introduces the following commands:
"print hint" - prints all hints in the current scope
"print hint simplify" - same but only for simplify hints
"print hint solve" - same but only for solve hints
"print hint rewrite" - same but only for rewrite hints
  • Loading branch information
Gustavo2622 committed Jan 13, 2025
1 parent 9eaff01 commit fa83649
Show file tree
Hide file tree
Showing 17 changed files with 396 additions and 171 deletions.
169 changes: 94 additions & 75 deletions src/ecCommands.ml
Original file line number Diff line number Diff line change
Expand Up @@ -255,6 +255,83 @@ module HiPrinting = struct
fmt (goal, `One sz)
end
end

let pr_axioms (fmt : Format.formatter) (env : EcEnv.env) =
let ax = EcEnv.Ax.all ~check:(fun _ ax -> EcDecl.is_axiom ax.ax_kind) env in
let ppe0 = EcPrinting.PPEnv.ofenv env in
EcPrinting.pp_by_theory ppe0 (EcPrinting.pp_axiom) fmt ax

let pr_hint_solve (fmt : Format.formatter) (env : EcEnv.env) =
let hint_solve = EcEnv.Auto.all env in
let hint_solve = List.map (fun (p, mode) ->
let ax = EcEnv.Ax.by_path p env in
(p, (ax, mode))
) hint_solve in

let ppe = EcPrinting.PPEnv.ofenv env in

let pp_hint_solve ppe fmt = (fun (p, (ax, mode)) ->
let mode =
match mode with
| `Default -> ""
| `Rigid -> "(rigid)" in
Format.fprintf fmt "%a %s" (EcPrinting.pp_axiom ppe) (p, ax) mode
)
in

EcPrinting.pp_by_theory ppe pp_hint_solve fmt hint_solve

let pr_hint_rewrite (fmt : Format.formatter) (env : EcEnv.env) =
let hint_rewrite = EcEnv.BaseRw.all env in

let ppe = EcPrinting.PPEnv.ofenv env in

let pp_hint_rewrite _ppe fmt = (fun (p, sp) ->
let elems = EcPath.Sp.ntr_elements sp in
if List.is_empty elems then
Format.fprintf fmt "%s (empty)@." (EcPath.tostring p)
else
Format.fprintf fmt "%s = @.@[<b 2>%a@]@." (EcPath.tostring p)
(EcPrinting.pp_list "@." (fun fmt p ->
Format.fprintf fmt "%s" (EcPath.tostring p)))
(EcPath.Sp.ntr_elements sp)
)
in

EcPrinting.pp_by_theory ppe pp_hint_rewrite fmt hint_rewrite

let pr_hint_simplify (fmt : Format.formatter) (env : EcEnv.env) =
let open EcTheory in

let (hint_simplify: (EcEnv.Reduction.topsym * rule list) list) = EcEnv.Reduction.all env in

let hint_simplify = List.filter_map (fun (ts, rl) ->
match ts with
| `Path p -> Some (p, rl)
| _ -> None
) hint_simplify
in

let ppe = EcPrinting.PPEnv.ofenv env in

let pp_hint_simplify ppe fmt = (fun (p, (rls : rule list)) ->
Format.fprintf fmt "%s:@.@[<b 2>%a@]" (EcPath.tostring p)
(EcPrinting.pp_list "@." (fun fmt rl ->
Format.fprintf fmt "Conditions: %[email protected]: %[email protected]: %a@."
(EcPrinting.pp_list "," (EcPrinting.pp_form ppe)) rl.rl_cond
(EcPrinting.pp_form ppe) rl.rl_tg
(EcPrinting.pp_rule_pattern ppe) rl.rl_ptn
))
rls
)
in

EcPrinting.pp_by_theory ppe pp_hint_simplify fmt hint_simplify

let pr_hints (fmt : Format.formatter) (env : EcEnv.env) =
let ax = EcEnv.Ax.all ~check:(fun _ ax -> EcDecl.is_axiom ax.ax_kind) env in
let ppe0 = EcPrinting.PPEnv.ofenv env in
EcPrinting.pp_by_theory ppe0 (EcPrinting.pp_axiom) fmt ax
end

(* -------------------------------------------------------------------- *)
Expand All @@ -280,6 +357,23 @@ let process_pr fmt scope p =
| Pr_glob pm -> HiPrinting.pr_glob fmt env pm
| Pr_goal n -> HiPrinting.pr_goal fmt scope n

| Pr_axioms -> HiPrinting.pr_axioms fmt env

| Pr_hint (Some `Simplify) -> HiPrinting.pr_hint_simplify fmt env
| Pr_hint (Some `Solve) -> HiPrinting.pr_hint_solve fmt env
| Pr_hint (Some `Rewrite) -> HiPrinting.pr_hint_rewrite fmt env

| Pr_hint None ->
let printers = [
("Solve" , (fun fmt -> HiPrinting.pr_hint_solve fmt env));
("Simplify", (fun fmt -> HiPrinting.pr_hint_simplify fmt env));
("Rewrite" , (fun fmt -> HiPrinting.pr_hint_rewrite fmt env));
] in

List.iter (fun (header, printer) ->
Format.fprintf fmt "%s hints:@.%t@." header printer
) printers

(* -------------------------------------------------------------------- *)
let check_opname_validity (scope : EcScope.scope) (x : string) =
if EcIo.is_binop x = `Invalid then
Expand All @@ -293,80 +387,6 @@ let check_opname_validity (scope : EcScope.scope) (x : string) =
let process_print scope p =
process_pr Format.std_formatter scope p

(* -------------------------------------------------------------------- *)
let process_print_ax (scope : EcScope.scope) =
let env = EcScope.env scope in
let ax = EcEnv.Ax.all ~check:(fun _ ax -> EcDecl.is_axiom ax.ax_kind) env in

let module Trie : sig
type ('a, 'b) t

val empty : ('a, 'b) t
val add : 'a list -> 'b -> ('a, 'b) t -> ('a, 'b) t
val iter : ('a list -> 'b list -> unit) -> ('a, 'b) t -> unit
end = struct
module Map = BatMap

type ('a, 'b) t =
{ children : ('a, ('a, 'b) t) Map.t
; value : 'b list }

let empty : ('a, 'b) t =
{ value = []; children = Map.empty; }

let add (path : 'a list) (value : 'b) (t : ('a, 'b) t) =
let rec doit (path : 'a list) (t : ('a, 'b) t) =
match path with
| [] ->
{ t with value = value :: t.value }
| v :: path ->
let children =
t.children |> Map.update_stdlib v (fun children ->
let subtrie = Option.value ~default:empty children in
Some (doit path subtrie)
)
in { t with children }
in doit path t

let iter (f : 'a list -> 'b list -> unit) (t : ('a, 'b) t) =
let rec doit (prefix : 'a list) (t : ('a, 'b) t) =
if not (List.is_empty t.value) then
f prefix t.value;
Map.iter (fun k v -> doit (k :: prefix) v) t.children
in

doit [] t
end in

let ax =
List.fold_left (fun axs ((p, _) as ax) ->
Trie.add (EcPath.tolist (oget (EcPath.prefix p))) ax axs
) Trie.empty ax in

let ppe0 = EcPrinting.PPEnv.ofenv env in
let buffer = Buffer.create 0 in
let fmt = Format.formatter_of_buffer buffer in

Trie.iter (fun prefix axs ->
let thpath =
match prefix with
| [] -> assert false
| name :: prefix -> (List.rev prefix, name) in

let thpath = EcPath.fromqsymbol thpath in

let ppe = EcPrinting.PPEnv.enter_theory ppe0 thpath in

Format.fprintf fmt
"@.========== %a ==========@.@." (EcPrinting.pp_thname ppe0) thpath;

List.iter (fun ax ->
Format.fprintf fmt "%a@." (EcPrinting.pp_axiom ppe) ax
) axs
) ax;

EcScope.notify scope `Warning "%s" (Buffer.contents buffer)

(* -------------------------------------------------------------------- *)
exception Pragma of [`Reset | `Restart]

Expand Down Expand Up @@ -734,7 +754,6 @@ and process (ld : Loader.loader) (scope : EcScope.scope) g =
| GsctOpen name -> `Fct (fun scope -> process_sct_open scope name)
| GsctClose name -> `Fct (fun scope -> process_sct_close scope name)
| Gprint p -> `Fct (fun scope -> process_print scope p; scope)
| Gpaxiom -> `Fct (fun scope -> process_print_ax scope; scope)
| Gsearch qs -> `Fct (fun scope -> process_search scope qs; scope)
| Glocate x -> `Fct (fun scope -> process_locate scope x; scope)
| Gtactics t -> `Fct (fun scope -> process_tactics scope t)
Expand Down
6 changes: 6 additions & 0 deletions src/ecCorePrinting.ml
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,12 @@ module type PrinterAPI = sig
val pp_hyps : PPEnv.t -> EcEnv.LDecl.hyps pp
val pp_goal : PPEnv.t -> prpo_display -> ppgoal pp

(* ------------------------------------------------------------------ *)
val pp_by_theory : PPEnv.t -> (PPEnv.t -> (EcPath.path * 'a) pp) -> ((EcPath.path * 'a) list) pp

(* ------------------------------------------------------------------ *)
val pp_rule_pattern : PPEnv.t -> EcTheory.rule_pattern pp

(* ------------------------------------------------------------------ *)
module ObjectInfo : sig
type db = [`Rewrite of qsymbol | `Solve of symbol]
Expand Down
75 changes: 56 additions & 19 deletions src/ecEnv.ml
Original file line number Diff line number Diff line change
Expand Up @@ -181,7 +181,7 @@ type preenv = {
env_tci : ((ty_params * ty) * tcinstance) list;
env_tc : TC.graph;
env_rwbase : Sp.t Mip.t;
env_atbase : (path list Mint.t) Msym.t;
env_atbase : atbase Msym.t;
env_redbase : mredinfo;
env_ntbase : ntbase Mop.t;
env_modlcs : Sid.t; (* declared modules *)
Expand Down Expand Up @@ -221,6 +221,10 @@ and env_notation = ty_params * EcDecl.notation

and ntbase = (path * env_notation) list

and atbase0 = path * [`Rigid | `Default]

and atbase = atbase0 list Mint.t

(* -------------------------------------------------------------------- *)
type env = preenv

Expand Down Expand Up @@ -1444,6 +1448,13 @@ module BaseRw = struct
(omap (fun s -> List.fold_left (fun s r -> Sp.add r s) s l))
(IPPath p) env.env_rwbase;
env_item = mkitem import (Th_addrw (p, l, lc)) :: env.env_item; }
let all env =
List.filter_map (fun (ip, sp) ->
match ip with
| IPPath p -> Some (p, sp)
| _ -> None) @@
Mip.bindings env.env_rwbase
end
(* -------------------------------------------------------------------- *)
Expand Down Expand Up @@ -1496,45 +1507,65 @@ module Reduction = struct
Mrd.find_opt p env.env_redbase
|> omap (fun x -> Lazy.force x.ri_list)
|> odfl []
(* FIXME: handle other cases, right now only used for print hint *)
let all (env : env) =
List.map (fun (ts, mr) ->
(ts, Lazy.force mr.ri_list))
(Mrd.bindings env.env_redbase)
end
(* -------------------------------------------------------------------- *)
module Auto = struct
type base0 = path * [`Rigid | `Default]
let dname : symbol = ""
let updatedb ~level ?base (ps : path list) (db : (path list Mint.t) Msym.t) =
let updatedb
~(level : int)
?(base : symbol option)
(ps : atbase0 list)
(db : atbase Msym.t)
=
let nbase = (odfl dname base) in
let ps' = Msym.find_def Mint.empty nbase db in
let ps' =
let base = Msym.find_def Mint.empty nbase db in
let levels =
let doit x = Some (ofold (fun x ps -> ps @ x) ps x) in
Mint.change doit level ps' in
Msym.add nbase ps' db
let add ?(import = import0) ~level ?base (ps : path list) lc (env : env) =
Mint.change doit level base in
Msym.add nbase levels db
let add
?(import = import0)
~(level : int)
?(base : symbol option)
(axioms : atbase0 list)
(locality : is_local)
(env : env)
=
let env =
if import.im_immediate then
{ env with
env_atbase = updatedb ?base ~level ps env.env_atbase; }
env_atbase = updatedb ?base ~level axioms env.env_atbase; }
else env
in
{ env with env_item = mkitem import
(Th_auto (level, base, ps, lc)) :: env.env_item; }
(Th_auto { level; base; axioms; locality; }) :: env.env_item; }
let add1 ?import ~level ?base (p : path) lc (env : env) =
let add1 ?import ~level ?base (p : atbase0) lc (env : env) =
add ?import ?base ~level [p] lc env
let get_core ?base (env : env) =
Msym.find_def Mint.empty (odfl dname base) env.env_atbase
let flatten_db (db : path list Mint.t) =
let flatten_db (db : atbase) =
Mint.fold_left (fun ps _ ps' -> ps @ ps') [] db
let get ?base (env : env) =
flatten_db (get_core ?base env)
let getall (bases : symbol list) (env : env) =
let getall (bases : symbol list) (env : env) : atbase0 list =
let dbs = List.map (fun base -> get_core ~base env) bases in
let dbs =
let dbs =
List.fold_left (fun db mi ->
Mint.union (fun _ sp1 sp2 -> Some (sp1 @ sp2)) db mi)
Mint.empty dbs
Expand All @@ -1543,6 +1574,9 @@ module Auto = struct
let getx (base : symbol) (env : env) =
let db = Msym.find_def Mint.empty base env.env_atbase in
Mint.bindings db
let all (env : env) : atbase0 list =
Msym.values env.env_atbase |> List.map flatten_db |> List.flatten
end
(* -------------------------------------------------------------------- *)
Expand Down Expand Up @@ -2932,8 +2966,8 @@ module Theory = struct
(* ------------------------------------------------------------------ *)
let bind_at_th =
let for1 _path db = function
| Th_auto (level, base, ps, _) ->
Some (Auto.updatedb ?base ~level ps db)
| Th_auto {level; base; axioms; _} ->
Some (Auto.updatedb ?base ~level axioms db)
| _ -> None
in bind_base_th for1
Expand Down Expand Up @@ -3106,9 +3140,12 @@ module Theory = struct
let ps = List.filter ((not) |- inclear |- oget |- EcPath.prefix) ps in
if List.is_empty ps then None else Some (Th_addrw (p, ps,lc))
| Th_auto (lvl, base, ps, lc) ->
let ps = List.filter ((not) |- inclear |- oget |- EcPath.prefix) ps in
if List.is_empty ps then None else Some (Th_auto (lvl, base, ps, lc))
| Th_auto ({ axioms } as auto_rl) ->
let axioms = List.filter (fun (p, _) ->
let p = oget (EcPath.prefix p) in
not (inclear p)
) axioms in
if List.is_empty axioms then None else Some (Th_auto {auto_rl with axioms})
| (Th_export (p, _)) as item ->
if Sp.mem p cleared then None else Some item
Expand Down
16 changes: 11 additions & 5 deletions src/ecEnv.mli
Original file line number Diff line number Diff line change
Expand Up @@ -398,26 +398,32 @@ module BaseRw : sig

val add : ?import:import -> symbol -> is_local -> env -> env
val addto : ?import:import -> path -> path list -> is_local -> env -> env

val all : env -> (path * Sp.t) list
end

(* -------------------------------------------------------------------- *)
module Reduction : sig
type rule = EcTheory.rule
type topsym = [ `Path of path | `Tuple | `Proj of int]

val all : env -> (topsym * rule list) list
val add1 : path * rule_option * rule option -> env -> env
val add : ?import:import -> (path * rule_option * rule option) list -> env -> env
val get : topsym -> env -> rule list
end

(* -------------------------------------------------------------------- *)
module Auto : sig
type base0 = path * [`Rigid | `Default]

val dname : symbol
val add1 : ?import:import -> level:int -> ?base:symbol -> path -> is_local -> env -> env
val add : ?import:import -> level:int -> ?base:symbol -> path list -> is_local -> env -> env
val get : ?base:symbol -> env -> path list
val getall : symbol list -> env -> path list
val getx : symbol -> env -> (int * path list) list
val add1 : ?import:import -> level:int -> ?base:symbol -> base0 -> is_local -> env -> env
val add : ?import:import -> level:int -> ?base:symbol -> base0 list -> is_local -> env -> env
val get : ?base:symbol -> env -> base0 list
val getall : symbol list -> env -> base0 list
val getx : symbol -> env -> (int * base0 list) list
val all : env -> base0 list
end

(* -------------------------------------------------------------------- *)
Expand Down
Loading

0 comments on commit fa83649

Please sign in to comment.