summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorSon Ho2022-03-03 13:02:00 +0100
committerSon Ho2022-03-03 13:02:00 +0100
commit3e9083a10b0dc8caf6cebcd89aba27ec7d0a1dac (patch)
tree0621f37cae76e39505aea3d52129b3fa749fe159 /src
parenteeac69419158552ef455a4197e78567837c546ca (diff)
Make good progress updating Translate
Diffstat (limited to '')
-rw-r--r--src/ExtractToFStar.ml10
-rw-r--r--src/Translate.ml93
2 files changed, 94 insertions, 9 deletions
diff --git a/src/ExtractToFStar.ml b/src/ExtractToFStar.ml
index 8bf02bd0..1d144a40 100644
--- a/src/ExtractToFStar.ml
+++ b/src/ExtractToFStar.ml
@@ -13,13 +13,14 @@ module F = Format
Controls whether we should use `type ...` or `and ...` (for mutually
recursive datatypes).
*)
-type type_decl_qualif = Type | And
+type type_decl_qualif = Type | And | AssumeType
(** A qualifier for function definitions.
- Controls whether we should use `let ...`, `let rec ...` or `and ...`
+ Controls whether we should use `let ...`, `let rec ...` or `and ...`,
+ or only generate a declaration with `val` or `assume val`
*)
-type fun_decl_qualif = Let | LetRec | And
+type fun_decl_qualif = Let | LetRec | And | Val | AssumeVal
(** Small helper to compute the name of an int type *)
let fstar_int_name (int_ty : integer_type) =
@@ -448,6 +449,9 @@ let extract_type_decl_register_names (ctx : extraction_ctx) (def : type_decl) :
(ctx_add_variants def
(VariantId.mapi (fun id v -> (id, v)) variants)
ctx)
+ | Opaque ->
+ (* Nothing to do *)
+ ctx
in
(* Return *)
ctx
diff --git a/src/Translate.ml b/src/Translate.ml
index 3b42d13c..9b748461 100644
--- a/src/Translate.ml
+++ b/src/Translate.ml
@@ -91,8 +91,8 @@ let translate_function_to_symbolics (config : C.partial_config)
let translate_function_to_pure (config : C.partial_config)
(mp_config : Micro.config) (trans_ctx : trans_ctx)
(fun_sigs : SymbolicToPure.fun_sig_named_outputs RegularFunIdMap.t)
- (pure_type_decls : Pure.type_decl Pure.TypeDeclId.Map.t) (fdef : A.fun_decl) :
- pure_fun_translation =
+ (pure_type_decls : Pure.type_decl Pure.TypeDeclId.Map.t) (fdef : A.fun_decl)
+ : pure_fun_translation =
(* Debug *)
log#ldebug
(lazy
@@ -317,9 +317,32 @@ type gen_config = {
extract_decreases_clauses : bool;
extract_template_decreases_clauses : bool;
extract_fun_decls : bool;
+ extract_transparent : bool;
+ (** If `true`, extract the transparent declarations, otherwise ignore. *)
+ extract_opaque : bool;
+ (** If `true`, extract the opaque declarations, otherwise ignore. *)
+ interface : bool;
+ (** `true` if we generate an interface file, `false` otherwise.
+ For now, this only impacts whether we use `val` or `assume val` for the
+ opaque definitions. In the future, we might want to extract all the
+ declarations in an interface file, together with an implementation file
+ if needed.
+ *)
test_unit_functions : bool;
}
+(** Returns the pair: (has opaque type decls, has opaque fun decls) *)
+let module_has_opaque_decls (ctx : gen_ctx) : bool * bool =
+ let has_opaque_types =
+ T.TypeDeclId.Map.exists
+ (fun _ (d : Pure.type_decl) ->
+ match d.kind with Opaque -> true | _ -> false)
+ ctx.trans_types
+ in
+ (* TODO: *)
+ let has_opaque_funs = false in
+ (has_opaque_types, has_opaque_funs)
+
(** A generic utility to generate the extracted definitions: as we may want to
split the definitions between different files (or not), we can control
what is precisely extracted.
@@ -329,8 +352,25 @@ let extract_definitions (fmt : Format.formatter) (config : gen_config)
(* Export the definition groups to the file, in the proper order *)
let export_type (qualif : ExtractToFStar.type_decl_qualif)
(id : Pure.TypeDeclId.id) : unit =
+ (* Retrive the declaration *)
let def = Pure.TypeDeclId.Map.find id ctx.trans_types in
- ExtractToFStar.extract_type_decl ctx.extract_ctx fmt qualif def
+ (* Update the qualifier, if the type is opaque *)
+ let is_opaque, qualif =
+ match def.kind with
+ | Enum _ | Struct _ -> (false, qualif)
+ | Opaque ->
+ let qualif =
+ if config.interface then ExtractToFStar.Type
+ else ExtractToFStar.AssumeType
+ in
+ (true, qualif)
+ in
+ (* Extract, if the config instructs to do so (depending on whether the type
+ * is opaque or not) *)
+ if
+ (is_opaque && config.extract_opaque)
+ || ((not is_opaque) && config.extract_transparent)
+ then ExtractToFStar.extract_type_decl ctx.extract_ctx fmt qualif def
in
(* Utility to check a function has a decrease clause *)
@@ -403,6 +443,7 @@ let extract_definitions (fmt : Format.formatter) (config : gen_config)
| Type (NonRec id) ->
if config.extract_types then export_type ExtractToFStar.Type id
| Type (Rec ids) ->
+ (* Rk.: we shouldn't have (mutually) recursive opaque types *)
if config.extract_types then
List.iteri
(fun i id ->
@@ -572,14 +613,30 @@ let translate_module (filename : string) (dest_dir : string) (config : config)
extract_decreases_clauses = config.extract_decreases_clauses;
extract_template_decreases_clauses = false;
extract_fun_decls = false;
+ extract_transparent = true;
+ extract_opaque = false;
+ interface = false;
test_unit_functions = false;
}
in
+ (* Check if there are opaque types and functions - in which case we need
+ * to split *)
+ let has_opaque_types, has_opaque_funs = module_has_opaque_decls gen_ctx in
+
(* Extract the types *)
- let types_filename = extract_filebasename ^ ".Types.fst" in
+ (* If there are opaque types, we extract in an interface *)
+ let types_filename_ext = if has_opaque_types then ".fst" else ".fsti" in
+ let types_filename = extract_filebasename ^ ".Types" ^ types_filename_ext in
let types_module = module_name ^ ".Types" in
- let types_config = { base_gen_config with extract_types = true } in
+ let types_config =
+ {
+ base_gen_config with
+ extract_types = true;
+ extract_opaque = true;
+ interface = has_opaque_types;
+ }
+ in
extract_file types_config gen_ctx types_filename m.M.name types_module
": type definitions" [] [];
@@ -597,6 +654,27 @@ let translate_module (filename : string) (dest_dir : string) (config : config)
clauses_module ": templates for the decreases clauses" [ types_module ]
[]);
+ (* Extract the opaque functions, if needed *)
+ let opaque_funs_module =
+ if has_opaque_funs then (
+ let opaque_filename =
+ extract_filebasename ^ ".Opaque" ^ types_filename_ext
+ in
+ let opaque_module = module_name ^ ".Opaque" in
+ let opaque_config =
+ {
+ base_gen_config with
+ extract_fun_decls = true;
+ extract_transparent = false;
+ extract_opaque = true;
+ }
+ in
+ extract_file opaque_config gen_ctx opaque_filename m.M.name
+ opaque_module ": type definitions" [] [];
+ [ opaque_module ])
+ else []
+ in
+
(* Extract the functions *)
let fun_filename = extract_filebasename ^ ".Funs.fst" in
let fun_module = module_name ^ ".Funs" in
@@ -610,7 +688,7 @@ let translate_module (filename : string) (dest_dir : string) (config : config)
let clauses_module = module_name ^ ".Clauses" in
extract_file fun_config gen_ctx fun_filename m.M.name fun_module
": function definitions" []
- [ types_module; clauses_module ])
+ ([ types_module ] @ opaque_funs_module @ [ clauses_module ]))
else
let gen_config =
{
@@ -619,6 +697,9 @@ let translate_module (filename : string) (dest_dir : string) (config : config)
extract_template_decreases_clauses =
config.extract_template_decreases_clauses;
extract_fun_decls = true;
+ extract_transparent = true;
+ extract_opaque = true;
+ interface = false;
test_unit_functions = config.test_unit_functions;
}
in