diff options
| author | 2016-08-30 10:23:01 -0700 | |
|---|---|---|
| committer | 2016-10-24 10:48:23 -0700 | |
| commit | 1ceb37c75544c5285fb87f27e1d9fa7261ced60c (patch) | |
| tree | a2c672170bb8424a148298706c6fd71d892a1b6e /compiler/optimizing | |
| parent | 3667e26de4856cccf24bcbab54ad3349a05267c0 (diff) | |
Remove unnecessary load class for new instance
Remove the load class for new instance if the load class has only one
use and can not throw. Previously many were not removed due to
MarkInDexCache nulling out the environment of the HLoadClass and
causing CanMoveClinitCheck to fail.
Also keep track of initialized HLoadClass and always remove clinit
checks for these.
Added checker regression test.
Code size savings: ARM64 CC boot.oat: 47896936 -> 47642488 (-0.53%)
Savings from IsInitialized optimization: 65984 bytes
Performance unmeasured, probably faster due to removing unnecessary
work.
Test: test-art-host with CC baker
Bug: 29516974
Change-Id: I43358762ffb380ebe7e6518d0d440a5e1cc03b61
Diffstat (limited to 'compiler/optimizing')
| -rw-r--r-- | compiler/optimizing/prepare_for_register_allocation.cc | 33 | ||||
| -rw-r--r-- | compiler/optimizing/sharpening.cc | 1 | 
2 files changed, 20 insertions, 14 deletions
| diff --git a/compiler/optimizing/prepare_for_register_allocation.cc b/compiler/optimizing/prepare_for_register_allocation.cc index a4df9e5503..7b66ef3627 100644 --- a/compiler/optimizing/prepare_for_register_allocation.cc +++ b/compiler/optimizing/prepare_for_register_allocation.cc @@ -129,6 +129,7 @@ void PrepareForRegisterAllocation::VisitClinitCheck(HClinitCheck* check) {    } else if (can_merge_with_load_class && !load_class->NeedsAccessCheck()) {      // Pass the initialization duty to the `HLoadClass` instruction,      // and remove the instruction from the graph. +    DCHECK(load_class->HasEnvironment());      load_class->SetMustGenerateClinitCheck(true);      check->GetBlock()->RemoveInstruction(check);    } @@ -136,7 +137,7 @@ void PrepareForRegisterAllocation::VisitClinitCheck(HClinitCheck* check) {  void PrepareForRegisterAllocation::VisitNewInstance(HNewInstance* instruction) {    HLoadClass* load_class = instruction->InputAt(0)->AsLoadClass(); -  bool has_only_one_use = load_class->HasOnlyOneNonEnvironmentUse(); +  const bool has_only_one_use = load_class->HasOnlyOneNonEnvironmentUse();    // Change the entrypoint to kQuickAllocObject if either:    // - the class is finalizable (only kQuickAllocObject handles finalizable classes),    // - the class needs access checks (we do not know if it's finalizable), @@ -144,19 +145,25 @@ void PrepareForRegisterAllocation::VisitNewInstance(HNewInstance* instruction) {    if (instruction->IsFinalizable() || has_only_one_use || load_class->NeedsAccessCheck()) {      instruction->SetEntrypoint(kQuickAllocObject);      instruction->ReplaceInput(GetGraph()->GetIntConstant(load_class->GetTypeIndex()), 0); -    // The allocation entry point that deals with access checks does not work with inlined -    // methods, so we need to check whether this allocation comes from an inlined method. -    // We also need to make the same check as for moving clinit check, whether the HLoadClass -    // has the clinit check responsibility or not (HLoadClass can throw anyway). -    if (has_only_one_use && -        !instruction->GetEnvironment()->IsFromInlinedInvoke() && -        CanMoveClinitCheck(load_class, instruction)) { -      // We can remove the load class from the graph. If it needed access checks, we delegate -      // the access check to the allocation. -      if (load_class->NeedsAccessCheck()) { -        instruction->SetEntrypoint(kQuickAllocObjectWithAccessCheck); +    if (has_only_one_use) { +      // We've just removed the only use of the HLoadClass. Since we don't run DCE after this pass, +      // do it manually if possible. +      if (!load_class->CanThrow()) { +        // If the load class can not throw, it has no side effects and can be removed if there is +        // only one use. +        load_class->GetBlock()->RemoveInstruction(load_class); +      } else if (!instruction->GetEnvironment()->IsFromInlinedInvoke() && +          CanMoveClinitCheck(load_class, instruction)) { +        // The allocation entry point that deals with access checks does not work with inlined +        // methods, so we need to check whether this allocation comes from an inlined method. +        // We also need to make the same check as for moving clinit check, whether the HLoadClass +        // has the clinit check responsibility or not (HLoadClass can throw anyway). +        // If it needed access checks, we delegate the access check to the allocation. +        if (load_class->NeedsAccessCheck()) { +          instruction->SetEntrypoint(kQuickAllocObjectWithAccessCheck); +        } +        load_class->GetBlock()->RemoveInstruction(load_class);        } -      load_class->GetBlock()->RemoveInstruction(load_class);      }    }  } diff --git a/compiler/optimizing/sharpening.cc b/compiler/optimizing/sharpening.cc index df1b351249..d938a70579 100644 --- a/compiler/optimizing/sharpening.cc +++ b/compiler/optimizing/sharpening.cc @@ -162,7 +162,6 @@ void HSharpening::ProcessLoadClass(HLoadClass* load_class) {          ? compilation_unit_.GetDexCache()          : hs.NewHandle(class_linker->FindDexCache(soa.Self(), dex_file));      mirror::Class* klass = dex_cache->GetResolvedType(type_index); -      if (codegen_->GetCompilerOptions().IsBootImage()) {        // Compiling boot image. Check if the class is a boot image class.        DCHECK(!runtime->UseJitCompilation()); |