ocaml/camlp4/Camlp4Bin.ml

317 lines
12 KiB
OCaml

(* camlp4r *)
(****************************************************************************)
(* *)
(* 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
*)
(* $Id$ *)
open Camlp4;
open PreCast.Syntax;
open PreCast;
open Format;
module CleanAst = Camlp4.Struct.CleanAst.Make Ast;
module SSet = Set.Make String;
value pa_r = "Camlp4Parsers.OCamlr";
(* value pa_rr = "Camlp4Parsers.OCamlrr"; *)
value pa_o = "Camlp4Parsers.OCaml";
value pa_rp = "Camlp4Parsers.OCamlRevisedParser";
value pa_op = "Camlp4Parsers.OCamlParser";
value pa_g = "Camlp4Parsers.Grammar";
value pa_m = "Camlp4Parsers.Macro";
value pa_qb = "Camlp4Parsers.OCamlQuotationBase";
value pa_q = "Camlp4Parsers.OCamlQuotation";
value pa_rq = "Camlp4Parsers.OCamlRevisedQuotation";
value pa_oq = "Camlp4Parsers.OCamlOriginalQuotation";
value dyn_loader = ref (fun []);
value rcall_callback = ref (fun () -> ());
value loaded_modules = ref SSet.empty;
value add_to_loaded_modules name =
loaded_modules.val := SSet.add name loaded_modules.val;
value file_of_module_name n =
let s = String.copy n in
let rec self pos =
try do {
let pos = String.index_from s pos '.';
s.[pos] := '/';
} with [ Not_found -> () ]
in do { self 0; s ^ ".cmo" };
value rewrite_and_load n x =
let dyn_loader = dyn_loader.val () in
let find_in_path = DynLoader.find_in_path dyn_loader in
let real_load name = do {
add_to_loaded_modules name;
DynLoader.load dyn_loader name
} in
let load = List.iter (fun n ->
if SSet.mem n loaded_modules.val then ()
else do {
add_to_loaded_modules n;
DynLoader.load dyn_loader (file_of_module_name n);
}) in
do {
match (n, x) with
[ ("Parsers"|"", "pa_r.cmo" | "r" | "OCamlr") -> load [pa_r]
(* | ("Parsers"|"", "rr" | "OCamlrr") -> load [pa_r; pa_rr] *)
| ("Parsers"|"", "pa_o.cmo" | "o" | "OCaml") -> load [pa_r; pa_o]
| ("Parsers"|"", "pa_rp.cmo" | "rp" | "OCamlRevisedParser") -> load [pa_r; pa_o; pa_rp]
| ("Parsers"|"", "pa_op.cmo" | "op" | "OCamlParser") -> load [pa_r; pa_o; pa_rp; pa_op]
| ("Parsers"|"", "pa_extend.cmo" | "pa_extend_m.cmo" | "g" | "Grammar") -> load [pa_r; pa_g]
| ("Parsers"|"", "pa_macro.cmo" | "m" | "Macro") -> load [pa_r; pa_m]
| ("Parsers"|"", "q" | "OCamlQuotation") -> load [pa_r; pa_qb; pa_q]
| ("Parsers"|"", "q_MLast.cmo" | "rq" | "OCamlRevisedQuotation") -> load [pa_r; pa_qb; pa_rq]
| ("Parsers"|"", "oq" | "OCamlOriginalQuotation") -> load [pa_r; pa_o; pa_qb; pa_oq]
| ("Parsers"|"", "rf") -> load [pa_r; pa_rp; pa_qb; pa_q; pa_g; pa_m]
| ("Parsers"|"", "of") -> load [pa_r; pa_o; pa_rp; pa_op; pa_qb; pa_rq; pa_g; pa_m]
| ("Filters"|"", "l" | "Lift" | "lift") -> load ["Camlp4Filters.LiftCamlp4Ast"]
| ("Printers"|"", "pr_r.cmo" | "r" | "OCamlr" | "Camlp4Printers/OCamlr.cmo") ->
Camlp4.Printers.OCamlr.enable ()
(* | ("Printers"|"", "rr" | "OCamlrr" | "Camlp4Printers/OCamlrr.cmo") -> *)
(* Camlp4.Printers.OCamlrr.enable () *)
| ("Printers"|"", "pr_o.cmo" | "o" | "OCaml" | "Camlp4Printers/OCaml.cmo") ->
Camlp4.Printers.OCaml.enable ()
| ("Printers"|"", "pr_dump.cmo" | "p" | "DumpOCamlAst" | "Camlp4Printers/DumpOCamlAst.cmo") ->
Camlp4.Printers.DumpOCamlAst.enable ()
| ("Printers"|"", "d" | "DumpCamlp4Ast" | "Camlp4Printers/DumpCamlp4Ast.cmo") ->
Camlp4.Printers.DumpCamlp4Ast.enable ()
| ("Printers"|"", "a" | "Auto" | "Camlp4Printers/Auto.cmo") ->
load ["Camlp4Printers.Auto"]
| _ ->
let y = "Camlp4"^n^"/"^x^".cmo" in
real_load (try find_in_path y with [ Not_found -> x ]) ];
rcall_callback.val ();
};
value print_warning = eprintf "%a:\n%s@." Loc.print;
value rec parse_file dyn_loader name pa getdir =
let directive_handler = Some (fun ast ->
match getdir ast with
[ Some x ->
match x with
[ (_, "load", s) -> do { rewrite_and_load "" s; None }
| (_, "directory", s) -> do { DynLoader.include_dir dyn_loader s; None }
| (_, "use", s) -> Some (parse_file dyn_loader s pa getdir)
| (_, "default_quotation", s) -> do { Quotation.default.val := s; None }
| (loc, _, _) -> Loc.raise loc (Stream.Error "bad directive") ]
| None -> None ]) in
let loc = Loc.mk name
in do {
Warning.current.val := print_warning;
let ic = if name = "-" then stdin else open_in_bin name in
let cs = Stream.of_channel ic in
let clear () = if name = "-" then () else close_in ic in
let phr =
try pa ?directive_handler loc cs
with x -> do { clear (); raise x }
in
clear ();
phr
};
value output_file = ref None;
value process dyn_loader name pa pr clean fold_filters getdir =
let ast = parse_file dyn_loader name pa getdir in
let ast = fold_filters (fun t filter -> filter t) ast in
let ast = clean ast in
pr ?input_file:(Some name) ?output_file:output_file.val ast;
value gind =
fun
[ <:sig_item@loc< # $n$ $str:s$ >> -> Some (loc, n, s)
| _ -> None ];
value gimd =
fun
[ <:str_item@loc< # $n$ $str:s$ >> -> Some (loc, n, s)
| _ -> None ];
open Register;
value process_intf dyn_loader name =
process dyn_loader name CurrentParser.parse_interf CurrentPrinter.print_interf
(new CleanAst.clean_ast)#sig_item
AstFilters.fold_interf_filters gind;
value process_impl dyn_loader name =
process dyn_loader name CurrentParser.parse_implem CurrentPrinter.print_implem
(new CleanAst.clean_ast)#str_item
AstFilters.fold_implem_filters gimd;
value just_print_the_version () =
do { printf "%s@." Config.version; exit 0 };
value print_version () =
do { eprintf "Camlp4 version %s@." Config.version; exit 0 };
value print_stdlib () =
do { printf "%s@." Config.camlp4_standard_library; exit 0 };
value usage ini_sl ext_sl =
do {
eprintf "\
Usage: camlp4 [load-options] [--] [other-options]
Options:
<file>.ml Parse this implementation file
<file>.mli Parse this interface file
<file>.(cmo|cma) Load this module inside the Camlp4 core@.";
Options.print_usage_list ini_sl;
(* loop (ini_sl @ ext_sl) where rec loop =
fun
[ [(y, _, _) :: _] when y = "-help" -> ()
| [_ :: sl] -> loop sl
| [] -> eprintf " -help Display this list of options.@." ]; *)
if ext_sl <> [] then do {
eprintf "Options added by loaded object files:@.";
Options.print_usage_list ext_sl;
}
else ();
};
value warn_noassert () =
do {
eprintf "\
camlp4 warning: option -noassert is obsolete
You should give the -noassert option to the ocaml compiler instead.@.";
};
type file_kind =
[ Intf of string
| Impl of string
| Str of string
| ModuleImpl of string
| IncludeDir of string ];
value search_stdlib = ref True;
value print_loaded_modules = ref False;
value (task, do_task) =
let t = ref None in
let task f x =
let () = Config.current_input_file.val := x in
t.val := Some (if t.val = None then (fun _ -> f x)
else (fun usage -> usage ())) in
let do_task usage = match t.val with [ Some f -> f usage | None -> () ] in
(task, do_task);
value input_file x =
let dyn_loader = dyn_loader.val () in
do {
rcall_callback.val ();
match x with
[ Intf file_name -> task (process_intf dyn_loader) file_name
| Impl file_name -> task (process_impl dyn_loader) file_name
| Str s -> do {
let (f, o) = Filename.open_temp_file "from_string" ".ml";
output_string o s;
close_out o;
task (process_impl dyn_loader) f;
}
| ModuleImpl file_name -> rewrite_and_load "" file_name
| IncludeDir dir -> DynLoader.include_dir dyn_loader dir ];
rcall_callback.val ();
};
value initial_spec_list =
[("-I", Arg.String (fun x -> input_file (IncludeDir x)),
"<directory> Add directory in search patch for object files.");
("-where", Arg.Unit print_stdlib,
"Print camlp4 library directory and exit.");
("-nolib", Arg.Clear search_stdlib,
"No automatic search for object files in library directory.");
("-intf", Arg.String (fun x -> input_file (Intf x)),
"<file> Parse <file> as an interface, whatever its extension.");
("-impl", Arg.String (fun x -> input_file (Impl x)),
"<file> Parse <file> as an implementation, whatever its extension.");
("-str", Arg.String (fun x -> input_file (Str x)),
"<string> Parse <string> as an implementation.");
("-unsafe", Arg.Set Config.unsafe,
"Generate unsafe accesses to array and strings.");
("-noassert", Arg.Unit warn_noassert,
"Obsolete, do not use this option.");
("-verbose", Arg.Set Config.verbose,
"More verbose in parsing errors.");
("-loc", Arg.Set_string Loc.name,
"<name> Name of the location variable (default: " ^ Loc.name.val ^ ").");
("-QD", Arg.String (fun x -> Quotation.dump_file.val := Some x),
"<file> Dump quotation expander result in case of syntax error.");
("-o", Arg.String (fun x -> output_file.val := Some x),
"<file> Output on <file> instead of standard output.");
("-v", Arg.Unit print_version,
"Print Camlp4 version and exit.");
("-version", Arg.Unit just_print_the_version,
"Print Camlp4 version number and exit.");
("-no_quot", Arg.Clear Config.quotations,
"Don't parse quotations, allowing to use, e.g. \"<:>\" as token.");
("-loaded-modules", Arg.Set print_loaded_modules, "Print the list of loaded modules.");
("-parser", Arg.String (rewrite_and_load "Parsers"),
"<name> Load the parser Camlp4Parsers/<name>.cmo");
("-printer", Arg.String (rewrite_and_load "Printers"),
"<name> Load the printer Camlp4Printers/<name>.cmo");
("-filter", Arg.String (rewrite_and_load "Filters"),
"<name> Load the filter Camlp4Filters/<name>.cmo");
("-ignore", Arg.String ignore, "ignore the next argument");
("--", Arg.Unit ignore, "Deprecated, does nothing")
];
Options.init initial_spec_list;
value anon_fun name =
input_file
(if Filename.check_suffix name ".mli" then Intf name
else if Filename.check_suffix name ".ml" then Impl name
else if Filename.check_suffix name ".cmo" then ModuleImpl name
else if Filename.check_suffix name ".cma" then ModuleImpl name
else raise (Arg.Bad ("don't know what to do with " ^ name)));
value main argv =
let usage () = do { usage initial_spec_list (Options.ext_spec_list ()); exit 0 } in
try do {
let dynloader = DynLoader.mk ~ocaml_stdlib:search_stdlib.val
~camlp4_stdlib:search_stdlib.val ();
dyn_loader.val := fun () -> dynloader;
let call_callback () =
Register.iter_and_take_callbacks
(fun (name, module_callback) ->
let () = add_to_loaded_modules name in
module_callback ());
call_callback ();
rcall_callback.val := call_callback;
match Options.parse anon_fun argv with
[ [] -> ()
| ["-help"|"--help"|"-h"|"-?" :: _] -> usage ()
| [s :: _] ->
do { eprintf "%s: unknown or misused option\n" s;
eprintf "Use option -help for usage@.";
exit 2 } ];
do_task usage;
call_callback ();
if print_loaded_modules.val then do {
SSet.iter (eprintf "%s@.") loaded_modules.val;
} else ()
}
with
[ Arg.Bad s -> do { eprintf "Error: %s\n" s;
eprintf "Use option -help for usage@.";
exit 2 }
| Arg.Help _ -> usage ()
| exc -> do { eprintf "@[<v0>%a@]@." ErrorHandler.print exc; exit 2 } ];
main Sys.argv;