[cost] Support a specific form of linked list iteration

Summary:
This diff tries to support a specific form of linked list iteration in Java.

```
while (p != null) {
    p = p.getNext();
}
```
This example was a constant cost before because the cost checker could not detect that it is an iteration on a linked list.

The heuristic this diff implemented is:

(1) `p = p.getNext()`: It tries to find this specific form of assignment. Then, it increments `p.linked_list_index` by 1. Note that `linked_list_index` is a virtual field for keeping an index in the linked list. Its initial value is always 0.

(2) At `p != null`, it tries to prune the value of `p.linked_list_index`: the upper-bound of `p.linked_list_index` is pruned by `<= p.linked_list_length`. Here again, `p.linked_list_length` is also a virtual field to denote the length of the linked list.

Reviewed By: ezgicicek

Differential Revision: D22234892

fbshipit-source-id: 2fee176bb
master
Sungkeun Cho 5 years ago committed by Facebook GitHub Bot
parent 3a338e6960
commit b48534c640

@ -379,6 +379,19 @@ module Loc = struct
append_field l fn ~typ append_field l fn ~typ
| BoField.(StarField _ | Prim (Var _ | Allocsite _)) -> | BoField.(StarField _ | Prim (Var _ | Allocsite _)) ->
x x
let get_linked_list_next ~lhs ~rhs =
match (get_path lhs, get_path rhs) with
| ( Some lhs_path
, Some
(Prim
(Deref (Deref_JavaPointer, Field {prefix= Prim (Deref (Deref_JavaPointer, rhs_path))})))
)
when Symb.SymbolPath.equal_partial lhs_path rhs_path ->
Some lhs
| _, _ ->
None
end end
module LocSet = PrettyPrintable.MakePPSet (Loc) module LocSet = PrettyPrintable.MakePPSet (Loc)
@ -582,6 +595,14 @@ module PowLoc = struct
LocSet.singleton Loc.unknown LocSet.singleton Loc.unknown
| Known ploc -> | Known ploc ->
ploc ploc
let get_linked_list_next ~lhs ~rhs =
match (is_singleton_or_more lhs, is_singleton_or_more rhs) with
| Singleton lhs, Singleton rhs ->
Loc.get_linked_list_next ~lhs ~rhs
| _, _ ->
None
end end
let always_strong_update = false let always_strong_update = false

@ -153,6 +153,11 @@ module PowLoc : sig
[Boolean.EqualOrder.ne], etc. *) [Boolean.EqualOrder.ne], etc. *)
val to_set : t -> LocSet.t val to_set : t -> LocSet.t
val get_linked_list_next : lhs:t -> rhs:t -> Loc.t option
(** It checks whether [rhs] is of [lhs.any_field], which is a heuristic for detecting a linked
list, e.g. [x = x.next()]. It returns [Some lhs] if the condition is satisfied, [None]
otherwise. *)
end end
val can_strong_update : PowLoc.t -> bool val can_strong_update : PowLoc.t -> bool

@ -243,6 +243,14 @@ module TransferFunctions = struct
mem ) mem )
let java_store_linked_list_next locs v mem =
PowLoc.get_linked_list_next ~lhs:locs ~rhs:(Dom.Val.get_all_locs v)
|> Option.value_map ~default:mem ~f:(fun loc ->
let linked_list_index = Loc.append_field loc BufferOverrunField.java_linked_list_index in
let v = Dom.Mem.find linked_list_index mem |> Dom.Val.plus_a Dom.Val.Itv.one in
Dom.Mem.add_heap linked_list_index v mem )
let modeled_load_of_empty_collection_opt = let modeled_load_of_empty_collection_opt =
let known_empty_collections = String.Set.of_list ["EMPTY_LIST"; "EMPTY_SET"; "EMPTY_MAP"] in let known_empty_collections = String.Set.of_list ["EMPTY_LIST"; "EMPTY_SET"; "EMPTY_MAP"] in
fun exp model_env ret mem -> fun exp model_env ret mem ->
@ -355,6 +363,7 @@ module TransferFunctions = struct
Sem.eval integer_type_widths exp2 mem |> Dom.Val.add_assign_trace_elem location locs Sem.eval integer_type_widths exp2 mem |> Dom.Val.add_assign_trace_elem location locs
in in
let mem = Dom.Mem.update_mem locs v mem in let mem = Dom.Mem.update_mem locs v mem in
let mem = java_store_linked_list_next locs v mem in
let mem = let mem =
if Language.curr_language_is Clang && Typ.is_char typ then if Language.curr_language_is Clang && Typ.is_char typ then
BoUtils.Exec.set_c_strlen ~tgt:(Sem.eval integer_type_widths exp1 mem) ~src:v mem BoUtils.Exec.set_c_strlen ~tgt:(Sem.eval integer_type_widths exp1 mem) ~src:v mem

