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.

236 lines
8.4 KiB

(*
* 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.
*)
(** This module creates extra ast constructs that are needed for the translation *)
open! IStd
module L = Logging
let dummy_source_range () =
let dummy_source_loc = {Clang_ast_t.sl_file= None; sl_line= None; sl_column= None} in
(dummy_source_loc, dummy_source_loc)
let dummy_stmt_info () =
{Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer (); si_source_range= dummy_source_range ()}
let stmt_info_with_fresh_pointer stmt_info =
{ Clang_ast_t.si_pointer= CAst_utils.get_fresh_pointer ()
; si_source_range= stmt_info.Clang_ast_t.si_source_range }
let create_qual_type ?(quals= Typ.mk_type_quals ()) qt_type_ptr =
{ Clang_ast_t.qt_type_ptr
; qt_is_const= Typ.is_const quals
; qt_is_volatile= Typ.is_volatile quals
; qt_is_restrict= Typ.is_restrict quals }
let builtin_to_qual_type kind = create_qual_type (Clang_ast_extend.Builtin kind)
let create_pointer_qual_type ?quals typ = create_qual_type ?quals (Clang_ast_extend.PointerOf typ)
let create_reference_qual_type ?quals typ =
create_qual_type ?quals (Clang_ast_extend.ReferenceOf typ)
let create_int_type = builtin_to_qual_type `Int
let create_void_type = builtin_to_qual_type `Void
let create_void_star_type = create_pointer_qual_type create_void_type
let create_id_type = create_pointer_qual_type (builtin_to_qual_type `ObjCId)
let create_char_type = builtin_to_qual_type `Char_S
let create_char_star_type ?quals () = create_pointer_qual_type ?quals create_char_type
let create_BOOL_type = builtin_to_qual_type `SChar
let create_class_qual_type ?quals typename =
create_qual_type ?quals (Clang_ast_extend.ClassType typename)
let create_integer_literal n =
let stmt_info = dummy_stmt_info () in
let expr_info =
{Clang_ast_t.ei_qual_type= create_int_type; ei_value_kind= `RValue; ei_object_kind= `Ordinary}
in
let integer_literal_info = {Clang_ast_t.ili_is_signed= true; ili_bitwidth= 32; ili_value= n} in
Clang_ast_t.IntegerLiteral (stmt_info, [], expr_info, integer_literal_info)
let create_cstyle_cast_expr stmt_info stmts qt =
let expr_info =
{ Clang_ast_t.ei_qual_type= create_void_star_type
; ei_value_kind= `RValue
; ei_object_kind= `Ordinary }
in
let cast_expr = {Clang_ast_t.cei_cast_kind= `NullToPointer; cei_base_path= []} in
Clang_ast_t.CStyleCastExpr (stmt_info, stmts, expr_info, cast_expr, qt)
let create_parent_expr stmt_info stmts =
let expr_info =
{ Clang_ast_t.ei_qual_type= create_void_star_type
; ei_value_kind= `RValue
; ei_object_kind= `Ordinary }
in
Clang_ast_t.ParenExpr (stmt_info, stmts, expr_info)
let create_implicit_cast_expr stmt_info stmts typ cast_kind =
let expr_info =
{Clang_ast_t.ei_qual_type= typ; ei_value_kind= `RValue; ei_object_kind= `Ordinary}
in
let cast_expr_info = {Clang_ast_t.cei_cast_kind= cast_kind; cei_base_path= []} in
Clang_ast_t.ImplicitCastExpr (stmt_info, stmts, expr_info, cast_expr_info)
let create_nil stmt_info =
let integer_literal = create_integer_literal "0" in
let cstyle_cast_expr = create_cstyle_cast_expr stmt_info [integer_literal] create_int_type in
let paren_expr = create_parent_expr stmt_info [cstyle_cast_expr] in
create_implicit_cast_expr stmt_info [paren_expr] create_id_type `NullToPointer
let dummy_stmt () =
let pointer = CAst_utils.get_fresh_pointer () in
let source_range = dummy_source_range () in
Clang_ast_t.NullStmt ({Clang_ast_t.si_pointer= pointer; si_source_range= source_range}, [])
let make_stmt_info di =
{ Clang_ast_t.si_pointer= di.Clang_ast_t.di_pointer
; si_source_range= di.Clang_ast_t.di_source_range }
let make_expr_info qt vk objc_kind =
{Clang_ast_t.ei_qual_type= qt; ei_value_kind= vk; ei_object_kind= objc_kind}
let make_expr_info_with_objc_kind qt objc_kind = make_expr_info qt `LValue objc_kind
let make_obj_c_message_expr_info_instance sel =
{ Clang_ast_t.omei_selector= sel
; omei_receiver_kind= `Instance
; omei_is_definition_found= false
; omei_decl_pointer= None (* TODO look into it *) }
let make_obj_c_message_expr_info_class selector tname pointer =
{ Clang_ast_t.omei_selector= selector
; omei_receiver_kind= `Class (create_class_qual_type tname)
; omei_is_definition_found= false
; omei_decl_pointer= pointer }
let make_decl_ref k decl_ptr name is_hidden qt_opt =
{ Clang_ast_t.dr_kind= k
; dr_decl_pointer= decl_ptr
; dr_name= Some name
; dr_is_hidden= is_hidden
; dr_qual_type= qt_opt }
let make_decl_ref_qt k decl_ptr name is_hidden qt =
make_decl_ref k decl_ptr name is_hidden (Some qt)
let make_decl_ref_expr_info decl_ref =
{Clang_ast_t.drti_decl_ref= Some decl_ref; drti_found_decl_ref= None}
let make_general_expr_info qt vk ok =
{Clang_ast_t.ei_qual_type= qt; ei_value_kind= vk; ei_object_kind= ok}
let make_message_expr param_qt selector decl_ref_exp stmt_info add_cast =
let stmt_info = stmt_info_with_fresh_pointer stmt_info in
let parameters =
if add_cast then
let cast_expr =
create_implicit_cast_expr stmt_info [decl_ref_exp] param_qt `LValueToRValue
in
[cast_expr]
else [decl_ref_exp]
in
let obj_c_message_expr_info = make_obj_c_message_expr_info_instance selector in
let expr_info = make_expr_info_with_objc_kind param_qt `ObjCProperty in
Clang_ast_t.ObjCMessageExpr (stmt_info, parameters, expr_info, obj_c_message_expr_info)
let make_binary_stmt stmt1 stmt2 stmt_info expr_info boi =
let stmt_info = stmt_info_with_fresh_pointer stmt_info in
Clang_ast_t.BinaryOperator (stmt_info, [stmt1; stmt2], expr_info, boi)
let make_next_object_exp stmt_info item items =
let var_decl_ref, var_type =
match item with
| Clang_ast_t.DeclStmt (_, _, [(Clang_ast_t.VarDecl (di, name_info, var_qual_type, _))]) ->
let decl_ptr = di.Clang_ast_t.di_pointer in
let decl_ref = make_decl_ref_qt `Var decl_ptr name_info false var_qual_type in
let stmt_info_var =
{ Clang_ast_t.si_pointer= di.Clang_ast_t.di_pointer
; si_source_range= di.Clang_ast_t.di_source_range }
in
let expr_info = make_expr_info_with_objc_kind var_qual_type `ObjCProperty in
let decl_ref_expr_info = make_decl_ref_expr_info decl_ref in
(Clang_ast_t.DeclRefExpr (stmt_info_var, [], expr_info, decl_ref_expr_info), var_qual_type)
| _ ->
CFrontend_config.incorrect_assumption "unexpected item %a"
(Pp.to_string ~f:Clang_ast_j.string_of_stmt)
item
in
let message_call =
make_message_expr create_id_type CFrontend_config.next_object items stmt_info false
in
let boi = {Clang_ast_t.boi_kind= `Assign} in
let expr_info = make_expr_info_with_objc_kind var_type `ObjCProperty in
let assignment = make_binary_stmt var_decl_ref message_call stmt_info expr_info boi in
let boi' = {Clang_ast_t.boi_kind= `NE} in
let cast = create_implicit_cast_expr stmt_info [var_decl_ref] var_type `LValueToRValue in
let nil_exp = create_nil stmt_info in
let loop_cond = make_binary_stmt cast nil_exp stmt_info expr_info boi' in
(assignment, loop_cond)
(* 1. dispatch_once(v,block_def) is transformed as: block_def() *)
(* 2. dispatch_once(v,block_var) is transformed as n$1 = *&block_var; n$2=n$1() *)
let translate_dispatch_function stmt_info stmt_list n =
let open Clang_ast_t in
match stmt_list with
| _ :: args_stmts ->
let expr_info_call = make_general_expr_info create_void_star_type `XValue `Ordinary in
let arg_stmt =
try List.nth_exn args_stmts n
with Failure _ -> assert false
in
CallExpr (stmt_info, [arg_stmt], expr_info_call)
| _ ->
assert false
(* We translate an expression with a conditional*)
(* x <=> x?1:0 *)
let trans_with_conditional stmt_info expr_info stmt_list =
let stmt_list_cond = stmt_list @ [create_integer_literal "1"] @ [create_integer_literal "0"] in
Clang_ast_t.ConditionalOperator (stmt_info, stmt_list_cond, expr_info)
(* We translate the logical negation of an expression with a conditional*)
(* !x <=> x?0:1 *)
let trans_negation_with_conditional stmt_info expr_info stmt_list =
let stmt_list_cond = stmt_list @ [create_integer_literal "0"] @ [create_integer_literal "1"] in
Clang_ast_t.ConditionalOperator (stmt_info, stmt_list_cond, expr_info)