(* * 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)