[inferbo] Revise finding constructors on std::make_shared

Summary:
When finding a proper constructor for `std::make_shared`, the given parameter types are sometimes
slightly different, e.g., const int vs int.  This diff loosens the condition of the types on finding
constructors.

Reviewed By: ngorogiannis

Differential Revision: D19743198

fbshipit-source-id: f90213109
master
Sungkeun Cho 5 years ago committed by Facebook Github Bot
parent e7d42838b5
commit ca04002f6c

@ -220,10 +220,14 @@ module T = struct
equal_ikind ikind1 ikind2 equal_ikind ikind1 ikind2
| Tfloat fkind1, Tfloat fkind2 -> | Tfloat fkind1, Tfloat fkind2 ->
equal_fkind fkind1 fkind2 equal_fkind fkind1 fkind2
| Tvoid, Tvoid -> | Tvoid, Tvoid | Tfun, Tfun ->
true true
| Tptr (t1, ptr_kind1), Tptr (t2, ptr_kind2) -> | Tptr (t1, ptr_kind1), Tptr (t2, ptr_kind2) ->
equal_ptr_kind ptr_kind1 ptr_kind2 && equal_ignore_quals t1 t2 equal_ptr_kind ptr_kind1 ptr_kind2 && equal_ignore_quals t1 t2
| Tstruct name1, Tstruct name2 ->
equal_name name1 name2
| TVar s1, TVar s2 ->
String.equal s1 s2
| Tarray {elt= t1}, Tarray {elt= t2} -> | Tarray {elt= t1}, Tarray {elt= t2} ->
equal_ignore_quals t1 t2 equal_ignore_quals t1 t2
| _, _ -> | _, _ ->
@ -573,7 +577,9 @@ let unsome s = function
(** turn a *T into a T. fails if [typ] is not a pointer type *) (** turn a *T into a T. fails if [typ] is not a pointer type *)
let strip_ptr typ = match typ.desc with Tptr (t, _) -> t | _ -> assert false let strip_ptr typ = match typ.desc with Tptr (t, _) -> t | _ -> assert false
let is_ptr_to t ~ptr = match ptr.desc with Tptr (t', _) -> equal t t' | _ -> false let is_ptr_to_ignore_quals t ~ptr =
match ptr.desc with Tptr (t', _) -> equal_ignore_quals t t' | _ -> false
(** If an array type, return the type of the element. (** If an array type, return the type of the element.
If not, return the default type if given, otherwise raise an exception *) If not, return the default type if given, otherwise raise an exception *)

@ -318,8 +318,8 @@ val name : t -> Name.t option
val strip_ptr : t -> t val strip_ptr : t -> t
(** turn a *T into a T. fails if [t] is not a pointer type *) (** turn a *T into a T. fails if [t] is not a pointer type *)
val is_ptr_to : t -> ptr:t -> bool val is_ptr_to_ignore_quals : t -> ptr:t -> bool
(** check if [ptr] is a pointer type to [t] *) (** check if [ptr] is a pointer type to [t], ignoring quals *)
val array_elem : t option -> t -> t val array_elem : t option -> t -> t
(** If an array type, return the type of the element. (** If an array type, return the type of the element.

@ -345,11 +345,12 @@ module ReplaceCallee = struct
when Int.equal (List.length parameters) num_params -> ( when Int.equal (List.length parameters) num_params -> (
match get_formals pname |> Option.map ~f:(List.map ~f:snd) with match get_formals pname |> Option.map ~f:(List.map ~f:snd) with
| Some (this_typ :: formal_typs) -> ( | Some (this_typ :: formal_typs) -> (
Typ.is_ptr_to class_typ ~ptr:this_typ Typ.is_ptr_to_ignore_quals class_typ ~ptr:this_typ
&& &&
match match
List.for_all2 param_ref_typs formal_typs ~f:(fun param_ref_typ formal_typ -> List.for_all2 param_ref_typs formal_typs ~f:(fun param_ref_typ formal_typ ->
Typ.is_ptr_to formal_typ ~ptr:param_ref_typ ) Typ.equal_ignore_quals formal_typ param_ref_typ
|| Typ.is_ptr_to_ignore_quals formal_typ ~ptr:param_ref_typ )
with with
| List.Or_unequal_lengths.Ok b -> | List.Or_unequal_lengths.Ok b ->
b b

@ -72,6 +72,8 @@ codetoanalyze/cpp/bufferoverrun/repro1.cpp, LM<TFM>::lI_FP, 2, INTEGER_OVERFLOW_
codetoanalyze/cpp/bufferoverrun/repro1.cpp, LM<TFM>::uI, 0, BUFFER_OVERRUN_U5, no_bucket, ERROR, [<Length trace>,Unknown value from: std::unique_ptr<LMB<TFM>,std::default_delete<LMB<TFM>>>::operator->,Array access: Offset: [-oo, +oo] Size: [0, +oo]] codetoanalyze/cpp/bufferoverrun/repro1.cpp, LM<TFM>::uI, 0, BUFFER_OVERRUN_U5, no_bucket, ERROR, [<Length trace>,Unknown value from: std::unique_ptr<LMB<TFM>,std::default_delete<LMB<TFM>>>::operator->,Array access: Offset: [-oo, +oo] Size: [0, +oo]]
codetoanalyze/cpp/bufferoverrun/repro1.cpp, am_Good_FP, 5, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [Call,Call,Call,Assignment,Assignment,Call,Parameter `t->bI`,Call,Assignment,Call,<LHS trace>,Parameter `bi`,Binary operation: ([-oo, +oo] - 1):signed32 by call to `ral_good` ] codetoanalyze/cpp/bufferoverrun/repro1.cpp, am_Good_FP, 5, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [Call,Call,Call,Assignment,Assignment,Call,Parameter `t->bI`,Call,Assignment,Call,<LHS trace>,Parameter `bi`,Binary operation: ([-oo, +oo] - 1):signed32 by call to `ral_good` ]
codetoanalyze/cpp/bufferoverrun/simple_vector.cpp, my_vector_oob_Bad, 2, BUFFER_OVERRUN_L2, no_bucket, ERROR, [Parameter `v->_size`,Call,<Offset trace>,Parameter `i`,<Length trace>,Parameter `this->_size`,Array declaration,Assignment,Array access: Offset: v->_size Size: v->_size by call to `int_vector::access_at` ] codetoanalyze/cpp/bufferoverrun/simple_vector.cpp, my_vector_oob_Bad, 2, BUFFER_OVERRUN_L2, no_bucket, ERROR, [Parameter `v->_size`,Call,<Offset trace>,Parameter `i`,<Length trace>,Parameter `this->_size`,Array declaration,Assignment,Array access: Offset: v->_size Size: v->_size by call to `int_vector::access_at` ]
codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::shared_ptr_with_const_int_Bad, 2, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `i`,<Length trace>,Array declaration,Array access: Offset: 8 Size: 5 by call to `smart_ptr::my_class::my_class` ]
codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::shared_ptr_with_std_string_Bad, 3, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `i`,<Length trace>,Array declaration,Array access: Offset: 8 Size: 5 by call to `smart_ptr::my_class::my_class` ]
codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::use_shared_ptr1_Bad, 2, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `i`,<Length trace>,Array declaration,Array access: Offset: 8 Size: 5 by call to `smart_ptr::my_class::my_class` ] codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::use_shared_ptr1_Bad, 2, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `i`,<Length trace>,Array declaration,Array access: Offset: 8 Size: 5 by call to `smart_ptr::my_class::my_class` ]
codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::use_shared_ptr2_Bad, 3, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `j`,<Length trace>,Array declaration,Array access: Offset: 16 Size: 10 by call to `smart_ptr::my_class::my_class` ] codetoanalyze/cpp/bufferoverrun/smart_ptr.cpp, smart_ptr::use_shared_ptr2_Bad, 3, BUFFER_OVERRUN_L1, no_bucket, ERROR, [Assignment,Call,<Offset trace>,Parameter `j`,<Length trace>,Array declaration,Array access: Offset: 16 Size: 10 by call to `smart_ptr::my_class::my_class` ]
codetoanalyze/cpp/bufferoverrun/std_array.cpp, array_iter1_Bad, 5, BUFFER_OVERRUN_L1, no_bucket, ERROR, [<Length trace>,Array declaration,Array access: Offset: 10 Size: 5] codetoanalyze/cpp/bufferoverrun/std_array.cpp, array_iter1_Bad, 5, BUFFER_OVERRUN_L1, no_bucket, ERROR, [<Length trace>,Array declaration,Array access: Offset: 10 Size: 5]

@ -6,6 +6,7 @@
*/ */
#include <memory> #include <memory>
#include <string>
class smart_ptr { class smart_ptr {
public: public:
@ -20,6 +21,11 @@ class smart_ptr {
int a[10]; int a[10];
a[i + j] = 0; a[i + j] = 0;
} }
my_class(const std::string& str, int i) {
int a[5];
a[i] = 0;
}
}; };
void use_shared_ptr1_Good() { void use_shared_ptr1_Good() {
@ -43,4 +49,26 @@ class smart_ptr {
int j = 8; int j = 8;
std::shared_ptr<my_class> p = std::make_shared<my_class>(i, j); std::shared_ptr<my_class> p = std::make_shared<my_class>(i, j);
} }
void shared_ptr_with_std_string_Good() {
std::string str = "abc";
int i = 3;
std::shared_ptr<my_class> p = std::make_shared<my_class>(str, i);
}
void shared_ptr_with_std_string_Bad() {
std::string str = "abc";
int i = 8;
std::shared_ptr<my_class> p = std::make_shared<my_class>(str, i);
}
void shared_ptr_with_const_int_Good() {
const int i = 3;
std::shared_ptr<my_class> p = std::make_shared<my_class>(i);
}
void shared_ptr_with_const_int_Bad() {
const int i = 8;
std::shared_ptr<my_class> p = std::make_shared<my_class>(i);
}
}; };

Loading…
Cancel
Save