diff options
Diffstat (limited to 'compiler/optimizing/inliner.cc')
| -rw-r--r-- | compiler/optimizing/inliner.cc | 352 | 
1 files changed, 280 insertions, 72 deletions
| diff --git a/compiler/optimizing/inliner.cc b/compiler/optimizing/inliner.cc index 68e96fba74..02a1acc240 100644 --- a/compiler/optimizing/inliner.cc +++ b/compiler/optimizing/inliner.cc @@ -190,28 +190,34 @@ static uint32_t FindMethodIndexIn(ArtMethod* method,    }  } -static uint32_t FindClassIndexIn(mirror::Class* cls, const DexFile& dex_file) +static uint32_t FindClassIndexIn(mirror::Class* cls, +                                 const DexFile& dex_file, +                                 Handle<mirror::DexCache> dex_cache)      SHARED_REQUIRES(Locks::mutator_lock_) { +  uint32_t index = DexFile::kDexNoIndex;    if (cls->GetDexCache() == nullptr) { -    DCHECK(cls->IsArrayClass()); -    // TODO: find the class in `dex_file`. -    return DexFile::kDexNoIndex; +    DCHECK(cls->IsArrayClass()) << PrettyClass(cls); +    index = cls->FindTypeIndexInOtherDexFile(dex_file);    } else if (cls->GetDexTypeIndex() == DexFile::kDexNoIndex16) { +    DCHECK(cls->IsProxyClass()) << PrettyClass(cls);      // TODO: deal with proxy classes. -    return DexFile::kDexNoIndex;    } else if (IsSameDexFile(cls->GetDexFile(), dex_file)) { +    index = cls->GetDexTypeIndex(); +  } else { +    index = cls->FindTypeIndexInOtherDexFile(dex_file); +  } + +  if (index != DexFile::kDexNoIndex) {      // Update the dex cache to ensure the class is in. The generated code will      // consider it is. We make it safe by updating the dex cache, as other      // dex files might also load the class, and there is no guarantee the dex      // cache of the dex file of the class will be updated. -    if (cls->GetDexCache()->GetResolvedType(cls->GetDexTypeIndex()) == nullptr) { -      cls->GetDexCache()->SetResolvedType(cls->GetDexTypeIndex(), cls); +    if (dex_cache->GetResolvedType(index) == nullptr) { +      dex_cache->SetResolvedType(index, cls);      } -    return cls->GetDexTypeIndex(); -  } else { -    // TODO: find the class in `dex_file`. -    return DexFile::kDexNoIndex;    } + +  return index;  }  bool HInliner::TryInline(HInvoke* invoke_instruction) { @@ -258,8 +264,9 @@ bool HInliner::TryInline(HInvoke* invoke_instruction) {    }    if (actual_method != nullptr) { -    return TryInline(invoke_instruction, actual_method); +    return TryInlineAndReplace(invoke_instruction, actual_method, /* do_rtp */ true);    } +    DCHECK(!invoke_instruction->IsInvokeStaticOrDirect());    // Check if we can use an inline cache. @@ -302,7 +309,7 @@ HInstanceFieldGet* HInliner::BuildGetReceiverClass(ClassLinker* class_linker,                                                     uint32_t dex_pc) const {    ArtField* field = class_linker->GetClassRoot(ClassLinker::kJavaLangObject)->GetInstanceField(0);    DCHECK_EQ(std::string(field->GetName()), "shadow$_klass_"); -  return new (graph_->GetArena()) HInstanceFieldGet( +  HInstanceFieldGet* result = new (graph_->GetArena()) HInstanceFieldGet(        receiver,        Primitive::kPrimNot,        field->GetOffset(), @@ -312,6 +319,9 @@ HInstanceFieldGet* HInliner::BuildGetReceiverClass(ClassLinker* class_linker,        *field->GetDexFile(),        handles_->NewHandle(field->GetDexCache()),        dex_pc); +  // The class of a field is effectively final, and does not have any memory dependencies. +  result->SetSideEffects(SideEffects::None()); +  return result;  }  bool HInliner::TryInlineMonomorphicCall(HInvoke* invoke_instruction, @@ -321,7 +331,8 @@ bool HInliner::TryInlineMonomorphicCall(HInvoke* invoke_instruction,        << invoke_instruction->DebugName();    const DexFile& caller_dex_file = *caller_compilation_unit_.GetDexFile(); -  uint32_t class_index = FindClassIndexIn(ic.GetMonomorphicType(), caller_dex_file); +  uint32_t class_index = FindClassIndexIn( +      ic.GetMonomorphicType(), caller_dex_file, caller_compilation_unit_.GetDexCache());    if (class_index == DexFile::kDexNoIndex) {      VLOG(compiler) << "Call to " << PrettyMethod(resolved_method)                     << " from inline cache is not inlined because its class is not" @@ -344,16 +355,44 @@ bool HInliner::TryInlineMonomorphicCall(HInvoke* invoke_instruction,    HInstruction* cursor = invoke_instruction->GetPrevious();    HBasicBlock* bb_cursor = invoke_instruction->GetBlock(); -  if (!TryInline(invoke_instruction, resolved_method, /* do_rtp */ false)) { +  if (!TryInlineAndReplace(invoke_instruction, resolved_method, /* do_rtp */ false)) {      return false;    }    // We successfully inlined, now add a guard. +  bool is_referrer = +      (ic.GetMonomorphicType() == outermost_graph_->GetArtMethod()->GetDeclaringClass()); +  AddTypeGuard(receiver, +               cursor, +               bb_cursor, +               class_index, +               is_referrer, +               invoke_instruction, +               /* with_deoptimization */ true); + +  // Run type propagation to get the guard typed, and eventually propagate the +  // type of the receiver. +  ReferenceTypePropagation rtp_fixup(graph_, handles_, /* is_first_run */ false); +  rtp_fixup.Run(); + +  MaybeRecordStat(kInlinedMonomorphicCall); +  return true; +} + +HInstruction* HInliner::AddTypeGuard(HInstruction* receiver, +                                     HInstruction* cursor, +                                     HBasicBlock* bb_cursor, +                                     uint32_t class_index, +                                     bool is_referrer, +                                     HInstruction* invoke_instruction, +                                     bool with_deoptimization) { +  ClassLinker* class_linker = caller_compilation_unit_.GetClassLinker();    HInstanceFieldGet* receiver_class = BuildGetReceiverClass(        class_linker, receiver, invoke_instruction->GetDexPc()); -  bool is_referrer = -      (ic.GetMonomorphicType() == outermost_graph_->GetArtMethod()->GetDeclaringClass()); +  const DexFile& caller_dex_file = *caller_compilation_unit_.GetDexFile(); +  // Note that we will just compare the classes, so we don't need Java semantics access checks. +  // Also, the caller of `AddTypeGuard` must have guaranteed that the class is in the dex cache.    HLoadClass* load_class = new (graph_->GetArena()) HLoadClass(graph_->GetCurrentMethod(),                                                                 class_index,                                                                 caller_dex_file, @@ -363,8 +402,6 @@ bool HInliner::TryInlineMonomorphicCall(HInvoke* invoke_instruction,                                                                 /* is_in_dex_cache */ true);    HNotEqual* compare = new (graph_->GetArena()) HNotEqual(load_class, receiver_class); -  HDeoptimize* deoptimize = new (graph_->GetArena()) HDeoptimize( -      compare, invoke_instruction->GetDexPc());    // TODO: Extend reference type propagation to understand the guard.    if (cursor != nullptr) {      bb_cursor->InsertInstructionAfter(receiver_class, cursor); @@ -373,16 +410,13 @@ bool HInliner::TryInlineMonomorphicCall(HInvoke* invoke_instruction,    }    bb_cursor->InsertInstructionAfter(load_class, receiver_class);    bb_cursor->InsertInstructionAfter(compare, load_class); -  bb_cursor->InsertInstructionAfter(deoptimize, compare); -  deoptimize->CopyEnvironmentFrom(invoke_instruction->GetEnvironment()); - -  // Run type propagation to get the guard typed, and eventually propagate the -  // type of the receiver. -  ReferenceTypePropagation rtp_fixup(graph_, handles_, /* is_first_run */ false); -  rtp_fixup.Run(); - -  MaybeRecordStat(kInlinedMonomorphicCall); -  return true; +  if (with_deoptimization) { +    HDeoptimize* deoptimize = new (graph_->GetArena()) HDeoptimize( +        compare, invoke_instruction->GetDexPc()); +    bb_cursor->InsertInstructionAfter(deoptimize, compare); +    deoptimize->CopyEnvironmentFrom(invoke_instruction->GetEnvironment()); +  } +  return compare;  }  bool HInliner::TryInlinePolymorphicCall(HInvoke* invoke_instruction, @@ -390,6 +424,174 @@ bool HInliner::TryInlinePolymorphicCall(HInvoke* invoke_instruction,                                          const InlineCache& ic) {    DCHECK(invoke_instruction->IsInvokeVirtual() || invoke_instruction->IsInvokeInterface())        << invoke_instruction->DebugName(); + +  if (TryInlinePolymorphicCallToSameTarget(invoke_instruction, resolved_method, ic)) { +    return true; +  } + +  ClassLinker* class_linker = caller_compilation_unit_.GetClassLinker(); +  size_t pointer_size = class_linker->GetImagePointerSize(); +  const DexFile& caller_dex_file = *caller_compilation_unit_.GetDexFile(); + +  bool all_targets_inlined = true; +  bool one_target_inlined = false; +  for (size_t i = 0; i < InlineCache::kIndividualCacheSize; ++i) { +    if (ic.GetTypeAt(i) == nullptr) { +      break; +    } +    ArtMethod* method = nullptr; +    if (invoke_instruction->IsInvokeInterface()) { +      method = ic.GetTypeAt(i)->FindVirtualMethodForInterface( +          resolved_method, pointer_size); +    } else { +      DCHECK(invoke_instruction->IsInvokeVirtual()); +      method = ic.GetTypeAt(i)->FindVirtualMethodForVirtual( +          resolved_method, pointer_size); +    } + +    HInstruction* receiver = invoke_instruction->InputAt(0); +    HInstruction* cursor = invoke_instruction->GetPrevious(); +    HBasicBlock* bb_cursor = invoke_instruction->GetBlock(); + +    uint32_t class_index = FindClassIndexIn( +        ic.GetTypeAt(i), caller_dex_file, caller_compilation_unit_.GetDexCache()); +    HInstruction* return_replacement = nullptr; +    if (class_index == DexFile::kDexNoIndex || +        !TryBuildAndInline(invoke_instruction, method, &return_replacement)) { +      all_targets_inlined = false; +    } else { +      one_target_inlined = true; +      bool is_referrer = (ic.GetTypeAt(i) == outermost_graph_->GetArtMethod()->GetDeclaringClass()); + +      // If we have inlined all targets before, and this receiver is the last seen, +      // we deoptimize instead of keeping the original invoke instruction. +      bool deoptimize = all_targets_inlined && +          (i != InlineCache::kIndividualCacheSize - 1) && +          (ic.GetTypeAt(i + 1) == nullptr); +      HInstruction* compare = AddTypeGuard( +          receiver, cursor, bb_cursor, class_index, is_referrer, invoke_instruction, deoptimize); +      if (deoptimize) { +        if (return_replacement != nullptr) { +          invoke_instruction->ReplaceWith(return_replacement); +        } +        invoke_instruction->GetBlock()->RemoveInstruction(invoke_instruction); +        // Because the inline cache data can be populated concurrently, we force the end of the +        // iteration. Otherhwise, we could see a new receiver type. +        break; +      } else { +        CreateDiamondPatternForPolymorphicInline(compare, return_replacement, invoke_instruction); +      } +    } +  } + +  if (!one_target_inlined) { +    VLOG(compiler) << "Call to " << PrettyMethod(resolved_method) +                   << " from inline cache is not inlined because none" +                   << " of its targets could be inlined"; +    return false; +  } +  MaybeRecordStat(kInlinedPolymorphicCall); + +  // Run type propagation to get the guards typed. +  ReferenceTypePropagation rtp_fixup(graph_, handles_, /* is_first_run */ false); +  rtp_fixup.Run(); +  return true; +} + +void HInliner::CreateDiamondPatternForPolymorphicInline(HInstruction* compare, +                                                        HInstruction* return_replacement, +                                                        HInstruction* invoke_instruction) { +  uint32_t dex_pc = invoke_instruction->GetDexPc(); +  HBasicBlock* cursor_block = compare->GetBlock(); +  HBasicBlock* original_invoke_block = invoke_instruction->GetBlock(); +  ArenaAllocator* allocator = graph_->GetArena(); + +  // Spit the block after the compare: `cursor_block` will now be the start of the diamond, +  // and the returned block is the start of the then branch (that could contain multiple blocks). +  HBasicBlock* then = cursor_block->SplitAfterForInlining(compare); + +  // Split the block containing the invoke before and after the invoke. The returned block +  // of the split before will contain the invoke and will be the otherwise branch of +  // the diamond. The returned block of the split after will be the merge block +  // of the diamond. +  HBasicBlock* end_then = invoke_instruction->GetBlock(); +  HBasicBlock* otherwise = end_then->SplitBeforeForInlining(invoke_instruction); +  HBasicBlock* merge = otherwise->SplitAfterForInlining(invoke_instruction); + +  // If the methods we are inlining return a value, we create a phi in the merge block +  // that will have the `invoke_instruction and the `return_replacement` as inputs. +  if (return_replacement != nullptr) { +    HPhi* phi = new (allocator) HPhi( +        allocator, kNoRegNumber, 0, HPhi::ToPhiType(invoke_instruction->GetType()), dex_pc); +    merge->AddPhi(phi); +    invoke_instruction->ReplaceWith(phi); +    phi->AddInput(return_replacement); +    phi->AddInput(invoke_instruction); +  } + +  // Add the control flow instructions. +  otherwise->AddInstruction(new (allocator) HGoto(dex_pc)); +  end_then->AddInstruction(new (allocator) HGoto(dex_pc)); +  cursor_block->AddInstruction(new (allocator) HIf(compare, dex_pc)); + +  // Add the newly created blocks to the graph. +  graph_->AddBlock(then); +  graph_->AddBlock(otherwise); +  graph_->AddBlock(merge); + +  // Set up successor (and implictly predecessor) relations. +  cursor_block->AddSuccessor(otherwise); +  cursor_block->AddSuccessor(then); +  end_then->AddSuccessor(merge); +  otherwise->AddSuccessor(merge); + +  // Set up dominance information. +  then->SetDominator(cursor_block); +  cursor_block->AddDominatedBlock(then); +  otherwise->SetDominator(cursor_block); +  cursor_block->AddDominatedBlock(otherwise); +  merge->SetDominator(cursor_block); +  cursor_block->AddDominatedBlock(merge); + +  // Update the revert post order. +  size_t index = IndexOfElement(graph_->reverse_post_order_, cursor_block); +  MakeRoomFor(&graph_->reverse_post_order_, 1, index); +  graph_->reverse_post_order_[++index] = then; +  index = IndexOfElement(graph_->reverse_post_order_, end_then); +  MakeRoomFor(&graph_->reverse_post_order_, 2, index); +  graph_->reverse_post_order_[++index] = otherwise; +  graph_->reverse_post_order_[++index] = merge; + +  // Set the loop information of the newly created blocks. +  HLoopInformation* loop_info = cursor_block->GetLoopInformation(); +  if (loop_info != nullptr) { +    then->SetLoopInformation(cursor_block->GetLoopInformation()); +    merge->SetLoopInformation(cursor_block->GetLoopInformation()); +    otherwise->SetLoopInformation(cursor_block->GetLoopInformation()); +    for (HLoopInformationOutwardIterator loop_it(*cursor_block); +         !loop_it.Done(); +         loop_it.Advance()) { +      loop_it.Current()->Add(then); +      loop_it.Current()->Add(merge); +      loop_it.Current()->Add(otherwise); +    } +    // In case the original invoke location was a back edge, we need to update +    // the loop to now have the merge block as a back edge. +    if (loop_info->IsBackEdge(*original_invoke_block)) { +      loop_info->RemoveBackEdge(original_invoke_block); +      loop_info->AddBackEdge(merge); +    } +  } + +  // Set the try/catch information of the newly created blocks. +  then->SetTryCatchInformation(cursor_block->GetTryCatchInformation()); +  merge->SetTryCatchInformation(cursor_block->GetTryCatchInformation()); +  otherwise->SetTryCatchInformation(cursor_block->GetTryCatchInformation()); +} + +bool HInliner::TryInlinePolymorphicCallToSameTarget(HInvoke* invoke_instruction, +                                                    ArtMethod* resolved_method, +                                                    const InlineCache& ic) {    // This optimization only works under JIT for now.    DCHECK(Runtime::Current()->UseJit());    if (graph_->GetInstructionSet() == kMips64) { @@ -431,7 +633,7 @@ bool HInliner::TryInlinePolymorphicCall(HInvoke* invoke_instruction,    HInstruction* cursor = invoke_instruction->GetPrevious();    HBasicBlock* bb_cursor = invoke_instruction->GetBlock(); -  if (!TryInline(invoke_instruction, actual_method, /* do_rtp */ false)) { +  if (!TryInlineAndReplace(invoke_instruction, actual_method, /* do_rtp */ false)) {      return false;    } @@ -485,14 +687,29 @@ bool HInliner::TryInlinePolymorphicCall(HInvoke* invoke_instruction,    return true;  } -bool HInliner::TryInline(HInvoke* invoke_instruction, ArtMethod* method, bool do_rtp) { +bool HInliner::TryInlineAndReplace(HInvoke* invoke_instruction, ArtMethod* method, bool do_rtp) { +  HInstruction* return_replacement = nullptr; +  if (!TryBuildAndInline(invoke_instruction, method, &return_replacement)) { +    return false; +  } +  if (return_replacement != nullptr) { +    invoke_instruction->ReplaceWith(return_replacement); +  } +  invoke_instruction->GetBlock()->RemoveInstruction(invoke_instruction); +  FixUpReturnReferenceType(invoke_instruction, method, return_replacement, do_rtp); +  return true; +} + +bool HInliner::TryBuildAndInline(HInvoke* invoke_instruction, +                                 ArtMethod* method, +                                 HInstruction** return_replacement) {    const DexFile& caller_dex_file = *caller_compilation_unit_.GetDexFile();    // Check whether we're allowed to inline. The outermost compilation unit is the relevant    // dex file here (though the transitivity of an inline chain would allow checking the calller).    if (!compiler_driver_->MayInline(method->GetDexFile(),                                     outer_compilation_unit_.GetDexFile())) { -    if (TryPatternSubstitution(invoke_instruction, method, do_rtp)) { +    if (TryPatternSubstitution(invoke_instruction, method, return_replacement)) {        VLOG(compiler) << "Successfully replaced pattern of invoke " << PrettyMethod(method);        MaybeRecordStat(kReplacedInvokeWithSimplePattern);        return true; @@ -541,8 +758,9 @@ bool HInliner::TryInline(HInvoke* invoke_instruction, ArtMethod* method, bool do    if (!method->GetDeclaringClass()->IsVerified()) {      uint16_t class_def_idx = method->GetDeclaringClass()->GetDexClassDefIndex(); -    if (!compiler_driver_->IsMethodVerifiedWithoutFailures( -          method->GetDexMethodIndex(), class_def_idx, *method->GetDexFile())) { +    if (Runtime::Current()->UseJit() || +        !compiler_driver_->IsMethodVerifiedWithoutFailures( +            method->GetDexMethodIndex(), class_def_idx, *method->GetDexFile())) {        VLOG(compiler) << "Method " << PrettyMethod(method_index, caller_dex_file)                       << " couldn't be verified, so it cannot be inlined";        return false; @@ -559,7 +777,7 @@ bool HInliner::TryInline(HInvoke* invoke_instruction, ArtMethod* method, bool do      return false;    } -  if (!TryBuildAndInline(method, invoke_instruction, same_dex_file, do_rtp)) { +  if (!TryBuildAndInlineHelper(invoke_instruction, method, same_dex_file, return_replacement)) {      return false;    } @@ -586,27 +804,27 @@ static HInstruction* GetInvokeInputForArgVRegIndex(HInvoke* invoke_instruction,  // Try to recognize known simple patterns and replace invoke call with appropriate instructions.  bool HInliner::TryPatternSubstitution(HInvoke* invoke_instruction,                                        ArtMethod* resolved_method, -                                      bool do_rtp) { +                                      HInstruction** return_replacement) {    InlineMethod inline_method;    if (!InlineMethodAnalyser::AnalyseMethodCode(resolved_method, &inline_method)) {      return false;    } -  HInstruction* return_replacement = nullptr;    switch (inline_method.opcode) {      case kInlineOpNop:        DCHECK_EQ(invoke_instruction->GetType(), Primitive::kPrimVoid); +      *return_replacement = nullptr;        break;      case kInlineOpReturnArg: -      return_replacement = GetInvokeInputForArgVRegIndex(invoke_instruction, -                                                         inline_method.d.return_data.arg); +      *return_replacement = GetInvokeInputForArgVRegIndex(invoke_instruction, +                                                          inline_method.d.return_data.arg);        break;      case kInlineOpNonWideConst:        if (resolved_method->GetShorty()[0] == 'L') {          DCHECK_EQ(inline_method.d.data, 0u); -        return_replacement = graph_->GetNullConstant(); +        *return_replacement = graph_->GetNullConstant();        } else { -        return_replacement = graph_->GetIntConstant(static_cast<int32_t>(inline_method.d.data)); +        *return_replacement = graph_->GetIntConstant(static_cast<int32_t>(inline_method.d.data));        }        break;      case kInlineOpIGet: { @@ -621,7 +839,7 @@ bool HInliner::TryPatternSubstitution(HInvoke* invoke_instruction,        DCHECK_EQ(iget->GetFieldOffset().Uint32Value(), data.field_offset);        DCHECK_EQ(iget->IsVolatile() ? 1u : 0u, data.is_volatile);        invoke_instruction->GetBlock()->InsertInstructionBefore(iget, invoke_instruction); -      return_replacement = iget; +      *return_replacement = iget;        break;      }      case kInlineOpIPut: { @@ -639,7 +857,7 @@ bool HInliner::TryPatternSubstitution(HInvoke* invoke_instruction,        invoke_instruction->GetBlock()->InsertInstructionBefore(iput, invoke_instruction);        if (data.return_arg_plus1 != 0u) {          size_t return_arg = data.return_arg_plus1 - 1u; -        return_replacement = GetInvokeInputForArgVRegIndex(invoke_instruction, return_arg); +        *return_replacement = GetInvokeInputForArgVRegIndex(invoke_instruction, return_arg);        }        break;      } @@ -694,19 +912,13 @@ bool HInliner::TryPatternSubstitution(HInvoke* invoke_instruction,          HMemoryBarrier* barrier = new (graph_->GetArena()) HMemoryBarrier(kStoreStore, kNoDexPc);          invoke_instruction->GetBlock()->InsertInstructionBefore(barrier, invoke_instruction);        } +      *return_replacement = nullptr;        break;      }      default:        LOG(FATAL) << "UNREACHABLE";        UNREACHABLE();    } - -  if (return_replacement != nullptr) { -    invoke_instruction->ReplaceWith(return_replacement); -  } -  invoke_instruction->GetBlock()->RemoveInstruction(invoke_instruction); - -  FixUpReturnReferenceType(resolved_method, invoke_instruction, return_replacement, do_rtp);    return true;  } @@ -760,10 +972,10 @@ HInstanceFieldSet* HInliner::CreateInstanceFieldSet(Handle<mirror::DexCache> dex    return iput;  } -bool HInliner::TryBuildAndInline(ArtMethod* resolved_method, -                                 HInvoke* invoke_instruction, -                                 bool same_dex_file, -                                 bool do_rtp) { +bool HInliner::TryBuildAndInlineHelper(HInvoke* invoke_instruction, +                                       ArtMethod* resolved_method, +                                       bool same_dex_file, +                                       HInstruction** return_replacement) {    ScopedObjectAccess soa(Thread::Current());    const DexFile::CodeItem* code_item = resolved_method->GetCodeItem();    const DexFile& callee_dex_file = *resolved_method->GetDexFile(); @@ -771,16 +983,16 @@ bool HInliner::TryBuildAndInline(ArtMethod* resolved_method,    ClassLinker* class_linker = caller_compilation_unit_.GetClassLinker();    Handle<mirror::DexCache> dex_cache(handles_->NewHandle(resolved_method->GetDexCache()));    DexCompilationUnit dex_compilation_unit( -    nullptr, -    caller_compilation_unit_.GetClassLoader(), -    class_linker, -    callee_dex_file, -    code_item, -    resolved_method->GetDeclaringClass()->GetDexClassDefIndex(), -    method_index, -    resolved_method->GetAccessFlags(), -    compiler_driver_->GetVerifiedMethod(&callee_dex_file, method_index), -    dex_cache); +      nullptr, +      caller_compilation_unit_.GetClassLoader(), +      class_linker, +      callee_dex_file, +      code_item, +      resolved_method->GetDeclaringClass()->GetDexClassDefIndex(), +      method_index, +      resolved_method->GetAccessFlags(), +      /* verified_method */ nullptr, +      dex_cache);    bool requires_ctor_barrier = false; @@ -873,7 +1085,7 @@ bool HInliner::TryBuildAndInline(ArtMethod* resolved_method,    HConstantFolding fold(callee_graph);    HSharpening sharpening(callee_graph, codegen_, dex_compilation_unit, compiler_driver_);    InstructionSimplifier simplify(callee_graph, stats_); -  IntrinsicsRecognizer intrinsics(callee_graph, compiler_driver_); +  IntrinsicsRecognizer intrinsics(callee_graph, compiler_driver_, stats_);    HOptimization* optimizations[] = {      &intrinsics, @@ -1016,16 +1228,12 @@ bool HInliner::TryBuildAndInline(ArtMethod* resolved_method,    }    number_of_inlined_instructions_ += number_of_instructions; -  HInstruction* return_replacement = callee_graph->InlineInto(graph_, invoke_instruction); -  if (return_replacement != nullptr) { -    DCHECK_EQ(graph_, return_replacement->GetBlock()->GetGraph()); -  } -  FixUpReturnReferenceType(resolved_method, invoke_instruction, return_replacement, do_rtp); +  *return_replacement = callee_graph->InlineInto(graph_, invoke_instruction);    return true;  } -void HInliner::FixUpReturnReferenceType(ArtMethod* resolved_method, -                                        HInvoke* invoke_instruction, +void HInliner::FixUpReturnReferenceType(HInvoke* invoke_instruction, +                                        ArtMethod* resolved_method,                                          HInstruction* return_replacement,                                          bool do_rtp) {    // Check the integrity of reference types and run another type propagation if needed. |