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.

208 lines
8.5 KiB

(*
* Copyright (c) 2018-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*)
open! IStd
module F = Format
module L = Logging
let store_statement =
ResultsDatabase.register_statement
{|
INSERT OR REPLACE INTO source_files
VALUES (:source, :tenv, :integer_type_widths, :proc_names, :freshly_captured) |}
let select_existing_statement =
ResultsDatabase.register_statement
"SELECT type_environment, procedure_names FROM source_files WHERE source_file = :source AND \
freshly_captured = 1"
let get_existing_data source_file =
ResultsDatabase.with_registered_statement select_existing_statement ~f:(fun db stmt ->
SourceFile.SQLite.serialize source_file
|> Sqlite3.bind stmt 1
(* :source *)
|> SqliteUtils.check_result_code db ~log:"get_existing_data bind source file" ;
SqliteUtils.result_option ~finalize:false db ~log:"looking for pre-existing source file data"
stmt ~read_row:(fun stmt ->
let tenv = Sqlite3.column stmt 0 |> Tenv.SQLite.deserialize
and proc_names = Sqlite3.column stmt 1 |> Typ.Procname.SQLiteList.deserialize in
(tenv, proc_names) ) )
let add source_file cfg tenv integer_type_widths =
Cfg.inline_java_synthetic_methods cfg ;
let tenv, proc_names =
(* The same source file may get captured several times in a single capture event, for instance
because it is compiled with different options, or from different symbolic links. This may
generate different procedures in each phase, so make an attempt to merge them into the same
tenv. *)
let new_proc_names = Cfg.get_all_defined_proc_names cfg in
match get_existing_data source_file with
| Some (old_tenv, old_proc_names) ->
L.debug Capture Quiet "Merging capture data for already-captured '%a'@\n" SourceFile.pp
source_file ;
(* merge the proc names defined in the source file using a hashtbl so that order is
preserved but merging is still linear time *)
let existing_proc_names = Caml.Hashtbl.create (List.length old_proc_names) in
List.iter old_proc_names ~f:(fun proc_name ->
Caml.Hashtbl.add existing_proc_names proc_name () ) ;
let proc_names =
List.fold new_proc_names ~init:old_proc_names ~f:(fun proc_names proc_name ->
if not (Caml.Hashtbl.mem existing_proc_names proc_name) then proc_name :: proc_names
else proc_names )
in
(Tenv.merge ~dst:old_tenv ~src:tenv, proc_names)
| None ->
(tenv, new_proc_names)
in
(* NOTE: it's important to write attribute files to disk before writing cfgs to disk.
OndemandCapture module relies on it - it uses existance of the cfg as a barrier to make
sure that all attributes were written to disk (but not necessarily flushed) *)
SqliteUtils.with_transaction (ResultsDatabase.get_database ()) ~f:(fun () ->
Cfg.store source_file cfg ) ;
ResultsDatabase.with_registered_statement store_statement ~f:(fun db store_stmt ->
SourceFile.SQLite.serialize source_file
|> Sqlite3.bind store_stmt 1
(* :source *)
|> SqliteUtils.check_result_code db ~log:"store bind source file" ;
Tenv.SQLite.serialize tenv |> Sqlite3.bind store_stmt 2
(* :tenv *)
|> SqliteUtils.check_result_code db ~log:"store bind type environment" ;
Typ.IntegerWidths.SQLite.serialize integer_type_widths
|> Sqlite3.bind store_stmt 3
(* :integer_type_widths *)
|> SqliteUtils.check_result_code db ~log:"store bind integer type widths" ;
Typ.Procname.SQLiteList.serialize proc_names
|> Sqlite3.bind store_stmt 4
(* :proc_names *)
|> SqliteUtils.check_result_code db ~log:"store bind proc names" ;
Sqlite3.bind store_stmt 5 (Sqlite3.Data.INT Int64.one)
(* :freshly_captured *)
|> SqliteUtils.check_result_code db ~log:"store freshness" ;
SqliteUtils.result_unit ~finalize:false ~log:"Cfg.store" db store_stmt )
let get_all ~filter () =
let db = ResultsDatabase.get_database () in
(* we could also register this statement but it's typically used only once per run so just prepare
it inside the function *)
Sqlite3.prepare db "SELECT source_file FROM source_files"
|> IContainer.rev_filter_map_to_list
~fold:(SqliteUtils.result_fold_single_column_rows db ~log:"getting all source files")
~f:(fun column ->
let source_file = SourceFile.SQLite.deserialize column in
Option.some_if (filter source_file) source_file )
let load_proc_names_statement =
ResultsDatabase.register_statement
"SELECT procedure_names FROM source_files WHERE source_file = :k"
let proc_names_of_source source =
ResultsDatabase.with_registered_statement load_proc_names_statement ~f:(fun db load_stmt ->
SourceFile.SQLite.serialize source
|> Sqlite3.bind load_stmt 1
|> SqliteUtils.check_result_code db ~log:"load bind source file" ;
SqliteUtils.result_single_column_option ~finalize:false db
~log:"SourceFiles.proc_names_of_source" load_stmt
|> Option.value_map ~default:[] ~f:Typ.Procname.SQLiteList.deserialize )
let exists_source_statement =
ResultsDatabase.register_statement "SELECT 1 FROM source_files WHERE source_file = :k"
let is_captured source =
ResultsDatabase.with_registered_statement exists_source_statement ~f:(fun db exists_stmt ->
SourceFile.SQLite.serialize source
|> Sqlite3.bind exists_stmt 1
(* :k *)
|> SqliteUtils.check_result_code db ~log:"load captured source file" ;
SqliteUtils.result_single_column_option ~finalize:false ~log:"SourceFiles.is_captured" db
exists_stmt
|> Option.is_some )
let is_non_empty_statement =
ResultsDatabase.register_statement "SELECT 1 FROM source_files LIMIT 1"
let is_empty () =
ResultsDatabase.with_registered_statement is_non_empty_statement ~f:(fun db stmt ->
SqliteUtils.result_single_column_option ~finalize:false ~log:"SourceFiles.is_empty" db stmt
|> Option.is_none )
let is_freshly_captured_statement =
ResultsDatabase.register_statement
"SELECT freshly_captured FROM source_files WHERE source_file = :k"
let deserialize_freshly_captured = function[@warning "-8"]
| Sqlite3.Data.INT p ->
Int64.equal p Int64.one
let is_freshly_captured source =
ResultsDatabase.with_registered_statement is_freshly_captured_statement ~f:(fun db load_stmt ->
SourceFile.SQLite.serialize source
|> Sqlite3.bind load_stmt 1
|> SqliteUtils.check_result_code db ~log:"load bind source file" ;
SqliteUtils.result_single_column_option ~finalize:false
~log:"SourceFiles.is_freshly_captured" db load_stmt
|> Option.value_map ~default:false ~f:deserialize_freshly_captured )
let mark_all_stale_statement =
ResultsDatabase.register_statement "UPDATE source_files SET freshly_captured = 0"
let mark_all_stale () =
ResultsDatabase.with_registered_statement mark_all_stale_statement ~f:(fun db stmt ->
SqliteUtils.result_unit db ~finalize:false ~log:"mark_all_stale" stmt )
let select_all_source_files_statement =
ResultsDatabase.register_statement "SELECT * FROM source_files"
let pp_all ~filter ~type_environment ~procedure_names ~freshly_captured fmt () =
let pp_procnames fmt procs =
F.fprintf fmt "@[<v>" ;
List.iter ~f:(F.fprintf fmt "%a@," Typ.Procname.pp) procs ;
F.fprintf fmt "@]"
in
let pp_if stmt title condition deserialize pp fmt column =
if condition then
F.fprintf fmt " @[<v2>%s@,%a@]@;" title pp (Sqlite3.column stmt column |> deserialize)
in
let pp_row stmt fmt source_file =
F.fprintf fmt "%a@,%a%a%a" SourceFile.pp source_file
(pp_if stmt "type_environment" type_environment Tenv.SQLite.deserialize Tenv.pp_per_file)
1
(pp_if stmt "procedure_names" procedure_names Typ.Procname.SQLiteList.deserialize
pp_procnames)
2
(pp_if stmt "freshly_captured" freshly_captured deserialize_freshly_captured
Format.pp_print_bool)
3
in
ResultsDatabase.with_registered_statement select_all_source_files_statement ~f:(fun db stmt ->
let pp fmt column =
let source_file = SourceFile.SQLite.deserialize column in
if filter source_file then pp_row stmt fmt source_file
in
let pp_result fmt stmt =
Container.iter stmt ~f:(pp fmt)
~fold:
(SqliteUtils.result_fold_single_column_rows ~finalize:false db
~log:"printing all source files")
in
F.fprintf fmt "@[<v>%a@]" pp_result stmt )