ocaml/camlp4/Camlp4/Options.ml

192 lines
6.4 KiB
OCaml

(****************************************************************************)
(* *)
(* Objective Caml *)
(* *)
(* INRIA Rocquencourt *)
(* *)
(* Copyright 2006 Institut National de Recherche en Informatique et *)
(* en Automatique. All rights reserved. This file is distributed under *)
(* the terms of the GNU Library General Public License, with the special *)
(* exception on linking described in LICENSE at the top of the Objective *)
(* Caml source tree. *)
(* *)
(****************************************************************************)
(* Authors:
* - Daniel de Rauglaudre: initial version
* - Nicolas Pouillard: refactoring
*)
type spec_list = list (string * Arg.spec * string);
open Format;
value rec action_arg s sl =
fun
[ Arg.Unit f -> if s = "" then do { f (); Some sl } else None
| Arg.Bool f ->
if s = "" then
match sl with
[ [s :: sl] ->
try do { f (bool_of_string s); Some sl } with
[ Invalid_argument "bool_of_string" -> None ]
| [] -> None ]
else
try do { f (bool_of_string s); Some sl } with
[ Invalid_argument "bool_of_string" -> None ]
| Arg.Set r -> if s = "" then do { r.val := True; Some sl } else None
| Arg.Clear r -> if s = "" then do { r.val := False; Some sl } else None
| Arg.Rest f -> do { List.iter f [s :: sl]; Some [] }
| Arg.String f ->
if s = "" then
match sl with
[ [s :: sl] -> do { f s; Some sl }
| [] -> None ]
else do { f s; Some sl }
| Arg.Set_string r ->
if s = "" then
match sl with
[ [s :: sl] -> do { r.val := s; Some sl }
| [] -> None ]
else do { r.val := s; Some sl }
| Arg.Int f ->
if s = "" then
match sl with
[ [s :: sl] ->
try do { f (int_of_string s); Some sl } with
[ Failure "int_of_string" -> None ]
| [] -> None ]
else
try do { f (int_of_string s); Some sl } with
[ Failure "int_of_string" -> None ]
| Arg.Set_int r ->
if s = "" then
match sl with
[ [s :: sl] ->
try do { r.val := (int_of_string s); Some sl } with
[ Failure "int_of_string" -> None ]
| [] -> None ]
else
try do { r.val := (int_of_string s); Some sl } with
[ Failure "int_of_string" -> None ]
| Arg.Float f ->
if s = "" then
match sl with
[ [s :: sl] -> do { f (float_of_string s); Some sl }
| [] -> None ]
else do { f (float_of_string s); Some sl }
| Arg.Set_float r ->
if s = "" then
match sl with
[ [s :: sl] -> do { r.val := (float_of_string s); Some sl }
| [] -> None ]
else do { r.val := (float_of_string s); Some sl }
| Arg.Tuple specs ->
let rec action_args s sl =
fun
[ [] -> Some sl
| [spec :: spec_list] ->
match action_arg s sl spec with
[ None -> action_args "" [] spec_list
| Some [s :: sl] -> action_args s sl spec_list
| Some sl -> action_args "" sl spec_list
]
] in
action_args s sl specs
| Arg.Symbol syms f ->
match (if s = "" then sl else [s :: sl]) with
[ [s :: sl] when List.mem s syms -> do { f s; Some sl }
| _ -> None ]
];
value common_start s1 s2 =
loop 0 where rec loop i =
if i == String.length s1 || i == String.length s2 then i
else if s1.[i] == s2.[i] then loop (i + 1)
else i;
value parse_arg fold s sl =
fold
(fun (name, action, _) acu ->
let i = common_start s name in
if i == String.length name then
try action_arg (String.sub s i (String.length s - i)) sl action with
[ Arg.Bad _ -> acu ]
else acu) None;
value rec parse_aux fold anon_fun =
fun
[ [] -> []
| [s :: sl] ->
if String.length s > 1 && s.[0] = '-' then
match parse_arg fold s sl with
[ Some sl -> parse_aux fold anon_fun sl
| None -> [s :: parse_aux fold anon_fun sl] ]
else do { (anon_fun s : unit); parse_aux fold anon_fun sl } ];
value align_doc key s =
let s =
loop 0 where rec loop i =
if i = String.length s then ""
else if s.[i] = ' ' then loop (i + 1)
else String.sub s i (String.length s - i)
in
let (p, s) =
if String.length s > 0 then
if s.[0] = '<' then
loop 0 where rec loop i =
if i = String.length s then ("", s)
else if s.[i] <> '>' then loop (i + 1)
else
let p = String.sub s 0 (i + 1) in
loop (i + 1) where rec loop i =
if i >= String.length s then (p, "")
else if s.[i] = ' ' then loop (i + 1)
else (p, String.sub s i (String.length s - i))
else ("", s)
else ("", "")
in
let tab =
String.make (max 1 (16 - String.length key - String.length p)) ' '
in
p ^ tab ^ s;
value make_symlist l =
match l with
[ [] -> "<none>"
| [h::t] -> (List.fold_left (fun x y -> x ^ "|" ^ y) ("{" ^ h) t) ^ "}" ];
value print_usage_list l =
List.iter
(fun (key, spec, doc) ->
match spec with
[ Arg.Symbol symbs _ ->
let s = make_symlist symbs in
let synt = key ^ " " ^ s in
eprintf " %s %s\n" synt (align_doc synt doc)
| _ -> eprintf " %s %s\n" key (align_doc key doc) ] )
l;
value remaining_args argv =
let rec loop l i =
if i == Array.length argv then l else loop [argv.(i) :: l] (i + 1)
in
List.rev (loop [] (Arg.current.val + 1));
value init_spec_list = ref [];
value ext_spec_list = ref [];
value init spec_list = init_spec_list.val := spec_list;
value add name spec descr =
ext_spec_list.val := [(name, spec, descr) :: ext_spec_list.val];
value fold f init =
let spec_list = init_spec_list.val @ ext_spec_list.val in
let specs = Sort.list (fun (k1, _, _) (k2, _, _) -> k1 >= k2) spec_list in
List.fold_right f specs init;
value parse anon_fun argv =
let remaining_args = remaining_args argv in
parse_aux fold anon_fun remaining_args;
value ext_spec_list () = ext_spec_list.val;