aboutsummaryrefslogtreecommitdiff
path: root/toplevel/locality.ml
diff options
context:
space:
mode:
authorgareuselesinge2013-04-15 13:33:29 +0000
committergareuselesinge2013-04-15 13:33:29 +0000
commit12330e6f63404c236614070b32a10b146f9b8a04 (patch)
tree6397b8bd8218f3835cead6efa58f5f955272f837 /toplevel/locality.ml
parent0ea545ef3301be9590d9a1ab7d3eee35d7caec92 (diff)
More functional implementation of locality_flag and program_mode
This commit introduces 2 new vernac_expr constructors: - VernacLocal (b,v) that represents a vernacular v with the "Local" modifier - VernacProgram v that represents a vernacular v with the "Program" modifier This allows the parser to avoid using side effects to model the two modifiers, that are now represented in the AST. This also decouples the parsing phase from the interpretation phase, since parsing a second phrase does not alter the locality flag for the first phrase. As a consequence all the locality_flag components of vernac_expr have been removed, but for the ones that (for retro compatibility) allow an "infix" Local flag. In these cases the boolean is renamed obsolete_locality (as the grammar entry that parses it), and during interpretation we check that at most one locality flag is specified, using the idiom (where the input local is the obsolete one): let local = enforce_XXX_locality locality local in Another improvement is that the default locality is not chosen in the parser, but in the interpreter where the idiom let local = make_XXX_locality locality in is used to default the locality to XXX (module/section/whatever). Unfortunately not all side effects have been removed: - Flags.program_mode is still used to signal that we are in program mode - Locality.LocalityFixme.* functions are used in commands that do not have an AST, but are parsed as VernacExtend (see vernacinterp.ml) I guess one could fix the latter case systematically adding an extra argument "locality" to commands attached using VERNAC COMMAND EXTEND. Fixing plugins adding commands that honour "Local" should look like this: VERNAC COMMAND EXTEND Set_Solver | [ "Obligation" "Tactic" ":=" tactic(t) ] -> [ set_default_tactic - (Locality.use_section_locality ()) + (Locality.make_section_locality (Locality.LocalityFixme.consume ())) (Tacintern.glob_tactic t) ] END In any case the side effects are set/consumed within then interpretation phase, and not set during the parsing phase and consumed during the interpretation phase. git-svn-id: svn+ssh://scm.gforge.inria.fr/svn/coq/trunk@16396 85f007b7-540e-0410-9357-904b9bb8a0f7
Diffstat (limited to 'toplevel/locality.ml')
-rw-r--r--toplevel/locality.ml76
1 files changed, 32 insertions, 44 deletions
diff --git a/toplevel/locality.ml b/toplevel/locality.ml
index 88c3b3c056..49478cb261 100644
--- a/toplevel/locality.ml
+++ b/toplevel/locality.ml
@@ -8,30 +8,27 @@
(** * Managing locality *)
-let locality_flag = ref None
-
let local_of_bool = function
| true -> Decl_kinds.Local
| false -> Decl_kinds.Global
-let check_locality () =
- match !locality_flag with
- | Some (loc,b) ->
+let check_locality locality_flag =
+ match locality_flag with
+ | Some b ->
let s = if b then "Local" else "Global" in
- Errors.user_err_loc
- (loc,"",Pp.str ("This command does not support the \""^s^"\" prefix."))
+ Errors.error ("This command does not support the \""^s^"\" prefix.")
| None -> ()
(** Extracting the locality flag *)
(* Commands which supported an inlined Local flag *)
-let enforce_locality_full local =
+let enforce_locality_full locality_flag local =
let local =
- match !locality_flag with
- | Some (_,false) when local ->
+ match locality_flag with
+ | Some false when local ->
Errors.error "Cannot be simultaneously Local and Global."
- | Some (_,true) when local ->
+ | Some true when local ->
Errors.error "Use only prefix \"Local\"."
| None ->
if local then begin
@@ -40,18 +37,9 @@ let enforce_locality_full local =
Some true
end else
None
- | Some (_,b) -> Some b in
- locality_flag := None;
+ | Some b -> Some b in
local
-(* Commands which did not supported an inlined Local flag (synonym of
- [enforce_locality_full false]) *)
-
-let use_locality_full () =
- let r = Option.map snd !locality_flag in
- locality_flag := None;
- r
-
(** Positioning locality for commands supporting discharging and export
outside of modules *)
@@ -59,23 +47,19 @@ let use_locality_full () =
Global is the default and is neutral;
Local in a section deactivates discharge,
Local not in a section deactivates export *)
+let make_non_locality = function Some false -> false | _ -> true
let make_locality = function Some true -> true | _ -> false
-let use_locality () = make_locality (use_locality_full ())
-
-let use_locality_exp () = local_of_bool (use_locality ())
-
-let enforce_locality local = make_locality (enforce_locality_full local)
+let enforce_locality locality_flag local =
+ make_locality (enforce_locality_full locality_flag local)
-let enforce_locality_exp local = local_of_bool (enforce_locality local)
-
-(* For commands whose default is not to discharge and not to export:
- Global forces discharge and export;
- Local is the default and is neutral *)
-
-let use_non_locality () =
- match use_locality_full () with Some false -> false | _ -> true
+let enforce_locality_exp locality_flag local =
+ match locality_flag, local with
+ | None, Some local -> local
+ | Some b, None -> local_of_bool b
+ | None, None -> Decl_kinds.Global
+ | Some _, Some _ -> Errors.error "Local non allowed in this case"
(* For commands whose default is to not discharge but to export:
Global in sections forces discharge, Global not in section is the default;
@@ -84,11 +68,8 @@ let use_non_locality () =
let make_section_locality =
function Some b -> b | None -> Lib.sections_are_opened ()
-let use_section_locality () =
- make_section_locality (use_locality_full ())
-
-let enforce_section_locality local =
- make_section_locality (enforce_locality_full local)
+let enforce_section_locality locality_flag local =
+ make_section_locality (enforce_locality_full locality_flag local)
(** Positioning locality for commands supporting export but not discharge *)
@@ -105,8 +86,15 @@ let make_module_locality = function
| Some true -> true
| None -> false
-let use_module_locality () =
- make_module_locality (use_locality_full ())
-
-let enforce_module_locality local =
- make_module_locality (enforce_locality_full local)
+let enforce_module_locality locality_flag local =
+ make_module_locality (enforce_locality_full locality_flag local)
+
+module LocalityFixme = struct
+ let locality = ref None
+ let set l = locality := l
+ let consume () =
+ let l = !locality in
+ locality := None;
+ l
+ let assert_consumed () = check_locality !locality
+end