(***********************************************************************) (* v * The Coq Proof Assistant / The Coq Development Team *) (* anomaly "Notation not in GRAMMAR summary" | Delimiters _ -> anomaly "Delimiters not in GRAMMAR summary" | Grammar gc -> Grammar (subst_grammar_command subst gc) | TacticGrammar g -> TacticGrammar g (* TODO ... *) let (grammar_state : all_grammar_command list ref) = ref [] (**************************************************************************) let assoc_level = function | Gramext.LeftA -> "L" | _ -> "" let constr_level = function | n,assoc -> (string_of_int n)^(assoc_level assoc) let default_levels_v7 = [10,Gramext.RightA; 9,Gramext.RightA; 8,Gramext.RightA; 1,Gramext.RightA; 0,Gramext.RightA] let default_levels_v8 = [200,Gramext.RightA; 100,Gramext.RightA; 80,Gramext.RightA; 10,Gramext.LeftA; 9,Gramext.RightA; 1,Gramext.LeftA; 0,Gramext.RightA] let level_stack = ref [default_levels_v7] (* At a same level, LeftA takes precedence over RightA and NoneA *) (* In case, several associativity exists for a level, we make two levels, *) (* first LeftA, then RightA and NoneA together *) exception Found of Gramext.g_assoc open Ppextend let admissible_assoc = function | Gramext.LeftA, Some (Gramext.RightA | Gramext.NonA) -> false | Gramext.RightA, Some Gramext.LeftA -> false | _ -> true let create_assoc = function | None -> Gramext.RightA | Some a -> a let find_position other assoc lev = let default = if !Options.v7 then (10,Gramext.RightA) else (200,Gramext.RightA) in let current = List.hd !level_stack in match lev with | None -> level_stack := current :: !level_stack; None, (if other then assoc else None), None | Some n -> if !Options.v7 & n = 8 & assoc = Some Gramext.LeftA then error "Left associativity not allowed at level 8"; let after = ref default in let rec add_level q = function | (p,_ as pa)::l when p > n -> pa :: add_level pa l | (p,a as pa)::l as l' when p = n -> if admissible_assoc (a,assoc) then raise (Found a); (* Maybe this was (p,Right) and p occurs a second time *) if a = Gramext.LeftA then match l with | (p,a)::_ as l' when p = n -> raise (Found a) | _ -> after := pa; pa::(n,create_assoc assoc)::l else (* This was not (p,LeftA) hence assoc is RightA *) (after := q; (n,create_assoc assoc)::l') | l -> after := q; (n,create_assoc assoc)::l in try (* Create the entry *) let current = List.hd !level_stack in level_stack := add_level default current :: !level_stack; let assoc = create_assoc assoc in Some (Gramext.After (constr_level !after)), Some assoc, Some (constr_level (n,assoc)) with Found a -> level_stack := current :: !level_stack; (* Just inherit the existing associativity and name (None) *) Some (Gramext.Level (constr_level (n,a))), None, None let remove_levels n = level_stack := list_skipn n !level_stack (* Interpretation of the right hand side of grammar rules *) (* When reporting errors, we add the name of the grammar rule that failed *) let specify_name name e = match e with | UserError(lab,strm) -> UserError(lab, (str"during interpretation of grammar rule " ++ str name ++ str"," ++ spc () ++ strm)) | Anomaly(lab,strm) -> Anomaly(lab, (str"during interpretation of grammar rule " ++ str name ++ str"," ++ spc () ++ strm)) | Failure s -> Failure("during interpretation of grammar rule "^name^", "^s) | e -> e (* Translation of environments: a production * [ nt1(x1) ... nti(xi) ] -> act(x1..xi) * is written (with camlp4 conventions): * (fun vi -> .... (fun v1 -> act(v1 .. vi) )..) * where v1..vi are the values generated by non-terminals nt1..nti. * Since the actions are executed by substituting an environment, * make_act builds the following closure: * * ((fun env -> * (fun vi -> * (fun env -> ... * * (fun v1 -> * (fun env -> gram_action .. env act) * ((x1,v1)::env)) * ...) * ((xi,vi)::env))) * []) *) open Names type 'a action_env = (identifier * 'a) list let make_act (f : loc -> constr_expr action_env -> constr_expr) pil = let rec make (env : constr_expr action_env) = function | [] -> Gramext.action (fun loc -> f loc env) | None :: tl -> (* parse a non-binding item *) Gramext.action (fun _ -> make env tl) | Some (p, (ETConstr _| ETOther _)) :: tl -> (* constr non-terminal *) Gramext.action (fun (v:constr_expr) -> make ((p,v) :: env) tl) | Some (p, ETReference) :: tl -> (* non-terminal *) Gramext.action (fun (v:reference) -> make ((p,CRef v) :: env) tl) | Some (p, ETIdent) :: tl -> (* non-terminal *) Gramext.action (fun (v:identifier) -> make ((p,CRef (Ident (dummy_loc,v))) :: env) tl) | Some (p, ETBigint) :: tl -> (* non-terminal *) Gramext.action (fun (v:Bignat.bigint) -> make ((p,CNumeral (dummy_loc,v)) :: env) tl) | Some (p, ETPattern) :: tl -> failwith "Unexpected entry of type cases pattern" in make [] (List.rev pil) let make_cases_pattern_act (f : loc -> cases_pattern_expr action_env -> cases_pattern_expr) pil = let rec make (env : cases_pattern_expr action_env) = function | [] -> Gramext.action (fun loc -> f loc env) | None :: tl -> (* parse a non-binding item *) Gramext.action (fun _ -> make env tl) | Some (p, ETPattern) :: tl -> (* non-terminal *) Gramext.action (fun v -> make ((p,v) :: env) tl) | Some (p, ETReference) :: tl -> (* non-terminal *) Gramext.action (fun v -> make ((p,CPatAtom(dummy_loc,Some v)) :: env) tl) | Some (p, ETBigint) :: tl -> (* non-terminal *) Gramext.action (fun v -> make ((p,CPatNumeral(dummy_loc,v)) :: env) tl) | Some (p, (ETIdent | ETConstr _ | ETOther _)) :: tl -> error "ident and constr entry not admitted in patterns cases syntax extensions" in make [] (List.rev pil) (* Grammar extension command. Rules are assumed correct. * Type-checking of grammar rules is done during the translation of * ast to the type grammar_command. We only check that the existing * entries have the type assumed in the grammar command (these types * annotations are added when type-checking the command, function * Extend.of_ast) *) let rec build_prod_item univ assoc fromlevel = function | ProdList0 s -> Gramext.Slist0 (build_prod_item univ assoc fromlevel s) | ProdList1 s -> Gramext.Slist1 (build_prod_item univ assoc fromlevel s) | ProdOpt s -> Gramext.Sopt (build_prod_item univ assoc fromlevel s) | ProdPrimitive typ -> symbol_of_production assoc fromlevel typ let symbol_of_prod_item univ assoc from = function | Term tok -> (Gramext.Stoken tok, None) | NonTerm (nt, ovar) -> let eobj = build_prod_item univ assoc from nt in (eobj, ovar) let coerce_to_id = function | CRef (Ident (_,id)) -> id | c -> user_err_loc (constr_loc c, "subst_rawconstr", str"This expression should be a simple identifier") let coerce_to_ref = function | CRef r -> r | c -> user_err_loc (constr_loc c, "subst_rawconstr", str"This expression should be a simple reference") let subst_ref loc subst id = try coerce_to_ref (List.assoc id subst) with Not_found -> Ident (loc,id) let subst_pat_id loc subst id = try List.assoc id subst with Not_found -> CPatAtom (loc,Some (Ident (loc,id))) let subst_id subst id = try coerce_to_id (List.assoc id subst) with Not_found -> id let name_app f = function | Name id -> Name (f id) | Anonymous -> Anonymous let subst_cases_pattern_expr a loc subs = let rec subst = function | CPatAlias (_,p,x) -> CPatAlias (loc,subst p,x) (* No subst in compound pattern ? *) | CPatCstr (_,ref,pl) -> CPatCstr (loc,ref,List.map subst pl) | CPatAtom (_,Some (Ident (_,id))) -> subst_pat_id loc subs id | CPatAtom (_,x) -> CPatAtom (loc,x) | CPatNumeral (_,n) -> CPatNumeral (loc,n) | CPatDelimiters (_,key,p) -> CPatDelimiters (loc,key,subst p) in subst a let subst_constr_expr a loc subs = let rec subst = function | CRef (Ident (_,id)) -> (try List.assoc id subs with Not_found -> CRef (Ident (loc,id))) (* Temporary: no robust treatment of substituted binders *) | CLambdaN (_,[],c) -> subst c | CLambdaN (_,([],t)::bl,c) -> subst (CLambdaN (loc,bl,c)) | CLambdaN (_,((_,na)::bl,t)::bll,c) -> let na = name_app (subst_id subs) na in CLambdaN (loc,[[loc,na],subst t], subst (CLambdaN (loc,(bl,t)::bll,c))) | CProdN (_,[],c) -> subst c | CProdN (_,([],t)::bl,c) -> subst (CProdN (loc,bl,c)) | CProdN (_,((_,na)::bl,t)::bll,c) -> let na = name_app (subst_id subs) na in CProdN (loc,[[loc,na],subst t], subst (CProdN (loc,(bl,t)::bll,c))) | CLetIn (_,(_,na),b,c) -> let na = name_app (subst_id subs) na in CLetIn (loc,(loc,na),subst b,subst c) | CArrow (_,a,b) -> CArrow (loc,subst a,subst b) | CAppExpl (_,(p,Ident (_,id)),l) -> CAppExpl (loc,(p,subst_ref loc subs id),List.map subst l) | CAppExpl (_,r,l) -> CAppExpl (loc,r,List.map subst l) | CApp (_,(p,a),l) -> CApp (loc,(p,subst a),List.map (fun (a,i) -> (subst a,i)) l) | CCast (_,a,b) -> CCast (loc,subst a,subst b) | CNotation (_,n,l) -> CNotation (loc,n,List.map subst l) | CDelimiters (_,s,a) -> CDelimiters (loc,s,subst a) | CHole _ | CEvar _ | CPatVar _ | CSort _ | CNumeral _ | CDynamic _ | CRef _ as x -> x | CCases (_,(po,rtntypo),a,bl) -> (* TODO: apply g on the binding variables in pat... *) let bl = List.map (fun (_,pat,rhs) -> (loc,pat,subst rhs)) bl in CCases (loc,(option_app subst po,option_app subst rtntypo), List.map (fun (tm,x) -> subst tm,x) a,bl) | COrderedCase (_,s,po,a,bl) -> COrderedCase (loc,s,option_app subst po,subst a,List.map subst bl) | CLetTuple (_,nal,(na,po),a,b) -> let na = name_app (subst_id subs) na in let nal = List.map (name_app (subst_id subs)) nal in CLetTuple (loc,nal,(na,option_app subst po),subst a,subst b) | CIf (_,c,(na,po),b1,b2) -> let na = name_app (subst_id subs) na in CIf (loc,subst c,(na,option_app subst po),subst b1,subst b2) | CFix (_,id,dl) -> CFix (loc,id,List.map (fun (id,n,t,d) -> (id,n,subst t,subst d)) dl) | CCoFix (_,id,dl) -> CCoFix (loc,id,List.map (fun (id,t,d) -> (id,subst t,subst d)) dl) in subst a let make_rule univ assoc etyp rule = let pil = List.map (symbol_of_prod_item univ assoc etyp) rule.gr_production in let (symbs,ntl) = List.split pil in let act = match etyp with | ETPattern -> (* Ugly *) let f loc env = match rule.gr_action, env with | CRef (Ident(_,p)), [p',a] when p=p' -> a | CDelimiters (_,s,CRef (Ident(_,p))), [p',a] when p=p' -> CPatDelimiters (loc,s,a) | _ -> error "Unable to handle this grammar extension of pattern" in make_cases_pattern_act f ntl | ETIdent | ETBigint | ETReference -> error "Cannot extend" | ETConstr _ | ETOther _ -> make_act (subst_constr_expr rule.gr_action) ntl in (symbs, act) (* Rules of a level are entered in reverse order, so that the first rules are applied before the last ones *) let extend_entry univ (te, etyp, pos, name, ass, p4ass, rls) = let rules = List.rev (List.map (make_rule univ ass etyp) rls) in grammar_extend te pos [(name, p4ass, rules)] (* Defines new entries. If the entry already exists, check its type *) let define_entry univ {ge_name=typ; gl_assoc=ass; gl_rules=rls} = let e,lev,keepassoc = get_constr_entry typ in let pos,p4ass,name = find_position keepassoc ass lev in (e,typ,pos,name,ass,p4ass,rls) (* Add a bunch of grammar rules. Does not check if it is well formed *) let extend_grammar_rules gram = let univ = get_univ gram.gc_univ in let tl = List.map (define_entry univ) gram.gc_entries in List.iter (extend_entry univ) tl (* Add a grammar rules for tactics *) type grammar_tactic_production = | TacTerm of string | TacNonTerm of loc * (Gram.te Gramext.g_symbol * argument_type) * string option let make_prod_item = function | TacTerm s -> (Gramext.Stoken (Extend.terminal s), None) | TacNonTerm (_,(nont,t), po) -> (nont, option_app (fun p -> (p,t)) po) let make_gen_act f pil = let rec make env = function | [] -> Gramext.action (fun loc -> f loc env) | None :: tl -> (* parse a non-binding item *) Gramext.action (fun _ -> make env tl) | Some (p, t) :: tl -> (* non-terminal *) Gramext.action (fun v -> make ((p,in_generic t v) :: env) tl) in make [] (List.rev pil) let extend_constr entry (n,level,assoc,keepassoc) make_act pt = let univ = get_univ "constr" in let pil = List.map (symbol_of_prod_item univ assoc n) pt in let (symbs,ntl) = List.split pil in let act = make_act ntl in let pos,p4assoc,name = find_position keepassoc assoc level in grammar_extend entry pos [(name, p4assoc, [symbs, act])] let extend_constr_notation (n,assoc,ntn,rule) = let mkact loc env = CNotation (loc,ntn,List.map snd env) in let (e,level,keepassoc) = get_constr_entry (ETConstr (n,())) in extend_constr e (ETConstr(n,()),level,assoc,keepassoc) (make_act mkact) rule let extend_constr_delimiters (sc,rule,pat_rule) = let mkact loc env = CDelimiters (loc,sc,snd (List.hd env)) in extend_constr Constr.operconstr (ETConstr(0,()),Some 0,Some Gramext.NonA,false) (make_act mkact) rule; let mkact loc env = CPatDelimiters (loc,sc,snd (List.hd env)) in extend_constr Constr.pattern (ETPattern,None,None,false) (make_cases_pattern_act mkact) pat_rule (* These grammars are not a removable *) let make_rule univ f g (s,pt) = let hd = Gramext.Stoken ("IDENT", s) in let pil = (hd,None) :: List.map g pt in let (symbs,ntl) = List.split pil in let act = make_gen_act f ntl in (symbs, act) let tac_exts = ref [] let get_extend_tactic_grammars () = !tac_exts let extend_tactic_grammar s gl = tac_exts := (s,gl) :: !tac_exts; let univ = get_univ "tactic" in let make_act loc l = Tacexpr.TacExtend (loc,s,List.map snd l) in let rules = List.map (make_rule univ make_act make_prod_item) gl in Gram.extend Tactic.simple_tactic None [(None, None, List.rev rules)] let vernac_exts = ref [] let get_extend_vernac_grammars () = !vernac_exts let extend_vernac_command_grammar s gl = vernac_exts := (s,gl) :: !vernac_exts; let univ = get_univ "vernac" in let make_act loc l = Vernacexpr.VernacExtend (s,List.map snd l) in let rules = List.map (make_rule univ make_act make_prod_item) gl in Gram.extend Vernac_.command None [(None, None, List.rev rules)] let rec interp_entry_name u s = let l = String.length s in if l > 8 & String.sub s 0 3 = "ne_" & String.sub s (l-5) 5 = "_list" then let t, g = interp_entry_name u (String.sub s 3 (l-8)) in List1ArgType t, Gramext.Slist1 g else if l > 5 & String.sub s (l-5) 5 = "_list" then let t, g = interp_entry_name u (String.sub s 0 (l-5)) in List0ArgType t, Gramext.Slist0 g else if l > 4 & String.sub s (l-4) 4 = "_opt" then let t, g = interp_entry_name u (String.sub s 0 (l-4)) in OptArgType t, Gramext.Sopt g else let e = get_entry (get_univ u) s in let o = object_of_typed_entry e in let t = type_of_typed_entry e in t,Gramext.Snterm (Pcoq.Gram.Entry.obj o) let qualified_nterm current_univ = function | NtQual (univ, en) -> (univ, en) | NtShort en -> (current_univ, en) let make_vprod_item univ = function | VTerm s -> (Gramext.Stoken (Extend.terminal s), None) | VNonTerm (loc, nt, po) -> let (u,nt) = qualified_nterm univ nt in let (etyp, e) = interp_entry_name u nt in e, option_app (fun p -> (p,etyp)) po let add_tactic_entries gl = let univ = get_univ "tactic" in let make_act s tac loc l = Tacexpr.TacAlias (loc,s,l,tac) in let f (s,l,tac) = make_rule univ (make_act s tac) (make_vprod_item "tactic") l in let rules = List.map f gl in let _ = find_position true None None (* for synchronisation with remove *) in grammar_extend Tactic.simple_tactic None [(None, None, List.rev rules)] let extend_grammar gram = (match gram with | Notation a -> extend_constr_notation a | Delimiters a -> extend_constr_delimiters a | Grammar g -> extend_grammar_rules g | TacticGrammar l -> add_tactic_entries l); grammar_state := gram :: !grammar_state let reset_extend_grammars_v8 () = let te = List.rev !tac_exts in let tv = List.rev !vernac_exts in tac_exts := []; vernac_exts := []; List.iter (fun (s,gl) -> extend_tactic_grammar s gl) te; List.iter (fun (s,gl) -> extend_vernac_command_grammar s gl) tv; level_stack := [default_levels_v8] (* Summary functions: the state of the lexer is included in that of the parser. Because the grammar affects the set of keywords when adding or removing grammar rules. *) type frozen_t = all_grammar_command list * Lexer.frozen_t let freeze () = (!grammar_state, Lexer.freeze ()) (* We compare the current state of the grammar and the state to unfreeze, by computing the longest common suffixes *) let factorize_grams l1 l2 = if l1 == l2 then ([], [], l1) else list_share_tails l1 l2 let number_of_entries gcl = List.fold_left (fun n -> function | Notation _ -> n + 1 | Delimiters _ -> n + 2 (* One rule for constr, one for pattern *) | Grammar gc -> n + (List.length gc.gc_entries) | TacticGrammar l -> n + 1) 0 gcl let unfreeze (grams, lex) = let (undo, redo, common) = factorize_grams !grammar_state grams in let n = number_of_entries undo in remove_grammars n; remove_levels n; grammar_state := common; Lexer.unfreeze lex; List.iter extend_grammar (List.rev redo) let init_grammar () = remove_grammars (number_of_entries !grammar_state); grammar_state := [] let init () = init_grammar ()