|
|
|
(*
|
|
|
|
* Copyright (c) 2016 - 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
|
|
|
|
|
|
|
|
type compilation_data = {
|
|
|
|
dir : string;
|
|
|
|
command : string;
|
|
|
|
args : string;
|
|
|
|
}
|
|
|
|
|
|
|
|
type t = compilation_data SourceFile.Map.t ref
|
|
|
|
let empty () = ref SourceFile.Map.empty
|
|
|
|
|
|
|
|
let get_size database = SourceFile.Map.cardinal !database
|
|
|
|
|
|
|
|
let iter database f = SourceFile.Map.iter f !database
|
|
|
|
|
|
|
|
let find database key = SourceFile.Map.find key !database
|
|
|
|
|
|
|
|
let parse_command_and_arguments command_and_arguments =
|
|
|
|
let regexp = Str.regexp "[^\\][ ]" in
|
|
|
|
let index = Str.search_forward regexp command_and_arguments 0 in
|
|
|
|
let command = Str.string_before command_and_arguments (index+1) in
|
|
|
|
let arguments = Str.string_after command_and_arguments (index+1) in
|
|
|
|
command, arguments
|
|
|
|
|
|
|
|
(** Parse the compilation database json file into the compilationDatabase
|
|
|
|
map. The json file consists of an array of json objects that contain the file
|
|
|
|
to be compiled, the directory to be compiled in, and the compilation command as a list
|
|
|
|
and as a string. We pack this information into the compilationDatabase map, and remove the
|
|
|
|
clang invocation part, because we will use a clang wrapper. *)
|
|
|
|
let decode_json_file (database : t) json_format =
|
|
|
|
let json_path = match json_format with | `Raw x | `Escaped x -> x in
|
|
|
|
let to_string s = match json_format with
|
|
|
|
| `Raw _ ->
|
|
|
|
s
|
|
|
|
| `Escaped _ ->
|
|
|
|
Utils.with_process_in (Printf.sprintf "/bin/sh -c 'printf \"%%s\" %s'" s) input_line
|
|
|
|
|> fst in
|
|
|
|
Logging.out "parsing compilation database from %s@\n" json_path;
|
|
|
|
let exit_format_error () =
|
|
|
|
failwith ("Json file doesn't have the expected format") in
|
|
|
|
let json = Yojson.Basic.from_file json_path in
|
|
|
|
let get_dir el =
|
|
|
|
match el with
|
|
|
|
| ("directory", `String dir) -> Some (to_string dir)
|
|
|
|
| _ -> None in
|
|
|
|
let get_file el =
|
|
|
|
match el with
|
|
|
|
| ("file", `String file) -> Some (to_string file)
|
|
|
|
| _ -> None in
|
|
|
|
let get_cmd el =
|
|
|
|
match el with
|
|
|
|
| ("command", `String cmd) -> Some cmd
|
|
|
|
| _ -> None in
|
|
|
|
let rec parse_json json =
|
|
|
|
match json with
|
|
|
|
| `List arguments ->
|
|
|
|
List.iter ~f:parse_json arguments
|
|
|
|
| `Assoc l ->
|
|
|
|
let dir = match List.find_map ~f:get_dir l with
|
|
|
|
| Some dir -> dir
|
|
|
|
| None -> exit_format_error () in
|
|
|
|
let file = match List.find_map ~f:get_file l with
|
|
|
|
| Some file -> file
|
|
|
|
| None -> exit_format_error () in
|
|
|
|
let cmd = match List.find_map ~f:get_cmd l with
|
|
|
|
| Some cmd -> cmd
|
|
|
|
| None -> exit_format_error () in
|
|
|
|
let command, args = parse_command_and_arguments cmd in
|
|
|
|
let compilation_data = { dir; command; args;} in
|
|
|
|
let abs_file = if Filename.is_relative file then dir ^/ file else file in
|
|
|
|
let source_file = SourceFile.from_abs_path abs_file in
|
|
|
|
database := SourceFile.Map.add source_file compilation_data !database
|
|
|
|
| _ -> exit_format_error () in
|
|
|
|
parse_json json
|
|
|
|
|
|
|
|
let from_json_files db_json_files =
|
|
|
|
let db = empty () in
|
|
|
|
List.iter ~f:(decode_json_file db) db_json_files;
|
|
|
|
Logging.out "created database with %d entries@\n" (get_size db);
|
|
|
|
db
|