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.

113 lines
3.6 KiB

(*
* Copyright (c) 2015 - 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 t = Clang_ast_t.pointer
module Map = Map.Make(Int)
let ivar_to_property_table = Int.Table.create ~size:256 ()
let pointer_decl_table = Int.Table.create ~size:256 ()
let pointer_stmt_table = Int.Table.create ~size:256 ()
let pointer_type_table = Int.Table.create ~size:256 ()
let empty_v = Clang_ast_visit.empty_visitor
(* This function is not thread-safe *)
let visit_ast
?(visit_decl=empty_v) ?(visit_stmt=empty_v) ?(visit_type=empty_v) ?(visit_src_loc=empty_v)
top_decl =
Clang_ast_visit.decl_visitor := visit_decl;
Clang_ast_visit.stmt_visitor := visit_stmt;
Clang_ast_visit.type_visitor := visit_type;
Clang_ast_visit.source_location_visitor := visit_src_loc;
match Clang_ast_v.validate_decl [] top_decl (* visit *) with
| None ->
()
| Some error ->
failwithf "visiting the clang AST failed with error %s"
(Ag_util.Validation.string_of_error error)
let get_ptr_from_node node =
match node with
| `DeclNode decl ->
let decl_info = Clang_ast_proj.get_decl_tuple decl in
decl_info.Clang_ast_t.di_pointer
| `StmtNode stmt ->
let stmt_info,_ = Clang_ast_proj.get_stmt_tuple stmt in
stmt_info.Clang_ast_t.si_pointer
| `TypeNode c_type ->
let type_info = Clang_ast_proj.get_type_tuple c_type in
type_info.Clang_ast_t.ti_pointer
let get_val_from_node node =
match node with
| `DeclNode decl -> decl
| `StmtNode stmt -> stmt
| `TypeNode c_type -> c_type
let add_node_to_cache node cache =
let key = get_ptr_from_node node in
let data = get_val_from_node node in
Int.Table.set cache ~key ~data
let process_decl _ decl =
add_node_to_cache (`DeclNode decl) pointer_decl_table;
match decl with
| Clang_ast_t.ObjCPropertyDecl (_, _, {opdi_ivar_decl=Some decl_ref}) ->
let ivar_pointer = decl_ref.Clang_ast_t.dr_decl_pointer in
Int.Table.set ivar_to_property_table ~key:ivar_pointer ~data:decl
| _ -> ()
let add_stmt_to_cache _ stmt =
add_node_to_cache (`StmtNode stmt) pointer_stmt_table
let add_type_to_cache _ c_type =
add_node_to_cache (`TypeNode c_type) pointer_type_table
let previous_sloc = { Clang_ast_t.sl_file = None; sl_line = None; sl_column = None }
let get_sloc current previous =
match current with
| None -> previous
| Some _ -> current
let mutate_sloc sloc file line column =
let open Clang_ast_t in
sloc.sl_file <- file;
sloc.sl_line <- line;
sloc.sl_column <- column
let reset_sloc sloc = mutate_sloc sloc None None None
let complete_source_location _ source_loc =
let open Clang_ast_t in
let file = get_sloc source_loc.sl_file previous_sloc.sl_file in
let line = get_sloc source_loc.sl_line previous_sloc.sl_line in
let column = get_sloc source_loc.sl_column previous_sloc.sl_column in
mutate_sloc source_loc file line column;
mutate_sloc previous_sloc file line column
let reset_cache () =
Int.Table.clear pointer_decl_table;
Int.Table.clear pointer_stmt_table;
Int.Table.clear pointer_type_table;
Int.Table.clear ivar_to_property_table;
reset_sloc previous_sloc
(* This function is not thread-safe *)
let populate_all_tables top_decl =
reset_cache ();
(* populate caches *)
visit_ast
~visit_decl:process_decl
~visit_stmt:add_stmt_to_cache
~visit_type:add_type_to_cache
~visit_src_loc:complete_source_location top_decl