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.

137 lines
3.6 KiB

(*
* Copyright (c) 2009 - 2013 Monoidics ltd.
* 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.
*)
open! Utils
(** This module handles buckets of memory leaks in Objective-C/C++ *)
let objc_arc_flag = "objc_arc"
let bucket_delimiter = ","
type mleak_bucket =
| MLeak_cf
| MLeak_arc
| MLeak_no_arc
| MLeak_cpp
| MLeak_unknown
let ml_buckets = ref []
let bucket_from_string bucket_s =
match bucket_s with
| "cf" -> MLeak_cf
| "arc" -> MLeak_arc
| "narc" -> MLeak_no_arc
| "cpp" -> MLeak_cpp
| "unknown_origin" -> MLeak_unknown
| _ -> assert false
let bucket_to_message bucket =
match bucket with
| MLeak_cf -> "[CF]"
| MLeak_arc -> "[ARC]"
| MLeak_no_arc -> "[NO ARC]"
| MLeak_cpp -> "[CPP]"
| MLeak_unknown -> "[UNKNOWN ORIGIN]"
let mleak_bucket_compare b1 b2 =
match b1, b2 with
| MLeak_cf, MLeak_cf -> 0
| MLeak_cf, _ -> -1
| _, MLeak_cf -> 1
| MLeak_arc, MLeak_arc -> 0
| MLeak_arc, _ -> -1
| _, MLeak_arc -> 1
| MLeak_no_arc, MLeak_no_arc -> 0
| MLeak_no_arc, _ -> -1
| _, MLeak_no_arc -> 1
| MLeak_unknown, MLeak_unknown -> 0
| MLeak_unknown, _ -> -1
| _, MLeak_unknown -> 1
| MLeak_cpp, MLeak_cpp -> 0
let mleak_bucket_eq b1 b2 =
mleak_bucket_compare b1 b2 = 0
let init_buckets ml_buckets_arg =
let buckets =
Str.split (Str.regexp bucket_delimiter) ml_buckets_arg in
let buckets =
match buckets with
| ["all"] -> []
| _ -> IList.map bucket_from_string buckets in
ml_buckets := buckets
let contains_cf ml_buckets =
IList.mem mleak_bucket_eq MLeak_cf ml_buckets
let contains_arc ml_buckets =
IList.mem mleak_bucket_eq MLeak_arc ml_buckets
let contains_narc ml_buckets =
IList.mem mleak_bucket_eq MLeak_no_arc ml_buckets
let contains_cpp ml_buckets =
IList.mem mleak_bucket_eq MLeak_cpp ml_buckets
let contains_unknown_origin ml_buckets =
IList.mem mleak_bucket_eq MLeak_unknown ml_buckets
let should_raise_leak_cf typ =
if contains_cf !ml_buckets then
Objc_models.is_core_lib_type typ
else false
let should_raise_leak_arc () =
if contains_arc !ml_buckets then
!Config.arc_mode
else false
let should_raise_leak_no_arc () =
if contains_narc !ml_buckets then
not (!Config.arc_mode)
else false
let should_raise_leak_unknown_origin () =
contains_unknown_origin !ml_buckets
let ml_bucket_unknown_origin () =
bucket_to_message MLeak_unknown
(* Returns whether a memory leak should be raised for a C++ object.*)
(* If ml_buckets contains cpp, then check leaks from C++ objects. *)
let should_raise_cpp_leak () =
if contains_cpp !ml_buckets then
Some (bucket_to_message MLeak_cpp)
else None
(* Returns whether a memory leak should be raised. *)
(* If cf is passed, then check leaks from Core Foundation. *)
(* If arc is passed, check leaks from code that compiles with arc*)
(* If no arc is passed check the leaks from code that compiles without arc *)
let should_raise_objc_leak typ =
if IList.length !ml_buckets = 0 then Some ""
else
if should_raise_leak_cf typ then Some (bucket_to_message MLeak_cf)
else if should_raise_leak_arc () then Some (bucket_to_message MLeak_arc)
else if should_raise_leak_no_arc () then Some (bucket_to_message MLeak_no_arc)
else None
(*
let bucket_to_string bucket =
match bucket with
| MLeak_cf -> "Core Foundation"
| MLeak_arc -> "Arc"
| MLeak_no_arc -> "No arc"
| MLeak_cpp -> "Cpp"
| MLeak_unknown -> "Unknown origin"
*)