ocaml/ocamldoc/odoc_analyse.ml

455 lines
15 KiB
OCaml

(***********************************************************************)
(* OCamldoc *)
(* *)
(* Maxence Guesdon, projet Cristal, INRIA Rocquencourt *)
(* *)
(* Copyright 2001 Institut National de Recherche en Informatique et *)
(* en Automatique. All rights reserved. This file is distributed *)
(* under the terms of the Q Public License version 1.0. *)
(* *)
(***********************************************************************)
(** Analysis of source files. This module is strongly inspired from driver/main.ml :-) *)
let print_DEBUG s = print_string s ; print_newline ()
open Config
open Clflags
open Misc
open Format
open Typedtree
(** Initialize the search path.
The current directory is always searched first,
then the directories specified with the -I option (in command-line order),
then the standard library directory. *)
let init_path () =
let dirs =
if !Clflags.thread_safe then
Filename.concat Config.standard_library "threads" :: !Clflags.include_dirs
else
!Clflags.include_dirs in
load_path := "" :: List.rev (Config.standard_library :: dirs);
Env.reset_cache()
(** Return the initial environment in which compilation proceeds. *)
let initial_env () =
try
if !Clflags.nopervasives
then Env.initial
else Env.open_pers_signature "Pervasives" Env.initial
with Not_found ->
fatal_error "cannot open pervasives.cmi"
(** Optionally preprocess a source file *)
let preprocess sourcefile =
match !Clflags.preprocessor with
None -> sourcefile
| Some pp ->
let tmpfile = Filename.temp_file "camlpp" "" in
let comm = Printf.sprintf "%s %s > %s" pp sourcefile tmpfile in
if Ccomp.command comm <> 0 then begin
remove_file tmpfile;
Printf.eprintf "Preprocessing error\n";
exit 2
end;
tmpfile
(** Remove the input file if this file was the result of a preprocessing.*)
let remove_preprocessed inputfile =
match !Clflags.preprocessor with
None -> ()
| Some _ -> remove_file inputfile
let remove_preprocessed_if_ast inputfile =
match !Clflags.preprocessor with
None -> ()
| Some _ -> if inputfile <> !Location.input_name then remove_file inputfile
exception Outdated_version
(** Parse a file or get a dumped syntax tree in it *)
let parse_file inputfile parse_fun ast_magic =
let ic = open_in_bin inputfile in
let is_ast_file =
try
let buffer = String.create (String.length ast_magic) in
really_input ic buffer 0 (String.length ast_magic);
if buffer = ast_magic then true
else if String.sub buffer 0 9 = String.sub ast_magic 0 9 then
raise Outdated_version
else false
with
Outdated_version ->
fatal_error "Ocaml and preprocessor have incompatible versions"
| _ -> false
in
let ast =
try
if is_ast_file then begin
Location.input_name := input_value ic;
input_value ic
end else begin
seek_in ic 0;
Location.input_name := inputfile;
parse_fun (Lexing.from_channel ic)
end
with x -> close_in ic; raise x
in
close_in ic;
ast
let (++) x f = f x
(** Analysis of an implementation file. Returns (Some typedtree) if
no error occured, else None and an error message is printed.*)
let process_implementation_file ppf sourcefile =
init_path();
let prefixname = Filename.chop_extension sourcefile in
let modulename = String.capitalize(Filename.basename prefixname) in
let inputfile = preprocess sourcefile in
let env = initial_env () in
let cmi_file = (Filename.chop_extension sourcefile)^".cmi" in
try
let _ = Sys.command ("cp "^cmi_file^" /tmp") in
let parsetree = parse_file inputfile Parse.implementation ast_impl_magic_number in
let typedtree = Typemod.type_implementation sourcefile prefixname modulename env parsetree in
let _ = Sys.command ("mv "^"/tmp/"^(Filename.basename cmi_file)^" "^cmi_file) in
(Some (parsetree, typedtree), inputfile)
with
e ->
let _ = Sys.command ("mv "^"/tmp/"^(Filename.basename cmi_file)^" "^cmi_file) in
match e with
Syntaxerr.Error err ->
fprintf Format.err_formatter "@[%a@]@."
Syntaxerr.report_error err;
None, inputfile
| Failure s ->
prerr_endline s;
incr Odoc_global.errors ;
None, inputfile
| e ->
raise e
(** Analysis of an interface file. Returns (Some signature) if
no error occured, else None and an error message is printed.*)
let process_interface_file ppf sourcefile =
init_path();
let prefixname = Filename.chop_extension sourcefile in
let modulename = String.capitalize(Filename.basename prefixname) in
let inputfile = preprocess sourcefile in
let ast = parse_file inputfile Parse.interface ast_intf_magic_number in
let sg = Typemod.transl_signature (initial_env()) ast in
Warnings.check_fatal ();
(ast, sg, inputfile)
(** The module used to analyse the parsetree and signature of an implementation file.*)
module Ast_analyser = Odoc_ast.Analyser (Odoc_comments.Basic_info_retriever)
(** The module used to analyse the parse tree and typed tree of an interface file.*)
module Sig_analyser = Odoc_sig.Analyser (Odoc_comments.Basic_info_retriever)
(** Handle an error. This is a partial copy of the compiler
driver/error.ml file. We do this because there are
some differences between the possibly raised exceptions
in the bytecode (error.ml) and opt (opterros.ml) compilers
and we don't want to take care of this. Besisdes, this
differences only concern code generation (i believe).*)
let process_error exn =
let report ppf = function
| Lexer.Error(err, start, stop) ->
Location.print ppf {Location.loc_start = start;
Location.loc_end = stop;
Location.loc_ghost = false};
Lexer.report_error ppf err
| Syntaxerr.Error err ->
Syntaxerr.report_error ppf err
| Env.Error err ->
Env.report_error ppf err
| Ctype.Tags(l, l') -> fprintf ppf
"In this program,@ variant constructors@ `%s and `%s@ \
have the same hash value." l l'
| Typecore.Error(loc, err) ->
Location.print ppf loc; Typecore.report_error ppf err
| Typetexp.Error(loc, err) ->
Location.print ppf loc; Typetexp.report_error ppf err
| Typedecl.Error(loc, err) ->
Location.print ppf loc; Typedecl.report_error ppf err
| Includemod.Error err ->
Includemod.report_error ppf err
| Typemod.Error(loc, err) ->
Location.print ppf loc; Typemod.report_error ppf err
| Translcore.Error(loc, err) ->
Location.print ppf loc; Translcore.report_error ppf err
| Sys_error msg ->
fprintf ppf "I/O error: %s" msg
| Typeclass.Error(loc, err) ->
Location.print ppf loc; Typeclass.report_error ppf err
| Translclass.Error(loc, err) ->
Location.print ppf loc; Translclass.report_error ppf err
| Warnings.Errors (n) ->
fprintf ppf "@.Error: %d error-enabled warnings occurred." n
| x ->
fprintf ppf "@]";
fprintf ppf "Compilation error. Use the OCaml compiler to get more details."
in
Format.fprintf Format.err_formatter "@[%a@]@." report exn
(** Process the given file, according to its extension. Return the Module.t created, if any.*)
let process_file ppf sourcefile =
if !Odoc_args.verbose then
(
print_string (Odoc_messages.analysing sourcefile) ;
print_newline ();
);
if Filename.check_suffix sourcefile "ml" then
(
try
let (parsetree_typedtree_opt, input_file) = process_implementation_file ppf sourcefile in
match parsetree_typedtree_opt with
None ->
None
| Some (parsetree, typedtree) ->
let file_module = Ast_analyser.analyse_typed_tree sourcefile !Location.input_name parsetree typedtree in
file_module.Odoc_module.m_top_deps <- Odoc_dep.impl_dependencies parsetree ;
if !Odoc_args.verbose then
(
print_string Odoc_messages.ok;
print_newline ()
);
remove_preprocessed input_file;
Some file_module
with
| Sys_error s
| Failure s ->
prerr_endline s ;
incr Odoc_global.errors ;
None
| e ->
process_error e ;
incr Odoc_global.errors ;
None
)
else
if Filename.check_suffix sourcefile "mli" then
(
try
let (ast, signat, input_file) = process_interface_file ppf sourcefile in
let file_module = Sig_analyser.analyse_signature sourcefile !Location.input_name ast signat in
file_module.Odoc_module.m_top_deps <- Odoc_dep.intf_dependencies ast ;
if !Odoc_args.verbose then
(
print_string Odoc_messages.ok;
print_newline ()
);
remove_preprocessed input_file;
Some file_module
with
| Sys_error s
| Failure s ->
prerr_endline s;
incr Odoc_global.errors ;
None
| e ->
process_error e ;
incr Odoc_global.errors ;
None
)
else
(
raise (Failure (Odoc_messages.unknown_extension sourcefile))
)
(** Remove the class elements after the stop special comment. *)
let rec remove_class_elements_after_stop eles =
match eles with
[] -> []
| ele :: q ->
match ele with
Odoc_class.Class_comment [ Odoc_types.Raw "/*" ] -> []
| Odoc_class.Class_attribute _
| Odoc_class.Class_method _
| Odoc_class.Class_comment _ -> ele :: (remove_class_elements_after_stop q)
(** Remove the class elements after the stop special comment in a class kind. *)
let rec remove_class_elements_after_stop_in_class_kind k =
match k with
Odoc_class.Class_structure (inher, l) ->
Odoc_class.Class_structure (inher, remove_class_elements_after_stop l)
| Odoc_class.Class_apply _ -> k
| Odoc_class.Class_constr _ -> k
| Odoc_class.Class_constraint (k1, ctk) ->
Odoc_class.Class_constraint (remove_class_elements_after_stop_in_class_kind k1,
remove_class_elements_after_stop_in_class_type_kind ctk)
(** Remove the class elements after the stop special comment in a class type kind. *)
and remove_class_elements_after_stop_in_class_type_kind tk =
match tk with
Odoc_class.Class_signature (inher, l) ->
Odoc_class.Class_signature (inher, remove_class_elements_after_stop l)
| Odoc_class.Class_type _ -> tk
(** Remove the module elements after the stop special comment. *)
let rec remove_module_elements_after_stop eles =
let f = remove_module_elements_after_stop in
match eles with
[] -> []
| ele :: q ->
match ele with
Odoc_module.Element_module_comment [ Odoc_types.Raw "/*" ] -> []
| Odoc_module.Element_module_comment _ ->
ele :: (f q)
| Odoc_module.Element_module m ->
m.Odoc_module.m_kind <- remove_module_elements_after_stop_in_module_kind m.Odoc_module.m_kind ;
(Odoc_module.Element_module m) :: (f q)
| Odoc_module.Element_module_type mt ->
mt.Odoc_module.mt_kind <- Odoc_misc.apply_opt
remove_module_elements_after_stop_in_module_type_kind mt.Odoc_module.mt_kind ;
(Odoc_module.Element_module_type mt) :: (f q)
| Odoc_module.Element_included_module _ ->
ele :: (f q)
| Odoc_module.Element_class c ->
c.Odoc_class.cl_kind <- remove_class_elements_after_stop_in_class_kind c.Odoc_class.cl_kind ;
(Odoc_module.Element_class c) :: (f q)
| Odoc_module.Element_class_type ct ->
ct.Odoc_class.clt_kind <- remove_class_elements_after_stop_in_class_type_kind ct.Odoc_class.clt_kind ;
(Odoc_module.Element_class_type ct) :: (f q)
| Odoc_module.Element_value _
| Odoc_module.Element_exception _
| Odoc_module.Element_type _ ->
ele :: (f q)
(** Remove the module elements after the stop special comment, in the given module kind. *)
and remove_module_elements_after_stop_in_module_kind k =
match k with
| Odoc_module.Module_struct l -> Odoc_module.Module_struct (remove_module_elements_after_stop l)
| Odoc_module.Module_alias _ -> k
| Odoc_module.Module_functor (params, k2) ->
Odoc_module.Module_functor (params, remove_module_elements_after_stop_in_module_kind k2)
| Odoc_module.Module_apply (k1, k2) ->
Odoc_module.Module_apply (remove_module_elements_after_stop_in_module_kind k1,
remove_module_elements_after_stop_in_module_kind k2)
| Odoc_module.Module_with (mtkind, s) ->
Odoc_module.Module_with (remove_module_elements_after_stop_in_module_type_kind mtkind, s)
| Odoc_module.Module_constraint (k2, mtkind) ->
Odoc_module.Module_constraint (remove_module_elements_after_stop_in_module_kind k2,
remove_module_elements_after_stop_in_module_type_kind mtkind)
(** Remove the module elements after the stop special comment, in the given module type kind. *)
and remove_module_elements_after_stop_in_module_type_kind tk =
match tk with
| Odoc_module.Module_type_struct l -> Odoc_module.Module_type_struct (remove_module_elements_after_stop l)
| Odoc_module.Module_type_functor (params, tk2) ->
Odoc_module.Module_type_functor (params, remove_module_elements_after_stop_in_module_type_kind tk2)
| Odoc_module.Module_type_alias _ -> tk
| Odoc_module.Module_type_with (tk2, s) ->
Odoc_module.Module_type_with (remove_module_elements_after_stop_in_module_type_kind tk2, s)
(** Remove elements after the stop special comment. *)
let remove_elements_after_stop module_list =
List.map
(fun m ->
m.Odoc_module.m_kind <- remove_module_elements_after_stop_in_module_kind m.Odoc_module.m_kind;
m
)
module_list
(** This function builds the modules from the given list of source files. *)
let analyse_files ?(init=[]) files =
let modules_pre =
init @
(List.fold_left
(fun acc -> fun file ->
try
match process_file Format.err_formatter file with
None ->
acc
| Some m ->
acc @ [ m ]
with
Failure s ->
prerr_endline s ;
incr Odoc_global.errors ;
acc
)
[]
files
)
in
(* Remove elements after the stop special comments, if needed. *)
let modules =
if !Odoc_args.no_stop then
modules_pre
else
remove_elements_after_stop modules_pre
in
if !Odoc_args.verbose then
(
print_string Odoc_messages.merging;
print_newline ()
);
let merged_modules = Odoc_merge.merge !Odoc_args.merge_options modules in
if !Odoc_args.verbose then
(
print_string Odoc_messages.ok;
print_newline ();
);
let modules_list =
(List.fold_left
(fun acc -> fun m -> acc @ (Odoc_module.module_all_submodules ~trans: false m))
merged_modules
merged_modules
)
in
if !Odoc_args.verbose then
(
print_string Odoc_messages.cross_referencing;
print_newline ()
);
let _ = Odoc_cross.associate modules_list in
if !Odoc_args.verbose then
(
print_string Odoc_messages.ok;
print_newline ();
);
if !Odoc_args.sort_modules then
Sort.list (fun m1 -> fun m2 -> m1.Odoc_module.m_name < m2.Odoc_module.m_name) merged_modules
else
merged_modules
let dump_modules file (modules : Odoc_module.t_module list) =
try
let chanout = open_out_bin file in
let dump = Odoc_types.make_dump modules in
output_value chanout dump;
close_out chanout
with
Sys_error s ->
raise (Failure s)
let load_modules file =
try
let chanin = open_in_bin file in
let dump = input_value chanin in
close_in chanin ;
let (l : Odoc_module.t_module list) = Odoc_types.open_dump dump in
l
with
Sys_error s ->
raise (Failure s)