(* * 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. *) (** Make sure callbacks are always unregistered. drive the point home by reporting possible NPE's *) module L = Logging module F = Format module P = Printf module IdSet = Ident.IdentSet module FldSet = Ident.FieldSet open Utils (** Automatically create a harness method to exercise code under test *) (** return the set of instance fields that are assigned to a null literal in [procdesc] *) let get_fields_nullified procdesc = (* walk through the instructions and look for instance fields that are assigned to null *) let collect_nullified_flds (nullified_flds, this_ids) _ = function | Sil.Set (Sil.Lfield (Sil.Var lhs, fld, _), typ, rhs, loc) when Sil.exp_is_null_literal rhs && IdSet.mem lhs this_ids -> (FldSet.add fld nullified_flds, this_ids) | Sil.Letderef (id, rhs, _, _) when Sil.exp_is_this rhs -> (nullified_flds, IdSet.add id this_ids) | _ -> (nullified_flds, this_ids) in let (nullified_flds, _) = Cfg.Procdesc.fold_instrs collect_nullified_flds (FldSet.empty, IdSet.empty) procdesc in nullified_flds (** set of instance fields belonging to the current file that are assigned to null literals *) let fields_nullified = ref FldSet.empty (** set of callbacks registered in the current file *) let registered_callback_procs = ref Procname.Set.empty let android_lifecycle_typs = ref [] (** resolve the list of android lifecycle type strings in [tenv] *) let get_or_create_lifecycle_typs tenv = match !android_lifecycle_typs with | [] -> let lifecycle_typs = IList.fold_left (fun typs (pkg, clazz, methods) -> let qualified_name = Mangled.from_package_class pkg clazz in match AndroidFramework.get_lifecycle_for_framework_typ_opt qualified_name methods tenv with | Some (framework_typ, _) -> framework_typ :: typs | None -> typs ) [] AndroidFramework.get_lifecycles in android_lifecycle_typs := lifecycle_typs; lifecycle_typs | typs -> typs let do_eradicate_check all_procs get_procdesc idenv tenv proc_name proc_desc = Eradicate.callback_eradicate all_procs get_procdesc idenv tenv proc_name proc_desc let num_methods_checked = ref 0 let done_checking num_methods = incr num_methods_checked; !num_methods_checked = num_methods (** ask Eradicate to check each of the procs in [registered_callback_procs] (and their transitive * callees) in a context where each of the fields in [fields_nullifed] is marked as @Nullable *) let do_eradicate_check all_procs get_procdesc idenv tenv = (* tell Eradicate to treat each of the fields nullified in on_destroy as nullable *) FldSet.iter (fun fld -> Models.Inference.field_add_nullable_annotation fld) !fields_nullified; Procname.Set.iter (fun proc_name -> match get_procdesc proc_name with | Some proc_desc -> do_eradicate_check all_procs get_procdesc idenv tenv proc_name proc_desc | None -> ()) !registered_callback_procs (** if [procname] belongs to an Android lifecycle type, save the set of callbacks registered in * [procname]. in addition, if [procname] is a special "destroy" /"cleanup" method, save the set of * fields that are nullified *) let callback_checker_main all_procs get_procdesc idenv tenv proc_name proc_desc = let typename = Typename.TN_csu (Csu.Class Csu.Java, Mangled.from_string (Procname.java_get_class proc_name)) in match Sil.tenv_lookup tenv typename with | Some (Sil.Tstruct { Sil.csu; struct_name = Some class_name; def_methods } as typ) -> let lifecycle_typs = get_or_create_lifecycle_typs tenv in let proc_belongs_to_lifecycle_typ = IList.exists (fun lifecycle_typ -> AndroidFramework.typ_is_lifecycle_typ typ lifecycle_typ tenv) lifecycle_typs in if proc_belongs_to_lifecycle_typ then (* TODO (tt4959422): get all of the callbacks registered by callees as well *) let registered_callback_typs = AndroidFramework.get_callbacks_registered_by_proc proc_desc tenv in (* find the callbacks registered by this procedure and update the list *) let registered_callback_procs' = IList.fold_left (fun callback_procs callback_typ -> match callback_typ with | Sil.Tptr (Sil.Tstruct { Sil.struct_name = Some _; def_methods = def_methods'}, _) -> IList.fold_left (fun callback_procs callback_proc -> if Procname.is_constructor callback_proc then callback_procs else Procname.Set.add callback_proc callback_procs) callback_procs def_methods' | typ -> callback_procs) !registered_callback_procs registered_callback_typs in registered_callback_procs := registered_callback_procs'; let _ = if AndroidFramework.is_destroy_method proc_name then (* compute the set of fields nullified by this procedure *) (* TODO (t4959422): get fields that are nullified in callees of the destroy method *) fields_nullified := FldSet.union (get_fields_nullified proc_desc) !fields_nullified in if done_checking (IList.length def_methods) then do_eradicate_check all_procs get_procdesc idenv tenv | _ -> ()