@ -634,6 +634,11 @@ module Val = struct
| Some s -> | Some s ->
of_itv (Itv.of_int (String.length s)) of_itv (Itv.of_int (String.length s))
| None -> ( | None -> (
match l with
| Field {fn} when Fieldname.equal fn BufferOverrunField.java_linked_list_index ->
L.d_printfln_escaped "Val.on_demand for %a as zero" Loc.pp l ;
of_itv Itv.zero
| _ -> (
match Loc.get_path l with match Loc.get_path l with
| None -> | None ->
L.d_printfln_escaped "Val.on_demand for %a -> no path" Loc.pp l ; L.d_printfln_escaped "Val.on_demand for %a -> no path" Loc.pp l ;
@ -650,7 +655,7 @@ module Val = struct
default ) default )
| Some typ -> | Some typ ->
L.d_printfln_escaped "Val.on_demand for %a" Loc.pp l ; L.d_printfln_escaped "Val.on_demand for %a" Loc.pp l ;
do_on_demand path typ ) ) ) do_on_demand path typ ) ) ) )
module Itv = struct module Itv = struct
@ -667,6 +672,8 @@ module Val = struct
let unknown_bool = of_itv Itv.unknown_bool let unknown_bool = of_itv Itv.unknown_bool
let zero = of_itv Itv.zero let zero = of_itv Itv.zero
let one = of_itv Itv.one
end end
end end
@ -705,6 +712,11 @@ module MemPure = struct
let bot = empty let bot = empty
let get_linked_list_index loc mem =
let linked_list_index = Loc.append_field loc BufferOverrunField.java_linked_list_index in
Option.map (find_opt linked_list_index mem) ~f:(fun (_, v) -> (linked_list_index, v))
let range : let range :
filter_loc:(Loc.t -> LoopHeadLoc.t option) filter_loc:(Loc.t -> LoopHeadLoc.t option)
-> node_id:ProcCfg.Normal.Node.id -> node_id:ProcCfg.Normal.Node.id
@ -715,6 +727,7 @@ module MemPure = struct
(fun loc (_, v) acc -> (fun loc (_, v) acc ->
match filter_loc loc with match filter_loc loc with
| Some loop_head_loc -> ( | Some loop_head_loc -> (
let loc, v = Option.value (get_linked_list_index loc mem) ~default:(loc, v) in
let itv_updated_by = Val.get_itv_updated_by v in let itv_updated_by = Val.get_itv_updated_by v in
match itv_updated_by with match itv_updated_by with
| Addition | Multiplication -> | Addition | Multiplication ->

@ -274,6 +274,9 @@ module Val : sig
val zero : t val zero : t
(** [\[0,0\]] *) (** [\[0,0\]] *)
val one : t
(** [\[1,1\]] *)
val zero_255 : t val zero_255 : t
(** [\[0,255\]] *) (** [\[0,255\]] *)

@ -31,6 +31,10 @@ let mk, get_type =
let java_collection_internal_array = mk "java.collection.elements" Typ.(mk_array void) let java_collection_internal_array = mk "java.collection.elements" Typ.(mk_array void)
let java_linked_list_index = mk "java.linked_list_index" Typ.(int)
let java_linked_list_length = mk "java.linked_list_length" Typ.(int)
let is_java_collection_internal_array fn = Fieldname.equal fn java_collection_internal_array let is_java_collection_internal_array fn = Fieldname.equal fn java_collection_internal_array
let c_strlen () = let c_strlen () =

@ -29,6 +29,12 @@ val cpp_vector_elem : vec_typ:Typ.t -> elt_typ:Typ.t -> Fieldname.t
val java_collection_internal_array : Fieldname.t val java_collection_internal_array : Fieldname.t
(** Field for Java collection's elements *) (** Field for Java collection's elements *)
val java_linked_list_index : Fieldname.t
(** Virtual field for index of Java's linked list *)
val java_linked_list_length : Fieldname.t
(** Virtual field for length of Java's linked list *)
val is_cpp_vector_elem : Fieldname.t -> bool val is_cpp_vector_elem : Fieldname.t -> bool
(** Check if the field is for C++ vector's elements *) (** Check if the field is for C++ vector's elements *)

@ -564,6 +564,19 @@ module Prune = struct
AliasTargets.fold accum_pruned (Mem.find_alias_loc iterator mem) astate AliasTargets.fold accum_pruned (Mem.find_alias_loc iterator mem) astate
let prune_linked_list_index loc mem acc =
let lv_linked_list_index = Loc.append_field loc BufferOverrunField.java_linked_list_index in
Option.value_map (Mem.find_opt lv_linked_list_index mem) ~default:acc ~f:(fun index_v ->
let linked_list_length = Loc.append_field loc BufferOverrunField.java_linked_list_length in
Option.value_map (Loc.get_path linked_list_length) ~default:acc
~f:(fun linked_list_length ->
let pruned_v =
Val.of_itv (Itv.of_normal_path ~unsigned:true linked_list_length)
|> Val.prune_binop Le index_v
in
update_mem_in_prune lv_linked_list_index pruned_v acc ) )
let prune_unop : Exp.t -> t -> t = let prune_unop : Exp.t -> t -> t =
fun e ({mem} as astate) -> fun e ({mem} as astate) ->
match e with match e with
@ -571,11 +584,12 @@ module Prune = struct
let accum_prune_var rhs tgt acc = let accum_prune_var rhs tgt acc =
match tgt with match tgt with
| AliasTarget.Simple {i} when IntLit.iszero i -> | AliasTarget.Simple {i} when IntLit.iszero i ->
let v = Mem.find rhs mem in (let v = Mem.find rhs mem in
if Val.is_bot v then acc if Val.is_bot v then acc
else else
let v' = Val.prune_ne_zero v in let v' = Val.prune_ne_zero v in
update_mem_in_prune rhs v' acc update_mem_in_prune rhs v' acc )
|> prune_linked_list_index rhs mem
| AliasTarget.Empty -> | AliasTarget.Empty ->
let v = Mem.find rhs mem in let v = Mem.find rhs mem in
if Val.is_bot v then acc if Val.is_bot v then acc

@ -148,6 +148,9 @@ val top : t
val zero : t val zero : t
(** 0 *) (** 0 *)
val one : t
(** 1 *)
val zero_one : t val zero_one : t
(** [0, 1] *) (** [0, 1] *)

@ -136,6 +136,12 @@ public class Loops {
} }
} }
void length_of_linked_list_simple_linear(MyLinkedList p) {
while (p != null) {
p = p.getNext();
}
}
void length_of_linked_list_linear_FP(MyLinkedList p) { void length_of_linked_list_linear_FP(MyLinkedList p) {
int n = 0; int n = 0;
while (p != null) { while (p != null) {

@ -269,6 +269,7 @@ codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.dumbSort_quadratic(long[],long[],int):void, 5 + 59 ⋅ (length - 1) × (length - 1) + 8 ⋅ length, OnUIThread:false, [{length},Loop at line 54,{length - 1},Loop at line 55,{length - 1},Loop at line 54] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.dumbSort_quadratic(long[],long[],int):void, 5 + 59 ⋅ (length - 1) × (length - 1) + 8 ⋅ length, OnUIThread:false, [{length},Loop at line 54,{length - 1},Loop at line 55,{length - 1},Loop at line 54]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.dumb_linear(long[],int):void, 5 + 25 ⋅ (length - 1), OnUIThread:false, [{length - 1},Loop at line 44] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.dumb_linear(long[],int):void, 5 + 25 ⋅ (length - 1), OnUIThread:false, [{length - 1},Loop at line 44]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, , OnUIThread:false, [Unbounded value x,call to void Loops.loop_linear(int),Loop at line 86] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, , OnUIThread:false, [Unbounded value x,call to void Loops.loop_linear(int),Loop at line 86]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_simple_linear(codetoanalyze.java.performance.Loops$MyLinkedList):void, 3 + 10 ⋅ (p.linked_list_length + 1), OnUIThread:false, [{p.linked_list_length + 1},Loop at line 140]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.loop_linear(int):void, 5 + 5 ⋅ x, OnUIThread:false, [{x},Loop at line 86] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.loop_linear(int):void, 5 + 5 ⋅ x, OnUIThread:false, [{x},Loop at line 86]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.modeled_range_linear(java.nio.channels.FileChannel,java.nio.ByteBuffer):void, 6 + 14 ⋅ FileChannel.read(...).modeled, OnUIThread:false, [{FileChannel.read(...).modeled},Modeled call to read(...)] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.modeled_range_linear(java.nio.channels.FileChannel,java.nio.ByteBuffer):void, 6 + 14 ⋅ FileChannel.read(...).modeled, OnUIThread:false, [{FileChannel.read(...).modeled},Modeled call to read(...)]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, , OnUIThread:false, [Unbounded loop,Loop at line 34] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, , OnUIThread:false, [Unbounded loop,Loop at line 34]

@ -64,6 +64,7 @@ codetoanalyze/java/performance/JsonUtils.java, libraries.marauder.analytics.util
codetoanalyze/java/performance/JsonUtils.java, libraries.marauder.analytics.utils.json.JsonUtils.serialize(java.lang.StringBuilder,java.lang.String):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Call to void JsonUtils.escape(StringBuilder,String),Unbounded loop,Loop at line 13] codetoanalyze/java/performance/JsonUtils.java, libraries.marauder.analytics.utils.json.JsonUtils.serialize(java.lang.StringBuilder,java.lang.String):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Call to void JsonUtils.escape(StringBuilder,String),Unbounded loop,Loop at line 13]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Unbounded value x,call to void Loops.loop_linear(int),Loop at line 86] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Unbounded value x,call to void Loops.loop_linear(int),Loop at line 86]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, 3, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 1):signed32] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, 3, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 1):signed32]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.length_of_linked_list_linear_FP(codetoanalyze.java.performance.Loops$MyLinkedList):void, 6, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [Assignment,Call,<LHS trace>,Parameter `x`,Binary operation: ([0, +oo] + 1):signed32 by call to `void Loops.loop_linear(int)` ]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.modeled_range_linear(java.nio.channels.FileChannel,java.nio.ByteBuffer):void, 9, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 8):signed32] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.modeled_range_linear(java.nio.channels.FileChannel,java.nio.ByteBuffer):void, 9, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 8):signed32]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Unbounded loop,Loop at line 34] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, 0, INFINITE_EXECUTION_TIME, no_bucket, ERROR, [Unbounded loop,Loop at line 34]
codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, 8, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 1):signed32] codetoanalyze/java/performance/Loops.java, codetoanalyze.java.performance.Loops.nested_do_while_FP(int):void, 8, INTEGER_OVERFLOW_L5, no_bucket, ERROR, [<LHS trace>,Assignment,Binary operation: ([0, +oo] + 1):signed32]

Loading…
Cancel
Save