/* * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ package codetoanalyze.java.checkers; import android.annotation.SuppressLint; import com.facebook.infer.annotation.Expensive; import com.facebook.infer.annotation.PerformanceCritical; interface I { void foo(); } class A implements I { @SuppressLint("CHECKERS_EXPENSIVE_OVERRIDES_UNANNOTATED") // Suppressing the sub-typing violation warning here as foo() is not annotated as @Expensive // in the interface. This report is legit but is not relevant for the current test. @Expensive public void foo() {} } class B extends A implements I { public void foo() {} } public class ExpensiveInheritanceExample { // The objective of this test is to document the limitations of the checker, which just // implements a type system. This means that the checker is flow insensitive and is only based // on the static type information. Especially, it does not try to resolve dynamic dispatch. // However, the checker is still exhaustive thanks to the sub-typing rule for // the @Expensive annotation. @PerformanceCritical void shouldNotReportBecauseInterfaceNotAnnotated(I i) { i.foo(); } @PerformanceCritical void reportsBecauseFooIsExpensiveInA(A a) { a.foo(); } @PerformanceCritical void doesNotreportBecauseFooIsNotExpensiveInB(B b) { b.foo(); } native B createB(); A actuallyReturnsObjectOfTypeB() { return createB(); } @PerformanceCritical void reportsAssumingObjectOfTypeA() { A a = actuallyReturnsObjectOfTypeB(); a.foo(); } @PerformanceCritical void doesReportBecauseTypeFlowInsensitive(A a) { if (a instanceof B) { a.foo(); } } }