You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

309 lines
11 KiB

(*
* Copyright (c) 2009 - 2013 Monoidics ltd.
* Copyright (c) 2013 - present Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*)
open! IStd
open! PVariant
open Javalib_pack
module L = Logging
(** version of Javalib.get_class that does not spam stderr *)
let javalib_get_class = Utils.suppress_stderr2 Javalib.get_class
let models_specs_filenames = ref String.Set.empty
let models_jar = ref ""
let models_tenv = ref (Tenv.create ())
let load_models_tenv zip_channel =
let models_tenv_filename_in_jar =
let root = Filename.concat Config.default_in_zip_results_dir Config.captured_dir_name in
Filename.concat root Config.global_tenv_filename
in
let temp_tenv_filename =
DB.filename_from_string (Filename.temp_file "tmp_" Config.global_tenv_filename)
in
let entry = Zip.find_entry zip_channel models_tenv_filename_in_jar in
let temp_tenv_file = DB.filename_to_string temp_tenv_filename in
let models_tenv =
try
Zip.copy_entry_to_file zip_channel entry temp_tenv_file ;
match Tenv.load_from_file temp_tenv_filename with
| None ->
L.(die InternalError) "Models tenv file could not be loaded"
| Some tenv ->
tenv
with
| Not_found ->
L.(die InternalError) "Models tenv not found in jar file"
| Sys_error msg ->
L.(die InternalError) "Models jar could not be opened: %s" msg
in
DB.file_remove temp_tenv_filename ; models_tenv
let collect_specs_filenames jar_filename =
let zip_channel = Zip.open_in jar_filename in
let collect set e =
let filename = e.Zip.filename in
if not (Filename.check_suffix filename Config.specs_files_suffix) then set
else
let proc_filename = Filename.chop_extension (Filename.basename filename) in
String.Set.add set proc_filename
in
models_specs_filenames
:= List.fold ~f:collect ~init:!models_specs_filenames (Zip.entries zip_channel) ;
models_tenv := load_models_tenv zip_channel ;
Zip.close_in zip_channel
let add_models jar_filename =
models_jar := jar_filename ;
if Sys.file_exists !models_jar = `Yes then collect_specs_filenames jar_filename
else L.(die InternalError) "Java model file not found"
let is_model procname = String.Set.mem !models_specs_filenames (Typ.Procname.to_filename procname)
let split_classpath cp = Str.split (Str.regexp JFile.sep) cp
let append_path classpath path =
if Sys.file_exists path = `Yes then
let root = Unix.getcwd () in
let full_path = Utils.filename_to_absolute ~root path in
if Int.equal (String.length classpath) 0 then full_path else classpath ^ JFile.sep ^ full_path
else classpath
type file_entry = Singleton of SourceFile.t | Duplicate of (string * SourceFile.t) list
type t = string * file_entry String.Map.t * JBasics.ClassSet.t
(* Open the source file and search for the package declaration.
Only the case where the package is declared in a single line is supported *)
let read_package_declaration source_file =
let path = SourceFile.to_abs_path source_file in
let file_in = In_channel.create path in
let remove_trailing_semicolon = Str.replace_first (Str.regexp ";") "" in
let empty_package = "" in
let rec loop () =
try
let line = remove_trailing_semicolon (In_channel.input_line_exn file_in) in
match Str.split (Str.regexp "[ \t]+") line with
| [] ->
loop ()
| [hd; package] when String.equal hd "package" ->
package
| _ ->
loop ()
with End_of_file -> In_channel.close file_in ; empty_package
in
loop ()
let add_source_file path map =
let convert_to_absolute p =
if Filename.is_relative p then Filename.concat (Sys.getcwd ()) p else p
in
let basename = Filename.basename path in
let entry =
let current_source_file = SourceFile.from_abs_path (convert_to_absolute path) in
try
match String.Map.find_exn map basename with
| Singleton previous_source_file ->
(* Another source file with the same base name has been found.
Reading the package from the source file to resolve the ambiguity
only happens in this case *)
let previous_package = read_package_declaration previous_source_file
and current_package = read_package_declaration current_source_file in
let source_list =
[(current_package, current_source_file); (previous_package, previous_source_file)]
in
Duplicate source_list
| Duplicate previous_source_files ->
(* Two or more source file with the same base name have been found *)
let current_package = read_package_declaration current_source_file in
Duplicate ((current_package, current_source_file) :: previous_source_files)
with Not_found ->
(* Most common case: there is no conflict with the base name of the source file *)
Singleton current_source_file
in
String.Map.add ~key:basename ~data:entry map
let add_root_path path roots = String.Set.add roots path
let load_from_verbose_output javac_verbose_out =
let file_in = In_channel.create javac_verbose_out in
let class_filename_re =
Str.regexp
(Printf.sprintf
(* the unreadable regexp below captures 3 possible forms:
1. [wrote DirectoryFileObject[/path/to/classes_out:path/to/File.java]], leaves `path/to/File.java` in match group 2
2. [wrote RegularFileObject[path/to/File.java]], leaves `path/to/File.java` in match group 5
3. [wrote SimpleFileObject[path/to/File.java]], also leaves `path/to/File.java` in match group 5 *)
"\\[wrote \\(DirectoryFileObject\\[%s:\\(.*\\)\\|\\(\\(Regular\\|Simple\\)FileObject\\[\\(.*\\)\\)\\)\\]\\]"
Config.javac_classes_out)
in
let source_filename_re =
Str.regexp "\\[parsing started \\(Regular\\|Simple\\)FileObject\\[\\(.*\\)\\]\\]"
in
let classpath_re = Str.regexp "\\[search path for class files: \\(.*\\)\\]" in
let rec loop paths roots sources classes =
try
let line = In_channel.input_line_exn file_in in
if Str.string_match class_filename_re line 0 then
let path =
try Str.matched_group 5 line with Not_found ->
(* either matched group 5 is found, or matched group 2 is found, see doc for [class_filename_re] above *)
Config.javac_classes_out ^/ Str.matched_group 2 line
in
let cn, root_info = Javalib.extract_class_name_from_file path in
let root_dir =
if String.equal root_info "" then Filename.current_dir_name else root_info
in
loop paths (add_root_path root_dir roots) sources (JBasics.ClassSet.add cn classes)
else if Str.string_match source_filename_re line 0 then
let path = Str.matched_group 2 line in
loop paths roots (add_source_file path sources) classes
else if Str.string_match classpath_re line 0 then
let classpath = Str.matched_group 1 line in
let parsed_paths = String.split ~on:',' classpath in
loop parsed_paths roots sources classes
else (* skip this line *)
loop paths roots sources classes
with
| JBasics.Class_structure_error _ | Invalid_argument _ ->
loop paths roots sources classes
| End_of_file ->
In_channel.close file_in ;
let classpath = List.fold ~f:append_path ~init:"" (String.Set.elements roots @ paths) in
(classpath, sources, classes)
in
loop [] String.Set.empty String.Map.empty JBasics.ClassSet.empty
let classname_of_class_filename class_filename =
JBasics.make_cn (String.map ~f:(function '/' -> '.' | c -> c) class_filename)
let extract_classnames classnames jar_filename =
let file_in = Zip.open_in jar_filename in
let collect classes entry =
let class_filename = entry.Zip.filename in
match Filename.split_extension class_filename with
| basename, Some "class" ->
classname_of_class_filename basename :: classes
| _ ->
classes
in
let classnames_after = List.fold ~f:collect ~init:classnames (Zip.entries file_in) in
Zip.close_in file_in ; classnames_after
let collect_classnames start_classmap jar_filename =
List.fold
~f:(fun map cn -> JBasics.ClassSet.add cn map)
~init:start_classmap
(extract_classnames [] jar_filename)
let search_classes path =
let add_class roots classes class_filename =
let cn, root_dir = Javalib.extract_class_name_from_file class_filename in
(add_root_path root_dir roots, JBasics.ClassSet.add cn classes)
in
Utils.directory_fold
(fun accu p ->
let paths, classes = accu in
if Filename.check_suffix p "class" then add_class paths classes p
else if Filename.check_suffix p "jar" then
(add_root_path p paths, collect_classnames classes p)
else accu)
(String.Set.empty, JBasics.ClassSet.empty) path
let search_sources () =
let initial_map =
List.fold ~f:(fun map path -> add_source_file path map) ~init:String.Map.empty Config.sources
in
match Config.sourcepath with
| None ->
initial_map
| Some sourcepath ->
Utils.directory_fold
(fun map p -> if Filename.check_suffix p "java" then add_source_file p map else map)
initial_map sourcepath
let load_from_arguments classes_out_path =
let roots, classes = search_classes classes_out_path in
let split cp_option = Option.value_map ~f:split_classpath ~default:[] cp_option in
let combine path_list classpath =
List.fold ~f:append_path ~init:classpath (List.rev path_list)
in
let classpath =
combine (split Config.classpath) "" |> combine (String.Set.elements roots)
|> combine (split Config.bootclasspath)
in
(classpath, search_sources (), classes)
type classmap = JCode.jcode Javalib.interface_or_class JBasics.ClassMap.t
type program = {classpath: Javalib.class_path; models: classmap; mutable classmap: classmap}
let get_classmap program = program.classmap
let get_classpath program = program.classpath
let get_models program = program.models
let add_class cn jclass program =
program.classmap <- JBasics.ClassMap.add cn jclass program.classmap
let cleanup program = Javalib.close_class_path program.classpath
let lookup_node cn program =
try Some (JBasics.ClassMap.find cn (get_classmap program)) with Not_found ->
try
let jclass = javalib_get_class (get_classpath program) cn in
add_class cn jclass program ; Some jclass
with
| JBasics.No_class_found _ | JBasics.Class_structure_error _ | Invalid_argument _ ->
None
let collect_classes start_classmap jar_filename =
let classpath = Javalib.class_path jar_filename in
let collect classmap cn =
try JBasics.ClassMap.add cn (javalib_get_class classpath cn) classmap
with JBasics.Class_structure_error _ -> classmap
in
let classmap = List.fold ~f:collect ~init:start_classmap (extract_classnames [] jar_filename) in
Javalib.close_class_path classpath ;
classmap
let load_program classpath classes =
L.(debug Capture Medium) "loading program ... %!" ;
let models =
if String.equal !models_jar "" then JBasics.ClassMap.empty
else collect_classes JBasics.ClassMap.empty !models_jar
in
let program =
{classpath= Javalib.class_path classpath; models; classmap= JBasics.ClassMap.empty}
in
JBasics.ClassSet.iter (fun cn -> ignore (lookup_node cn program)) classes ;
L.(debug Capture Medium) "done@." ;
program