ocaml/camlp4/lib/grammar.ml

975 lines
29 KiB
OCaml

(* camlp4r *)
(***********************************************************************)
(* *)
(* Camlp4 *)
(* *)
(* Daniel de Rauglaudre, projet Cristal, INRIA Rocquencourt *)
(* *)
(* Copyright 2001 Institut National de Recherche en Informatique et *)
(* Automatique. Distributed only by permission. *)
(* *)
(***********************************************************************)
(* $Id$ *)
open Stdpp;
open Gramext;
open Format;
value rec flatten_tree =
fun
[ DeadEnd -> []
| LocAct _ _ -> [[]]
| Node {node = n; brother = b; son = s} ->
List.map (fun l -> [n :: l]) (flatten_tree s) @ flatten_tree b ]
;
value print_str s = print_string ("\"" ^ String.escaped s ^ "\"");
value rec print_symbol =
fun
[ Slist0 s -> do { print_string "LIST0"; print_string " "; print_symbol1 s }
| Slist0sep s t ->
do {
print_string "LIST0";
print_string " ";
print_symbol1 s;
print_string " SEP ";
print_symbol1 t
}
| Slist1 s -> do { print_string "LIST1"; print_string " "; print_symbol1 s }
| Slist1sep s t ->
do {
print_string "LIST1";
print_string " ";
print_symbol1 s;
print_string " SEP ";
print_symbol1 t
}
| Sopt s -> do { print_string "OPT "; print_symbol1 s }
| Stoken (con, prm) when con <> "" && prm <> "" ->
do { print_string con; print_space (); print_str prm }
| Snterml e l ->
do {
print_string e.ename;
print_space ();
print_string "LEVEL";
print_space ();
print_str l
}
| s -> print_symbol1 s ]
and print_symbol1 =
fun
[ Stoken ("", s) -> print_str s
| Snterm e -> print_string e.ename
| Sself -> print_string "SELF"
| Snext -> print_string "NEXT"
| Stoken (con, "") -> print_string con
| Stree t -> print_level print_space (flatten_tree t)
| s -> do { print_string "("; print_symbol s; print_string ")" } ]
and print_rule symbols =
do {
open_hovbox 0;
let _ =
List.fold_left
(fun sep symbol ->
do {
sep ();
print_symbol symbol;
fun () -> do { print_string ";"; print_space () }
})
(fun () -> ()) symbols
in
close_box ()
}
and print_level print_space rules =
do {
open_hovbox 0;
print_string "[ ";
let _ =
List.fold_left
(fun sep rule ->
do {
sep ();
print_rule rule;
fun () -> do { print_space (); print_string "| " }
})
(fun () -> ()) rules
in
print_string " ]";
close_box ()
}
;
value print_levels elev =
let _ =
List.fold_left
(fun sep lev ->
let rules =
List.map (fun t -> [Sself :: t]) (flatten_tree lev.lsuffix) @
flatten_tree lev.lprefix
in
do {
sep ();
open_hovbox 2;
match lev.lname with
[ Some n ->
do {
print_string ("\"" ^ String.escaped n ^ "\"");
print_break 1 2
}
| _ -> () ];
match lev.assoc with
[ LeftA -> print_string "LEFTA"
| RightA -> print_string "RIGHTA"
| NonA -> print_string "NONA" ];
close_box ();
print_break 1 2;
print_level force_newline rules;
fun () -> do { print_cut (); print_string "| " }
})
(fun () -> ()) elev
in
()
;
value print_entry e =
do {
open_vbox 0;
print_string "[ ";
match e.edesc with
[ Dlevels elev -> print_levels elev
| Dparser _ -> print_string "<parser>" ];
print_string " ]";
close_box ();
print_newline ()
}
;
type g = Gramext.grammar;
external grammar_obj : g -> grammar = "%identity";
value floc = ref (fun _ -> failwith "internal error when computing location");
value loc_of_token_interval bp ep =
if bp == ep then
if bp == 0 then (0, 1)
else
let a = snd (floc.val (bp - 1)) in
(a, a + 1)
else
let (bp1, bp2) = floc.val bp in
let (ep1, ep2) = floc.val (pred ep) in
(if bp1 < ep1 then bp1 else ep1, if bp2 > ep2 then bp2 else ep2)
;
value rec name_of_symbol entry =
fun
[ Snterm e -> "[" ^ e.ename ^ "]"
| Snterml e l -> "[" ^ e.ename ^ " level " ^ l ^ "]"
| Sself | Snext -> "[" ^ entry.ename ^ "]"
| Stoken tok -> entry.egram.glexer.Token.text tok
| _ -> "???" ]
;
value rec get_token_list entry tokl last_tok tree =
match tree with
[ Node {node = (Stoken tok as s); son = son; brother = DeadEnd} ->
match entry.egram.glexer.Token.tparse tok with
[ Some _ ->
if tokl = [] then None
else Some (List.rev [last_tok :: tokl], last_tok, tree)
| None -> get_token_list entry [last_tok :: tokl] tok son ]
| _ ->
if tokl = [] then None
else Some (List.rev [last_tok :: tokl], last_tok, tree) ]
;
value rec name_of_symbol_failed entry =
fun
[ Slist0 s -> name_of_symbol_failed entry s
| Slist0sep s _ -> name_of_symbol_failed entry s
| Slist1 s -> name_of_symbol_failed entry s
| Slist1sep s _ -> name_of_symbol_failed entry s
| Sopt s -> name_of_symbol_failed entry s
| Stree t -> name_of_tree_failed entry t
| s -> name_of_symbol entry s ]
and name_of_tree_failed entry =
fun
[ Node {node = s; brother = bro; son = son} ->
let tokl =
match s with
[ Stoken tok when entry.egram.glexer.Token.tparse tok = None ->
get_token_list entry [] tok son
| _ -> None ]
in
match tokl with
[ None ->
let txt = name_of_symbol_failed entry s in
let txt =
match (s, son) with
[ (Sopt _, Node _) -> txt ^ " or " ^ name_of_tree_failed entry son
| _ -> txt ]
in
let txt =
match bro with
[ DeadEnd | LocAct _ _ -> txt
| _ -> txt ^ " or " ^ name_of_tree_failed entry bro ]
in
txt
| Some (tokl, last_tok, son) ->
List.fold_left
(fun s tok ->
(if s = "" then "" else s ^ " ") ^
entry.egram.glexer.Token.text tok)
"" tokl ]
| DeadEnd | LocAct _ _ -> "???" ]
;
value search_tree_in_entry prev_symb tree =
fun
[ Dlevels levels ->
let rec search_levels =
fun
[ [] -> tree
| [level :: levels] ->
match search_level level with
[ Some tree -> tree
| None -> search_levels levels ] ]
and search_level level =
match search_tree level.lsuffix with
[ Some t -> Some (Node {node = Sself; son = t; brother = DeadEnd})
| None -> search_tree level.lprefix ]
and search_tree t =
if tree <> DeadEnd && t == tree then Some t
else
match t with
[ Node n ->
match search_symbol n.node with
[ Some symb ->
Some (Node {node = symb; son = n.son; brother = DeadEnd})
| None ->
match search_tree n.son with
[ Some t ->
Some (Node {node = n.node; son = t; brother = DeadEnd})
| None -> search_tree n.brother ] ]
| _ -> None ]
and search_symbol symb =
match symb with
[ Snterm _ | Snterml _ _ | Slist0 _ | Slist0sep _ _ | Slist1 _ |
Slist1sep _ _ | Sopt _ | Stoken _ | Stree _
when symb == prev_symb ->
Some symb
| Slist0 symb ->
match search_symbol symb with
[ Some symb -> Some (Slist0 symb)
| None -> None ]
| Slist0sep symb sep ->
match search_symbol symb with
[ Some symb -> Some (Slist0sep symb sep)
| None ->
match search_symbol sep with
[ Some sep -> Some (Slist0sep symb sep)
| None -> None ] ]
| Slist1 symb ->
match search_symbol symb with
[ Some symb -> Some (Slist1 symb)
| None -> None ]
| Slist1sep symb sep ->
match search_symbol symb with
[ Some symb -> Some (Slist1sep symb sep)
| None ->
match search_symbol sep with
[ Some sep -> Some (Slist1sep symb sep)
| None -> None ] ]
| Sopt symb ->
match search_symbol symb with
[ Some symb -> Some (Sopt symb)
| None -> None ]
| Stree t ->
match search_tree t with
[ Some t -> Some (Stree t)
| None -> None ]
| _ -> None ]
in
search_levels levels
| _ -> tree ]
;
value error_verbose = ref False;
value tree_failed entry prev_symb_result prev_symb tree =
let txt = name_of_tree_failed entry tree in
let txt =
match prev_symb with
[ Slist0 s ->
let txt1 = name_of_symbol_failed entry s in
txt1 ^ " or " ^ txt ^ " expected"
| Slist1 s ->
let txt1 = name_of_symbol_failed entry s in
txt1 ^ " or " ^ txt ^ " expected"
| Slist0sep s sep ->
match Obj.magic prev_symb_result with
[ [] ->
let txt1 = name_of_symbol_failed entry s in
txt1 ^ " or " ^ txt ^ " expected"
| _ ->
let txt1 = name_of_symbol_failed entry sep in
txt1 ^ " or " ^ txt ^ " expected" ]
| Slist1sep s sep ->
match Obj.magic prev_symb_result with
[ [] ->
let txt1 = name_of_symbol_failed entry s in
txt1 ^ " or " ^ txt ^ " expected"
| _ ->
let txt1 = name_of_symbol_failed entry sep in
txt1 ^ " or " ^ txt ^ " expected" ]
| Sopt _ | Stree _ -> txt ^ " expected"
| _ -> txt ^ " expected after " ^ name_of_symbol entry prev_symb ]
in
do {
if error_verbose.val then do {
let tree = search_tree_in_entry prev_symb tree entry.edesc in
set_formatter_out_channel stderr;
open_vbox 0;
print_newline ();
print_string "----------------------------------";
print_newline ();
printf "Parse error in entry [%s], rule:" entry.ename;
print_break 0 2;
open_vbox 0;
print_level force_newline (flatten_tree tree);
close_box ();
print_newline ();
print_string "----------------------------------";
print_newline ();
close_box ();
print_newline ()
}
else ();
txt ^ " (in [" ^ entry.ename ^ "])"
}
;
value symb_failed entry prev_symb_result prev_symb symb =
let tree = Node {node = symb; brother = DeadEnd; son = DeadEnd} in
tree_failed entry prev_symb_result prev_symb tree
;
external app : Obj.t -> 'a = "%identity";
value is_level_labelled n lev =
match lev.lname with
[ Some n1 -> n = n1
| None -> False ]
;
value level_number entry lab =
let rec lookup levn =
fun
[ [] -> failwith ("unknown level " ^ lab)
| [lev :: levs] ->
if is_level_labelled lab lev then levn else lookup (succ levn) levs ]
in
match entry.edesc with
[ Dlevels elev -> lookup 0 elev
| Dparser _ -> raise Not_found ]
;
value rec top_symb entry =
fun
[ Sself | Snext -> Snterm entry
| Snterml e _ -> Snterm e
| Slist1sep s sep -> Slist1sep (top_symb entry s) sep
| _ -> raise Stream.Failure ]
;
value entry_of_symb entry =
fun
[ Sself | Snext -> entry
| Snterm e -> e
| Snterml e _ -> e
| _ -> raise Stream.Failure ]
;
value top_tree entry =
fun
[ Node {node = s; brother = bro; son = son} ->
Node {node = top_symb entry s; brother = bro; son = son}
| _ -> raise Stream.Failure ]
;
value skip_if_empty bp p strm =
if Stream.count strm == bp then Gramext.action (fun a -> p strm)
else raise Stream.Failure
;
value continue entry bp a s son p1 =
parser
[: a = (entry_of_symb entry s).econtinue 0 bp a;
act = p1 ? tree_failed entry a s son :] ->
Gramext.action (fun _ -> app act a)
;
value do_recover parser_of_tree entry nlevn alevn bp a s son =
parser
[ [: a = parser_of_tree entry nlevn alevn (top_tree entry son) :] -> a
| [: a = skip_if_empty bp (parser []) :] -> a
| [: a =
continue entry bp a s son
(parser_of_tree entry nlevn alevn son) :] ->
a ]
;
value strict_parsing = ref False;
value recover parser_of_tree entry nlevn alevn bp a s son strm =
if strict_parsing.val then raise (Stream.Error (tree_failed entry a s son))
else do_recover parser_of_tree entry nlevn alevn bp a s son strm
;
value std_token_parse =
fun
[ (p_con, "") -> parser [: `(con, prm) when con = p_con :] -> prm
| (p_con, p_prm) ->
parser [: `(con, prm) when con = p_con && prm = p_prm :] -> prm ]
;
value peek_nth n strm =
let list = Stream.npeek n strm in
let rec loop list n =
match (list, n) with
[ ([x :: _], 1) -> Some x
| ([_ :: l], n) -> loop l (n - 1)
| ([], _) -> None ]
in
loop list n
;
value rec parser_of_tree entry nlevn alevn =
fun
[ DeadEnd -> parser []
| LocAct act _ -> parser [: :] -> act
| Node {node = Sself; son = LocAct act _; brother = DeadEnd} ->
parser [: a = entry.estart alevn :] -> app act a
| Node {node = Sself; son = LocAct act _; brother = bro} ->
let p2 = parser_of_tree entry nlevn alevn bro in
parser
[ [: a = entry.estart alevn :] -> app act a
| [: a = p2 :] -> a ]
| Node {node = s; son = son; brother = DeadEnd} ->
let tokl =
match s with
[ Stoken tok when entry.egram.glexer.Token.tparse tok = None ->
get_token_list entry [] tok son
| _ -> None ]
in
match tokl with
[ None ->
let ps = parser_of_symbol entry nlevn s in
let p1 = parser_of_tree entry nlevn alevn son in
let p1 = parser_cont p1 entry nlevn alevn s son in
parser bp [: a = ps; act = p1 bp a :] -> app act a
| Some (tokl, last_tok, son) ->
let p1 = parser_of_tree entry nlevn alevn son in
let p1 = parser_cont p1 entry nlevn alevn (Stoken last_tok) son in
parser_of_token_list p1 tokl ]
| Node {node = s; son = son; brother = bro} ->
let tokl =
match s with
[ Stoken tok when entry.egram.glexer.Token.tparse tok = None ->
get_token_list entry [] tok son
| _ -> None ]
in
match tokl with
[ None ->
let ps = parser_of_symbol entry nlevn s in
let p1 = parser_of_tree entry nlevn alevn son in
let p1 = parser_cont p1 entry nlevn alevn s son in
let p2 = parser_of_tree entry nlevn alevn bro in
parser bp
[ [: a = ps; act = p1 bp a :] -> app act a
| [: a = p2 :] -> a ]
| Some (tokl, last_tok, son) ->
let p1 = parser_of_tree entry nlevn alevn son in
let p1 = parser_cont p1 entry nlevn alevn (Stoken last_tok) son in
let p1 = parser_of_token_list p1 tokl in
let p2 = parser_of_tree entry nlevn alevn bro in
parser
[ [: a = p1 :] -> a
| [: a = p2 :] -> a ] ] ]
and parser_cont p1 entry nlevn alevn s son bp a =
parser
[ [: a = p1 :] -> a
| [: a = recover parser_of_tree entry nlevn alevn bp a s son :] -> a
| [: :] -> raise (Stream.Error (tree_failed entry a s son)) ]
and parser_of_token_list p1 tokl =
loop 1 tokl where rec loop n =
fun
[ [(p_con, "")] ->
let ps strm =
match peek_nth n strm with
[ Some (con, prm) when p_con = "ANY" || con = p_con ->
do { for i = 1 to n do { Stream.junk strm }; Obj.repr prm }
| _ -> raise Stream.Failure ]
in
parser bp [: a = ps; act = p1 bp a :] -> app act a
| [(p_con, p_prm)] ->
let ps strm =
match peek_nth n strm with
[ Some (con, prm)
when (p_con = "ANY" || con = p_con) && prm = p_prm ->
do { for i = 1 to n do { Stream.junk strm }; Obj.repr prm }
| _ -> raise Stream.Failure ]
in
parser bp [: a = ps; act = p1 bp a :] -> app act a
| [(p_con, "") :: tokl] ->
let ps strm =
match peek_nth n strm with
[ Some (con, prm) when p_con = "ANY" || con = p_con -> prm
| _ -> raise Stream.Failure ]
in
let p1 = loop (n + 1) tokl in
parser
[: a = ps; s :] ->
let act = p1 s in
app act a
| [(p_con, p_prm) :: tokl] ->
let ps strm =
match peek_nth n strm with
[ Some (con, prm)
when (p_con = "ANY" || con = p_con) && prm = p_prm ->
prm
| _ -> raise Stream.Failure ]
in
let p1 = loop (n + 1) tokl in
parser
[: a = ps; s :] ->
let act = p1 s in
app act a
| [] -> invalid_arg "parser_of_token_list" ]
and parser_of_symbol entry nlevn =
fun
[ Slist0 s ->
let ps = parser_of_symbol entry nlevn s in
let rec loop al =
parser
[ [: a = ps; s :] -> loop [a :: al] s
| [: :] -> al ]
in
parser [: a = loop [] :] -> Obj.repr (List.rev a)
| Slist0sep symb sep ->
let ps = parser_of_symbol entry nlevn symb in
let pt = parser_of_symbol entry nlevn sep in
let rec kont al =
parser
[ [: v = pt; a = ps ? symb_failed entry v sep symb; s :] ->
kont [a :: al] s
| [: :] -> al ]
in
parser
[ [: a = ps; s :] -> Obj.repr (List.rev (kont [a] s))
| [: :] -> Obj.repr [] ]
| Slist1 s ->
let ps = parser_of_symbol entry nlevn s in
let rec loop al =
parser
[ [: a = ps; s :] -> loop [a :: al] s
| [: :] -> al ]
in
parser [: a = ps; s :] -> Obj.repr (List.rev (loop [a] s))
| Slist1sep symb sep ->
let ps = parser_of_symbol entry nlevn symb in
let pt = parser_of_symbol entry nlevn sep in
let rec kont al =
parser
[ [: v = pt;
a =
parser
[ [: a = ps :] -> a
| [: a =
parser_of_symbol entry nlevn (top_symb entry symb) :] ->
a
| [: :] ->
raise (Stream.Error (symb_failed entry v sep symb)) ];
s :] ->
kont [a :: al] s
| [: :] -> al ]
in
parser [: a = ps; s :] -> Obj.repr (List.rev (kont [a] s))
| Sopt s ->
let ps = parser_of_symbol entry nlevn s in
parser
[ [: a = ps :] -> Obj.repr (Some a)
| [: :] -> Obj.repr None ]
| Stree t ->
let pt = parser_of_tree entry 1 0 t in
parser bp
[: a = pt :] ep ->
let loc = loc_of_token_interval bp ep in
app a loc
| Snterm e -> parser [: a = e.estart 0 :] -> a
| Snterml e l -> parser [: a = e.estart (level_number e l) :] -> a
| Sself -> parser [: a = entry.estart 0 :] -> a
| Snext -> parser [: a = entry.estart nlevn :] -> a
| Stoken ("ANY", v) ->
if v = "" then parser [: `(_, x) :] -> Obj.repr x
else parser [: `(_, x) when x = v :] -> Obj.repr x
| Stoken tok ->
match entry.egram.glexer.Token.tparse tok with
[ Some f -> (Obj.magic f : Stream.t Token.t -> Obj.t)
| None ->
(Obj.magic (std_token_parse tok) : Stream.t Token.t -> Obj.t) ] ]
;
value rec continue_parser_of_levels entry clevn =
fun
[ [] -> fun levn bp a -> parser []
| [lev :: levs] ->
let p1 = continue_parser_of_levels entry (succ clevn) levs in
match lev.lsuffix with
[ DeadEnd -> p1
| tree ->
let alevn =
match lev.assoc with
[ LeftA | NonA -> succ clevn
| RightA -> clevn ]
in
let p2 = parser_of_tree entry (succ clevn) alevn tree in
fun levn bp a strm ->
if levn > clevn then p1 levn bp a strm
else
match strm with parser
[ [: a = p1 levn bp a :] -> a
| [: act = p2 :] ep ->
let a = app act a (loc_of_token_interval bp ep) in
entry.econtinue levn bp a strm ] ] ]
;
value rec start_parser_of_levels entry clevn =
fun
[ [] -> fun levn -> parser []
| [lev :: levs] ->
let p1 = start_parser_of_levels entry (succ clevn) levs in
match lev.lprefix with
[ DeadEnd -> p1
| tree ->
let alevn =
match lev.assoc with
[ LeftA | NonA -> succ clevn
| RightA -> clevn ]
in
let p2 = parser_of_tree entry (succ clevn) alevn tree in
match levs with
[ [] ->
fun levn strm ->
match strm with parser bp
[ [: act = p2 :] ep ->
let a = app act (loc_of_token_interval bp ep) in
entry.econtinue levn bp a strm ]
| _ ->
fun levn strm ->
if levn > clevn then p1 levn strm
else
match strm with parser bp
[ [: act = p2 :] ep ->
let a = app act (loc_of_token_interval bp ep) in
entry.econtinue levn bp a strm
| [: a = p1 levn :] -> a ] ] ] ]
;
value continue_parser_of_entry entry =
match entry.edesc with
[ Dlevels elev ->
let p = continue_parser_of_levels entry 0 elev in
fun levn bp a ->
parser
[ [: a = p levn bp a :] -> a
| [: :] -> a ]
| Dparser p -> fun levn bp a -> parser [] ]
;
value empty_entry ename levn strm =
raise (Stream.Error ("entry [" ^ ename ^ "] is empty"))
;
value start_parser_of_entry entry =
match entry.edesc with
[ Dlevels [] -> empty_entry entry.ename
| Dlevels elev -> start_parser_of_levels entry 0 elev
| Dparser p -> fun levn strm -> p strm ]
;
value parse_parsable entry efun (cs, (ts, fun_loc)) =
let restore =
let old_floc = floc.val in
fun () -> floc.val := old_floc
in
do {
floc.val := fun_loc;
try
let r = efun ts in
do { restore (); r }
with
[ Stream.Failure ->
let loc =
try fun_loc (Stream.count ts) with _ ->
(Stream.count cs, Stream.count cs + 1)
in
do {
restore ();
raise_with_loc loc
(Stream.Error ("illegal begin of " ^ entry.ename))
}
| Stream.Error _ as exc ->
let loc =
try fun_loc (Stream.count ts) with _ ->
(Stream.count cs, Stream.count cs + 1)
in
do { restore (); raise_with_loc loc exc }
| exc ->
let loc = (Stream.count cs, Stream.count cs + 1) in
do { restore (); raise_with_loc loc exc } ]
}
;
value wrap_parse entry efun cs =
let parsable = (cs, entry.egram.glexer.Token.func cs) in
parse_parsable entry efun parsable
;
value create_toktab () = Hashtbl.create 301;
value create lexer = {gtokens = create_toktab (); glexer = lexer};
(* Extend syntax *)
value extend_entry entry position rules =
try
let elev = Gramext.levels_of_rules entry position rules in
do {
entry.edesc := Dlevels elev;
entry.estart :=
fun lev strm ->
let f = start_parser_of_entry entry in
do { entry.estart := f; f lev strm };
entry.econtinue :=
fun lev bp a strm ->
let f = continue_parser_of_entry entry in
do { entry.econtinue := f; f lev bp a strm }
}
with
[ Token.Error s ->
do {
Printf.eprintf "Lexer initialization error:\n- %s\n" s;
flush stderr;
failwith "Grammar.extend"
} ]
;
value extend entry_rules_list =
let gram = ref None in
List.iter
(fun (entry, position, rules) ->
do {
match gram.val with
[ Some g ->
if g != entry.egram then do {
Printf.eprintf "Error: entries with different grammars\n";
flush stderr;
failwith "Grammar.extend"
}
else ()
| None -> gram.val := Some entry.egram ];
extend_entry entry position rules
})
entry_rules_list
;
(* Deleting a rule *)
value delete_rule entry sl =
match entry.edesc with
[ Dlevels levs ->
let levs = Gramext.delete_rule_in_level_list entry sl levs in
do {
entry.edesc := Dlevels levs;
entry.estart :=
fun lev strm ->
let f = start_parser_of_entry entry in
do { entry.estart := f; f lev strm };
entry.econtinue :=
fun lev bp a strm ->
let f = continue_parser_of_entry entry in
do { entry.econtinue := f; f lev bp a strm }
}
| _ -> () ]
;
(* Unsafe *)
value clear_entry e =
do {
e.estart := fun _ -> parser [];
e.econtinue := fun _ _ _ -> parser [];
match e.edesc with
[ Dlevels _ -> e.edesc := Dlevels []
| Dparser _ -> () ]
}
;
value reinit_gram g lexer = do { Hashtbl.clear g.gtokens; g.glexer := lexer };
module Unsafe =
struct value clear_entry = clear_entry; value reinit_gram = reinit_gram; end
;
exception EntryFound of g_entry;
value find_entry e s =
let rec find_levels levs =
try
do {
List.iter
(fun lev -> do { find_tree lev.lsuffix; find_tree lev.lprefix })
levs;
raise Not_found
}
with
[ EntryFound e -> e
| _ -> raise Not_found ]
and find_symbol =
fun
[ Snterm e -> if e.ename = s then raise (EntryFound e) else ()
| Snterml e _ -> if e.ename = s then raise (EntryFound e) else ()
| Slist0 s -> find_symbol s
| Slist0sep s _ -> find_symbol s
| Slist1 s -> find_symbol s
| Slist1sep s _ -> find_symbol s
| Sopt s -> find_symbol s
| Stree t -> find_tree t
| _ -> () ]
and find_tree =
fun
[ Node {node = s; brother = bro; son = son} ->
do { find_symbol s; find_tree bro; find_tree son }
| _ -> () ]
in
match e.edesc with
[ Dlevels levs -> find_levels levs
| Dparser _ -> raise Not_found ]
;
value of_entry e = e.egram;
module Entry =
struct
type e 'a = g_entry;
value create g n =
{egram = g; ename = n; estart = empty_entry n;
econtinue _ _ _ = parser []; edesc = Dlevels []}
;
value parse (entry : e 'a) cs : 'a =
Obj.magic (wrap_parse entry (entry.estart 0) cs)
;
value parse_token (entry : e 'a) ts : 'a = Obj.magic (entry.estart 0 ts);
value name e = e.ename;
value of_parser g n (p : Stream.t Token.t -> 'a) : e 'a =
{egram = g; ename = n; estart _ = Obj.magic p;
econtinue _ _ _ = parser []; edesc = Dparser (Obj.magic p)}
;
external obj : e 'a -> Gramext.g_entry = "%identity";
value print e = print_entry (obj e);
value find e = Obj.magic (find_entry (obj e));
end
;
value tokens g con =
let g = grammar_obj g in
let list = ref [] in
do {
Hashtbl.iter
(fun (p_con, p_prm) c ->
if p_con = con then list.val := [(p_prm, c.val) :: list.val] else ())
g.gtokens;
list.val
}
;
value warning_verbose = Gramext.warning_verbose;
(* Functorial interface *)
module type LexerType = sig value lexer : Token.lexer; end;
module type S =
sig
type parsable = 'x;
value parsable : Stream.t char -> parsable;
value tokens : string -> list (string * int);
module Entry :
sig
type e 'a = 'x;
value create : string -> e 'a;
value parse : e 'a -> parsable -> 'a;
value parse_token : e 'a -> Stream.t Token.t -> 'a;
value name : e 'a -> string;
value of_parser : string -> (Stream.t Token.t -> 'a) -> e 'a;
value print : e 'a -> unit;
external obj : e 'a -> Gramext.g_entry = "%identity";
end
;
module Unsafe :
sig
value reinit_gram : Token.lexer -> unit;
value clear_entry : Entry.e 'a -> unit;
end
;
value extend :
Entry.e 'a -> option Gramext.position ->
list
(option string * option Gramext.g_assoc *
list (list Gramext.g_symbol * Gramext.g_action)) ->
unit;
value delete_rule : Entry.e 'a -> list Gramext.g_symbol -> unit;
end
;
module Make (L : LexerType) : S =
struct
type parsable =
(Stream.t char * (Stream.t Token.t * Token.location_function))
;
value gram = create L.lexer;
value parsable cs = (cs, L.lexer.Token.func cs);
value tokens = tokens gram;
module Entry =
struct
type e 'a = g_entry;
value create n =
{egram = gram; ename = n; estart = empty_entry n;
econtinue _ _ _ = parser []; edesc = Dlevels []}
;
value parse (e : e 'a) p : 'a =
Obj.magic (parse_parsable e (e.estart 0) p)
;
value parse_token (e : e 'a) ts : 'a = Obj.magic (e.estart 0 ts);
value name e = e.ename;
value of_parser n (p : Stream.t Token.t -> 'a) : e 'a =
{egram = gram; ename = n; estart _ = Obj.magic p;
econtinue _ _ _ = parser []; edesc = Dparser (Obj.magic p)}
;
external obj : e 'a -> Gramext.g_entry = "%identity";
value print e = print_entry (obj e);
end
;
module Unsafe =
struct
value reinit_gram = Unsafe.reinit_gram gram;
value clear_entry = Unsafe.clear_entry;
end
;
value extend = extend_entry;
value delete_rule e r = delete_rule (Entry.obj e) r;
end
;