diff options
Diffstat (limited to 'compiler/optimizing')
| -rw-r--r-- | compiler/optimizing/builder.cc | 12 | ||||
| -rw-r--r-- | compiler/optimizing/builder.h | 3 | ||||
| -rw-r--r-- | compiler/optimizing/code_generator.h | 1 | ||||
| -rw-r--r-- | compiler/optimizing/code_generator_arm64.cc | 43 | ||||
| -rw-r--r-- | compiler/optimizing/code_generator_arm64.h | 4 | ||||
| -rw-r--r-- | compiler/optimizing/inliner.cc | 5 | ||||
| -rw-r--r-- | compiler/optimizing/instruction_builder.cc | 183 | ||||
| -rw-r--r-- | compiler/optimizing/instruction_builder.h | 27 | ||||
| -rw-r--r-- | compiler/optimizing/instruction_simplifier_arm.cc | 5 | ||||
| -rw-r--r-- | compiler/optimizing/instruction_simplifier_shared.cc | 4 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics.h | 1 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_arm64.cc | 34 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_arm_vixl.cc | 1 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_mips.cc | 2 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_mips64.cc | 1 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_x86.cc | 1 | ||||
| -rw-r--r-- | compiler/optimizing/intrinsics_x86_64.cc | 1 | ||||
| -rw-r--r-- | compiler/optimizing/nodes.h | 2 | ||||
| -rw-r--r-- | compiler/optimizing/optimizing_compiler.cc | 40 | 
19 files changed, 187 insertions, 183 deletions
| diff --git a/compiler/optimizing/builder.cc b/compiler/optimizing/builder.cc index a1a5692ef6..64aa1b9358 100644 --- a/compiler/optimizing/builder.cc +++ b/compiler/optimizing/builder.cc @@ -21,6 +21,7 @@  #include "base/bit_vector-inl.h"  #include "base/logging.h"  #include "block_builder.h" +#include "code_generator.h"  #include "data_type-inl.h"  #include "dex/verified_method.h"  #include "driver/compiler_options.h" @@ -40,7 +41,6 @@ HGraphBuilder::HGraphBuilder(HGraph* graph,                               const CodeItemDebugInfoAccessor& accessor,                               const DexCompilationUnit* dex_compilation_unit,                               const DexCompilationUnit* outer_compilation_unit, -                             CompilerDriver* driver,                               CodeGenerator* code_generator,                               OptimizingCompilerStats* compiler_stats,                               ArrayRef<const uint8_t> interpreter_metadata, @@ -50,7 +50,6 @@ HGraphBuilder::HGraphBuilder(HGraph* graph,        code_item_accessor_(accessor),        dex_compilation_unit_(dex_compilation_unit),        outer_compilation_unit_(outer_compilation_unit), -      compiler_driver_(driver),        code_generator_(code_generator),        compilation_stats_(compiler_stats),        interpreter_metadata_(interpreter_metadata), @@ -67,19 +66,18 @@ HGraphBuilder::HGraphBuilder(HGraph* graph,        code_item_accessor_(accessor),        dex_compilation_unit_(dex_compilation_unit),        outer_compilation_unit_(nullptr), -      compiler_driver_(nullptr),        code_generator_(nullptr),        compilation_stats_(nullptr),        handles_(handles),        return_type_(return_type) {}  bool HGraphBuilder::SkipCompilation(size_t number_of_branches) { -  if (compiler_driver_ == nullptr) { -    // Note that the compiler driver is null when unit testing. +  if (code_generator_ == nullptr) { +    // Note that the codegen is null when unit testing.      return false;    } -  const CompilerOptions& compiler_options = compiler_driver_->GetCompilerOptions(); +  const CompilerOptions& compiler_options = code_generator_->GetCompilerOptions();    CompilerFilter::Filter compiler_filter = compiler_options.GetCompilerFilter();    if (compiler_filter == CompilerFilter::kEverything) {      return false; @@ -131,7 +129,6 @@ GraphAnalysisResult HGraphBuilder::BuildGraph() {                                            return_type_,                                            dex_compilation_unit_,                                            outer_compilation_unit_, -                                          compiler_driver_,                                            code_generator_,                                            interpreter_metadata_,                                            compilation_stats_, @@ -203,7 +200,6 @@ void HGraphBuilder::BuildIntrinsicGraph(ArtMethod* method) {                                            return_type_,                                            dex_compilation_unit_,                                            outer_compilation_unit_, -                                          compiler_driver_,                                            code_generator_,                                            interpreter_metadata_,                                            compilation_stats_, diff --git a/compiler/optimizing/builder.h b/compiler/optimizing/builder.h index 5a1914ce08..6152740324 100644 --- a/compiler/optimizing/builder.h +++ b/compiler/optimizing/builder.h @@ -22,7 +22,6 @@  #include "dex/code_item_accessors.h"  #include "dex/dex_file-inl.h"  #include "dex/dex_file.h" -#include "driver/compiler_driver.h"  #include "nodes.h"  namespace art { @@ -38,7 +37,6 @@ class HGraphBuilder : public ValueObject {                  const CodeItemDebugInfoAccessor& accessor,                  const DexCompilationUnit* dex_compilation_unit,                  const DexCompilationUnit* outer_compilation_unit, -                CompilerDriver* driver,                  CodeGenerator* code_generator,                  OptimizingCompilerStats* compiler_stats,                  ArrayRef<const uint8_t> interpreter_metadata, @@ -70,7 +68,6 @@ class HGraphBuilder : public ValueObject {    // The compilation unit of the enclosing method being compiled.    const DexCompilationUnit* const outer_compilation_unit_; -  CompilerDriver* const compiler_driver_;    CodeGenerator* const code_generator_;    OptimizingCompilerStats* const compilation_stats_; diff --git a/compiler/optimizing/code_generator.h b/compiler/optimizing/code_generator.h index 3f560780ce..39966ff8ea 100644 --- a/compiler/optimizing/code_generator.h +++ b/compiler/optimizing/code_generator.h @@ -59,7 +59,6 @@ static constexpr ReadBarrierOption kCompilerReadBarrierOption =  class Assembler;  class CodeGenerator; -class CompilerDriver;  class CompilerOptions;  class StackMapStream;  class ParallelMoveResolver; diff --git a/compiler/optimizing/code_generator_arm64.cc b/compiler/optimizing/code_generator_arm64.cc index a9acf90762..48b50ea5d6 100644 --- a/compiler/optimizing/code_generator_arm64.cc +++ b/compiler/optimizing/code_generator_arm64.cc @@ -2319,9 +2319,10 @@ void LocationsBuilderARM64::VisitArrayGet(HArrayGet* instruction) {        if (offset >= kReferenceLoadMinFarOffset) {          locations->AddTemp(FixedTempLocation());        } -    } else { +    } else if (!instruction->GetArray()->IsIntermediateAddress()) {        // We need a non-scratch temporary for the array data pointer in -      // CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier(). +      // CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier() for the case with no +      // intermediate address.        locations->AddTemp(Location::RequiresRegister());      }    } @@ -2351,11 +2352,12 @@ void InstructionCodeGeneratorARM64::VisitArrayGet(HArrayGet* instruction) {    MacroAssembler* masm = GetVIXLAssembler();    UseScratchRegisterScope temps(masm); -  // The read barrier instrumentation of object ArrayGet instructions +  // The non-Baker read barrier instrumentation of object ArrayGet instructions    // does not support the HIntermediateAddress instruction.    DCHECK(!((type == DataType::Type::kReference) &&             instruction->GetArray()->IsIntermediateAddress() && -           kEmitCompilerReadBarrier)); +           kEmitCompilerReadBarrier && +           !kUseBakerReadBarrier));    if (type == DataType::Type::kReference && kEmitCompilerReadBarrier && kUseBakerReadBarrier) {      // Object ArrayGet with Baker's read barrier case. @@ -2363,6 +2365,7 @@ void InstructionCodeGeneratorARM64::VisitArrayGet(HArrayGet* instruction) {      // CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier call.      DCHECK(!instruction->CanDoImplicitNullCheckOn(instruction->InputAt(0)));      if (index.IsConstant()) { +      DCHECK(!instruction->GetArray()->IsIntermediateAddress());        // Array load with a constant index can be treated as a field load.        offset += Int64FromLocation(index) << DataType::SizeShift(type);        Location maybe_temp = @@ -2375,9 +2378,8 @@ void InstructionCodeGeneratorARM64::VisitArrayGet(HArrayGet* instruction) {                                                        /* needs_null_check */ false,                                                        /* use_load_acquire */ false);      } else { -      Register temp = WRegisterFrom(locations->GetTemp(0));        codegen_->GenerateArrayLoadWithBakerReadBarrier( -          out, obj.W(), offset, index, temp, /* needs_null_check */ false); +          instruction, out, obj.W(), offset, index, /* needs_null_check */ false);      }    } else {      // General case. @@ -2426,8 +2428,8 @@ void InstructionCodeGeneratorARM64::VisitArrayGet(HArrayGet* instruction) {          // input instruction has done it already. See the comment in          // `TryExtractArrayAccessAddress()`.          if (kIsDebugBuild) { -          HIntermediateAddress* tmp = instruction->GetArray()->AsIntermediateAddress(); -          DCHECK_EQ(tmp->GetOffset()->AsIntConstant()->GetValueAsUint64(), offset); +          HIntermediateAddress* interm_addr = instruction->GetArray()->AsIntermediateAddress(); +          DCHECK_EQ(interm_addr->GetOffset()->AsIntConstant()->GetValueAsUint64(), offset);          }          temp = obj;        } else { @@ -2539,8 +2541,8 @@ void InstructionCodeGeneratorARM64::VisitArraySet(HArraySet* instruction) {          // input instruction has done it already. See the comment in          // `TryExtractArrayAccessAddress()`.          if (kIsDebugBuild) { -          HIntermediateAddress* tmp = instruction->GetArray()->AsIntermediateAddress(); -          DCHECK(tmp->GetOffset()->AsIntConstant()->GetValueAsUint64() == offset); +          HIntermediateAddress* interm_addr = instruction->GetArray()->AsIntermediateAddress(); +          DCHECK(interm_addr->GetOffset()->AsIntConstant()->GetValueAsUint64() == offset);          }          temp = array;        } else { @@ -5957,11 +5959,11 @@ void CodeGeneratorARM64::GenerateFieldLoadWithBakerReadBarrier(HInstruction* ins        instruction, ref, obj, src, needs_null_check, use_load_acquire);  } -void CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier(Location ref, +void CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier(HArrayGet* instruction, +                                                               Location ref,                                                                 Register obj,                                                                 uint32_t data_offset,                                                                 Location index, -                                                               Register temp,                                                                 bool needs_null_check) {    DCHECK(kEmitCompilerReadBarrier);    DCHECK(kUseBakerReadBarrier); @@ -6000,9 +6002,24 @@ void CodeGeneratorARM64::GenerateArrayLoadWithBakerReadBarrier(Location ref,    DCHECK(temps.IsAvailable(ip0));    DCHECK(temps.IsAvailable(ip1));    temps.Exclude(ip0, ip1); + +  Register temp; +  if (instruction->GetArray()->IsIntermediateAddress()) { +    // We do not need to compute the intermediate address from the array: the +    // input instruction has done it already. See the comment in +    // `TryExtractArrayAccessAddress()`. +    if (kIsDebugBuild) { +      HIntermediateAddress* interm_addr = instruction->GetArray()->AsIntermediateAddress(); +      DCHECK_EQ(interm_addr->GetOffset()->AsIntConstant()->GetValueAsUint64(), data_offset); +    } +    temp = obj; +  } else { +    temp = WRegisterFrom(instruction->GetLocations()->GetTemp(0)); +    __ Add(temp.X(), obj.X(), Operand(data_offset)); +  } +    uint32_t custom_data = EncodeBakerReadBarrierArrayData(temp.GetCode()); -  __ Add(temp.X(), obj.X(), Operand(data_offset));    {      ExactAssemblyScope guard(GetVIXLAssembler(),                               (kPoisonHeapReferences ? 4u : 3u) * vixl::aarch64::kInstructionSize); diff --git a/compiler/optimizing/code_generator_arm64.h b/compiler/optimizing/code_generator_arm64.h index 1ba58b1a9f..ada5742fc0 100644 --- a/compiler/optimizing/code_generator_arm64.h +++ b/compiler/optimizing/code_generator_arm64.h @@ -694,11 +694,11 @@ class CodeGeneratorARM64 : public CodeGenerator {                                               bool use_load_acquire);    // Fast path implementation of ReadBarrier::Barrier for a heap    // reference array load when Baker's read barriers are used. -  void GenerateArrayLoadWithBakerReadBarrier(Location ref, +  void GenerateArrayLoadWithBakerReadBarrier(HArrayGet* instruction, +                                             Location ref,                                               vixl::aarch64::Register obj,                                               uint32_t data_offset,                                               Location index, -                                             vixl::aarch64::Register temp,                                               bool needs_null_check);    // Emit code checking the status of the Marking Register, and diff --git a/compiler/optimizing/inliner.cc b/compiler/optimizing/inliner.cc index dd781c288f..f56f9cb84f 100644 --- a/compiler/optimizing/inliner.cc +++ b/compiler/optimizing/inliner.cc @@ -1757,6 +1757,7 @@ bool HInliner::TryBuildAndInlineHelper(HInvoke* invoke_instruction,                             caller_compilation_unit_.GetClassLoader(),                             handles_); +  Handle<mirror::Class> compiling_class = handles_->NewHandle(resolved_method->GetDeclaringClass());    DexCompilationUnit dex_compilation_unit(        class_loader,        class_linker, @@ -1766,7 +1767,8 @@ bool HInliner::TryBuildAndInlineHelper(HInvoke* invoke_instruction,        method_index,        resolved_method->GetAccessFlags(),        /* verified_method */ nullptr, -      dex_cache); +      dex_cache, +      compiling_class);    InvokeType invoke_type = invoke_instruction->GetInvokeType();    if (invoke_type == kInterface) { @@ -1805,7 +1807,6 @@ bool HInliner::TryBuildAndInlineHelper(HInvoke* invoke_instruction,                          code_item_accessor,                          &dex_compilation_unit,                          &outer_compilation_unit_, -                        compiler_driver_,                          codegen_,                          inline_stats_,                          resolved_method->GetQuickenedInfo(), diff --git a/compiler/optimizing/instruction_builder.cc b/compiler/optimizing/instruction_builder.cc index 63b2705b43..e9b5b5a93d 100644 --- a/compiler/optimizing/instruction_builder.cc +++ b/compiler/optimizing/instruction_builder.cc @@ -20,11 +20,11 @@  #include "base/arena_bit_vector.h"  #include "base/bit_vector-inl.h"  #include "block_builder.h" -#include "class_linker.h" +#include "class_linker-inl.h" +#include "code_generator.h"  #include "data_type-inl.h"  #include "dex/bytecode_utils.h"  #include "dex/dex_instruction-inl.h" -#include "driver/compiler_driver-inl.h"  #include "driver/dex_compilation_unit.h"  #include "driver/compiler_options.h"  #include "imtable-inl.h" @@ -47,7 +47,6 @@ HInstructionBuilder::HInstructionBuilder(HGraph* graph,                                           DataType::Type return_type,                                           const DexCompilationUnit* dex_compilation_unit,                                           const DexCompilationUnit* outer_compilation_unit, -                                         CompilerDriver* compiler_driver,                                           CodeGenerator* code_generator,                                           ArrayRef<const uint8_t> interpreter_metadata,                                           OptimizingCompilerStats* compiler_stats, @@ -61,7 +60,6 @@ HInstructionBuilder::HInstructionBuilder(HGraph* graph,        return_type_(return_type),        block_builder_(block_builder),        ssa_builder_(ssa_builder), -      compiler_driver_(compiler_driver),        code_generator_(code_generator),        dex_compilation_unit_(dex_compilation_unit),        outer_compilation_unit_(outer_compilation_unit), @@ -73,7 +71,8 @@ HInstructionBuilder::HInstructionBuilder(HGraph* graph,        current_locals_(nullptr),        latest_result_(nullptr),        current_this_parameter_(nullptr), -      loop_headers_(local_allocator->Adapter(kArenaAllocGraphBuilder)) { +      loop_headers_(local_allocator->Adapter(kArenaAllocGraphBuilder)), +      class_cache_(std::less<dex::TypeIndex>(), local_allocator->Adapter(kArenaAllocGraphBuilder)) {    loop_headers_.reserve(kDefaultNumberOfLoops);  } @@ -319,8 +318,8 @@ bool HInstructionBuilder::Build() {    // Find locations where we want to generate extra stackmaps for native debugging.    // This allows us to generate the info only at interesting points (for example,    // at start of java statement) rather than before every dex instruction. -  const bool native_debuggable = compiler_driver_ != nullptr && -                                 compiler_driver_->GetCompilerOptions().GetNativeDebuggable(); +  const bool native_debuggable = code_generator_ != nullptr && +                                 code_generator_->GetCompilerOptions().GetNativeDebuggable();    ArenaBitVector* native_debug_info_locations = nullptr;    if (native_debuggable) {      native_debug_info_locations = FindNativeDebugInfoLocations(); @@ -709,20 +708,18 @@ void HInstructionBuilder::Binop_22b(const Instruction& instruction, bool reverse  // Does the method being compiled need any constructor barriers being inserted?  // (Always 'false' for methods that aren't <init>.) -static bool RequiresConstructorBarrier(const DexCompilationUnit* cu, CompilerDriver* driver) { +static bool RequiresConstructorBarrier(const DexCompilationUnit* cu) {    // Can be null in unit tests only.    if (UNLIKELY(cu == nullptr)) {      return false;    } -  Thread* self = Thread::Current(); -  return cu->IsConstructor() -      && !cu->IsStatic() -      // RequiresConstructorBarrier must only be queried for <init> methods; -      // it's effectively "false" for every other method. -      // -      // See CompilerDriver::RequiresConstructBarrier for more explanation. -      && driver->RequiresConstructorBarrier(self, cu->GetDexFile(), cu->GetClassDefIndex()); +  // Constructor barriers are applicable only for <init> methods. +  if (LIKELY(!cu->IsConstructor() || cu->IsStatic())) { +    return false; +  } + +  return cu->RequiresConstructorBarrier();  }  // Returns true if `block` has only one successor which starts at the next @@ -768,7 +765,7 @@ void HInstructionBuilder::BuildReturn(const Instruction& instruction,      // Only <init> (which is a return-void) could possibly have a constructor fence.      // This may insert additional redundant constructor fences from the super constructors.      // TODO: remove redundant constructor fences (b/36656456). -    if (RequiresConstructorBarrier(dex_compilation_unit_, compiler_driver_)) { +    if (RequiresConstructorBarrier(dex_compilation_unit_)) {        // Compiling instance constructor.        DCHECK_STREQ("<init>", graph_->GetMethodName()); @@ -782,7 +779,7 @@ void HInstructionBuilder::BuildReturn(const Instruction& instruction,      }      AppendInstruction(new (allocator_) HReturnVoid(dex_pc));    } else { -    DCHECK(!RequiresConstructorBarrier(dex_compilation_unit_, compiler_driver_)); +    DCHECK(!RequiresConstructorBarrier(dex_compilation_unit_));      HInstruction* value = LoadLocal(instruction.VRegA(), type);      AppendInstruction(new (allocator_) HReturn(value, dex_pc));    } @@ -849,7 +846,7 @@ ArtMethod* HInstructionBuilder::ResolveMethod(uint16_t method_idx, InvokeType in    // make this an invoke-unresolved to handle cross-dex invokes or abstract super methods, both of    // which require runtime handling.    if (invoke_type == kSuper) { -    ObjPtr<mirror::Class> compiling_class = ResolveCompilingClass(soa); +    ObjPtr<mirror::Class> compiling_class = dex_compilation_unit_->GetCompilingClass().Get();      if (compiling_class == nullptr) {        // We could not determine the method's class we need to wait until runtime.        DCHECK(Runtime::Current()->IsAotCompiler()); @@ -879,8 +876,8 @@ ArtMethod* HInstructionBuilder::ResolveMethod(uint16_t method_idx, InvokeType in        // The back-end code generator relies on this check in order to ensure that it will not        // attempt to read the dex_cache with a dex_method_index that is not from the correct        // dex_file. If we didn't do this check then the dex_method_index will not be updated in the -      // builder, which means that the code-generator (and compiler driver during sharpening and -      // inliner, maybe) might invoke an incorrect method. +      // builder, which means that the code-generator (and sharpening and inliner, maybe) +      // might invoke an incorrect method.        // TODO: The actual method could still be referenced in the current dex file, so we        //       could try locating it.        // TODO: Remove the dex_file restriction. @@ -969,7 +966,7 @@ bool HInstructionBuilder::BuildInvoke(const Instruction& instruction,      ScopedObjectAccess soa(Thread::Current());      if (invoke_type == kStatic) {        clinit_check = -          ProcessClinitCheckForInvoke(soa, dex_pc, resolved_method, &clinit_check_requirement); +          ProcessClinitCheckForInvoke(dex_pc, resolved_method, &clinit_check_requirement);      } else if (invoke_type == kSuper) {        if (IsSameDexFile(*resolved_method->GetDexFile(), *dex_compilation_unit_->GetDexFile())) {          // Update the method index to the one resolved. Note that this may be a no-op if @@ -1055,7 +1052,7 @@ HNewInstance* HInstructionBuilder::BuildNewInstance(dex::TypeIndex type_index, u    HInstruction* cls = load_class;    Handle<mirror::Class> klass = load_class->GetClass(); -  if (!IsInitialized(soa, klass)) { +  if (!IsInitialized(klass)) {      cls = new (allocator_) HClinitCheck(load_class, dex_pc);      AppendInstruction(cls);    } @@ -1284,7 +1281,7 @@ static bool HasTrivialInitialization(ObjPtr<mirror::Class> cls,    return true;  } -bool HInstructionBuilder::IsInitialized(ScopedObjectAccess& soa, Handle<mirror::Class> cls) const { +bool HInstructionBuilder::IsInitialized(Handle<mirror::Class> cls) const {    if (cls == nullptr) {      return false;    } @@ -1299,7 +1296,7 @@ bool HInstructionBuilder::IsInitialized(ScopedObjectAccess& soa, Handle<mirror::      }      // Assume loaded only if klass is in the boot image. App classes cannot be assumed      // loaded because we don't even know what class loader will be used to load them. -    if (IsInBootImage(cls.Get(), compiler_driver_->GetCompilerOptions())) { +    if (IsInBootImage(cls.Get(), code_generator_->GetCompilerOptions())) {        return true;      }    } @@ -1312,29 +1309,20 @@ bool HInstructionBuilder::IsInitialized(ScopedObjectAccess& soa, Handle<mirror::    // can be completely initialized while the superclass is initializing and the subclass    // remains initialized when the superclass initializer throws afterwards. b/62478025    // Note: The HClinitCheck+HInvokeStaticOrDirect merging can still apply. -  ObjPtr<mirror::Class> outermost_cls = ResolveOutermostCompilingClass(soa); -  bool is_outer_static_or_constructor = -      (outer_compilation_unit_->GetAccessFlags() & (kAccStatic | kAccConstructor)) != 0u; -  if (is_outer_static_or_constructor && outermost_cls == cls.Get()) { +  auto is_static_method_or_constructor_of_cls = [cls](const DexCompilationUnit& compilation_unit) +      REQUIRES_SHARED(Locks::mutator_lock_) { +    return (compilation_unit.GetAccessFlags() & (kAccStatic | kAccConstructor)) != 0u && +           compilation_unit.GetCompilingClass().Get() == cls.Get(); +  }; +  if (is_static_method_or_constructor_of_cls(*outer_compilation_unit_) || +      // Check also the innermost method. Though excessive copies of ClinitCheck can be +      // eliminated by GVN, that happens only after the decision whether to inline the +      // graph or not and that may depend on the presence of the ClinitCheck. +      // TODO: We should walk over the entire inlined method chain, but we don't pass that +      // information to the builder. +      is_static_method_or_constructor_of_cls(*dex_compilation_unit_)) {      return true;    } -  // Remember if the compiled class is a subclass of `cls`. By the time this is used -  // below the `outermost_cls` may be invalidated by calling ResolveCompilingClass(). -  bool is_subclass = IsSubClass(outermost_cls, cls.Get()); -  if (dex_compilation_unit_ != outer_compilation_unit_) { -    // Check also the innermost method. Though excessive copies of ClinitCheck can be -    // eliminated by GVN, that happens only after the decision whether to inline the -    // graph or not and that may depend on the presence of the ClinitCheck. -    // TODO: We should walk over the entire inlined method chain, but we don't pass that -    // information to the builder. -    ObjPtr<mirror::Class> innermost_cls = ResolveCompilingClass(soa); -    bool is_inner_static_or_constructor = -        (dex_compilation_unit_->GetAccessFlags() & (kAccStatic | kAccConstructor)) != 0u; -    if (is_inner_static_or_constructor && innermost_cls == cls.Get()) { -      return true; -    } -    is_subclass = is_subclass || IsSubClass(innermost_cls, cls.Get()); -  }    // Otherwise, we may be able to avoid the check if `cls` is a superclass of a method being    // compiled here (anywhere in the inlining chain) as the `cls` must have started initializing @@ -1355,7 +1343,12 @@ bool HInstructionBuilder::IsInitialized(ScopedObjectAccess& soa, Handle<mirror::    // TODO: We should walk over the entire inlined methods chain, but we don't pass that    // information to the builder. (We could also check if we're guaranteed a non-null instance    // of `cls` at this location but that's outside the scope of the instruction builder.) -  if (is_subclass && HasTrivialInitialization(cls.Get(), compiler_driver_->GetCompilerOptions())) { +  bool is_subclass = IsSubClass(outer_compilation_unit_->GetCompilingClass().Get(), cls.Get()); +  if (dex_compilation_unit_ != outer_compilation_unit_) { +    is_subclass = is_subclass || +                  IsSubClass(dex_compilation_unit_->GetCompilingClass().Get(), cls.Get()); +  } +  if (is_subclass && HasTrivialInitialization(cls.Get(), code_generator_->GetCompilerOptions())) {      return true;    } @@ -1363,18 +1356,16 @@ bool HInstructionBuilder::IsInitialized(ScopedObjectAccess& soa, Handle<mirror::  }  HClinitCheck* HInstructionBuilder::ProcessClinitCheckForInvoke( -    ScopedObjectAccess& soa,      uint32_t dex_pc,      ArtMethod* resolved_method,      HInvokeStaticOrDirect::ClinitCheckRequirement* clinit_check_requirement) {    Handle<mirror::Class> klass = handles_->NewHandle(resolved_method->GetDeclaringClass());    HClinitCheck* clinit_check = nullptr; -  if (IsInitialized(soa, klass)) { +  if (IsInitialized(klass)) {      *clinit_check_requirement = HInvokeStaticOrDirect::ClinitCheckRequirement::kNone;    } else { -    HLoadClass* cls = BuildLoadClass(soa, -                                     klass->GetDexTypeIndex(), +    HLoadClass* cls = BuildLoadClass(klass->GetDexTypeIndex(),                                       klass->GetDexFile(),                                       klass,                                       dex_pc, @@ -1610,43 +1601,6 @@ bool HInstructionBuilder::BuildInstanceFieldAccess(const Instruction& instructio    return true;  } -static ObjPtr<mirror::Class> ResolveClassFrom(ScopedObjectAccess& soa, -                                              CompilerDriver* driver, -                                              const DexCompilationUnit& compilation_unit) -    REQUIRES_SHARED(Locks::mutator_lock_) { -  Handle<mirror::ClassLoader> class_loader = compilation_unit.GetClassLoader(); -  Handle<mirror::DexCache> dex_cache = compilation_unit.GetDexCache(); - -  return driver->ResolveCompilingMethodsClass(soa, dex_cache, class_loader, &compilation_unit); -} - -ObjPtr<mirror::Class> HInstructionBuilder::ResolveOutermostCompilingClass( -    ScopedObjectAccess& soa) const { -  return ResolveClassFrom(soa, compiler_driver_, *outer_compilation_unit_); -} - -ObjPtr<mirror::Class> HInstructionBuilder::ResolveCompilingClass(ScopedObjectAccess& soa) const { -  return ResolveClassFrom(soa, compiler_driver_, *dex_compilation_unit_); -} - -bool HInstructionBuilder::IsOutermostCompilingClass(dex::TypeIndex type_index) const { -  ScopedObjectAccess soa(Thread::Current()); -  StackHandleScope<2> hs(soa.Self()); -  Handle<mirror::DexCache> dex_cache = dex_compilation_unit_->GetDexCache(); -  Handle<mirror::ClassLoader> class_loader = dex_compilation_unit_->GetClassLoader(); -  Handle<mirror::Class> cls(hs.NewHandle(compiler_driver_->ResolveClass( -      soa, dex_cache, class_loader, type_index, dex_compilation_unit_))); -  Handle<mirror::Class> outer_class(hs.NewHandle(ResolveOutermostCompilingClass(soa))); - -  // GetOutermostCompilingClass returns null when the class is unresolved -  // (e.g. if it derives from an unresolved class). This is bogus knowing that -  // we are compiling it. -  // When this happens we cannot establish a direct relation between the current -  // class and the outer class, so we return false. -  // (Note that this is only used for optimizing invokes and field accesses) -  return (cls != nullptr) && (outer_class.Get() == cls.Get()); -} -  void HInstructionBuilder::BuildUnresolvedStaticFieldAccess(const Instruction& instruction,                                                             uint32_t dex_pc,                                                             bool is_put, @@ -1666,18 +1620,17 @@ void HInstructionBuilder::BuildUnresolvedStaticFieldAccess(const Instruction& in  ArtField* HInstructionBuilder::ResolveField(uint16_t field_idx, bool is_static, bool is_put) {    ScopedObjectAccess soa(Thread::Current()); -  StackHandleScope<2> hs(soa.Self());    ClassLinker* class_linker = dex_compilation_unit_->GetClassLinker();    Handle<mirror::ClassLoader> class_loader = dex_compilation_unit_->GetClassLoader(); -  Handle<mirror::Class> compiling_class(hs.NewHandle(ResolveCompilingClass(soa)));    ArtField* resolved_field = class_linker->ResolveField(field_idx,                                                          dex_compilation_unit_->GetDexCache(),                                                          class_loader,                                                          is_static); +  DCHECK_EQ(resolved_field == nullptr, soa.Self()->IsExceptionPending());    if (UNLIKELY(resolved_field == nullptr)) { -    // Clean up any exception left by type resolution. +    // Clean up any exception left by field resolution.      soa.Self()->ClearException();      return nullptr;    } @@ -1689,6 +1642,7 @@ ArtField* HInstructionBuilder::ResolveField(uint16_t field_idx, bool is_static,    }    // Check access. +  Handle<mirror::Class> compiling_class = dex_compilation_unit_->GetCompilingClass();    if (compiling_class == nullptr) {      if (!resolved_field->IsPublic()) {        return nullptr; @@ -1731,8 +1685,7 @@ void HInstructionBuilder::BuildStaticFieldAccess(const Instruction& instruction,    DataType::Type field_type = GetFieldAccessType(*dex_file_, field_index);    Handle<mirror::Class> klass = handles_->NewHandle(resolved_field->GetDeclaringClass()); -  HLoadClass* constant = BuildLoadClass(soa, -                                        klass->GetDexTypeIndex(), +  HLoadClass* constant = BuildLoadClass(klass->GetDexTypeIndex(),                                          klass->GetDexFile(),                                          klass,                                          dex_pc, @@ -1748,7 +1701,7 @@ void HInstructionBuilder::BuildStaticFieldAccess(const Instruction& instruction,    }    HInstruction* cls = constant; -  if (!IsInitialized(soa, klass)) { +  if (!IsInitialized(klass)) {      cls = new (allocator_) HClinitCheck(constant, dex_pc);      AppendInstruction(cls);    } @@ -1989,12 +1942,11 @@ HLoadClass* HInstructionBuilder::BuildLoadClass(dex::TypeIndex type_index, uint3    ScopedObjectAccess soa(Thread::Current());    const DexFile& dex_file = *dex_compilation_unit_->GetDexFile();    Handle<mirror::Class> klass = ResolveClass(soa, type_index); -  bool needs_access_check = LoadClassNeedsAccessCheck(soa, klass); -  return BuildLoadClass(soa, type_index, dex_file, klass, dex_pc, needs_access_check); +  bool needs_access_check = LoadClassNeedsAccessCheck(klass); +  return BuildLoadClass(type_index, dex_file, klass, dex_pc, needs_access_check);  } -HLoadClass* HInstructionBuilder::BuildLoadClass(ScopedObjectAccess& soa, -                                                dex::TypeIndex type_index, +HLoadClass* HInstructionBuilder::BuildLoadClass(dex::TypeIndex type_index,                                                  const DexFile& dex_file,                                                  Handle<mirror::Class> klass,                                                  uint32_t dex_pc, @@ -2011,11 +1963,8 @@ HLoadClass* HInstructionBuilder::BuildLoadClass(ScopedObjectAccess& soa,    }    // Note: `klass` must be from `handles_`. -  bool is_referrers_class = false; -  if (klass != nullptr) { -    ObjPtr<mirror::Class> outermost_cls = ResolveOutermostCompilingClass(soa); -    is_referrers_class = (outermost_cls == klass.Get()); -  } +  bool is_referrers_class = +      (klass != nullptr) && (outer_compilation_unit_->GetCompilingClass().Get() == klass.Get());    HLoadClass* load_class = new (allocator_) HLoadClass(        graph_->GetCurrentMethod(),        type_index, @@ -2041,22 +1990,28 @@ HLoadClass* HInstructionBuilder::BuildLoadClass(ScopedObjectAccess& soa,  Handle<mirror::Class> HInstructionBuilder::ResolveClass(ScopedObjectAccess& soa,                                                          dex::TypeIndex type_index) { -  Handle<mirror::ClassLoader> class_loader = dex_compilation_unit_->GetClassLoader(); -  ObjPtr<mirror::Class> klass = compiler_driver_->ResolveClass( -      soa, dex_compilation_unit_->GetDexCache(), class_loader, type_index, dex_compilation_unit_); -  // TODO: Avoid creating excessive handles if the method references the same class repeatedly. -  // (Use a map on the local_allocator_.) -  return handles_->NewHandle(klass); +  auto it = class_cache_.find(type_index); +  if (it != class_cache_.end()) { +    return it->second; +  } + +  ObjPtr<mirror::Class> klass = dex_compilation_unit_->GetClassLinker()->ResolveType( +      type_index, dex_compilation_unit_->GetDexCache(), dex_compilation_unit_->GetClassLoader()); +  DCHECK_EQ(klass == nullptr, soa.Self()->IsExceptionPending()); +  soa.Self()->ClearException();  // Clean up the exception left by type resolution if any. + +  Handle<mirror::Class> h_klass = handles_->NewHandle(klass); +  class_cache_.Put(type_index, h_klass); +  return h_klass;  } -bool HInstructionBuilder::LoadClassNeedsAccessCheck(ScopedObjectAccess& soa, -                                                    Handle<mirror::Class> klass) { +bool HInstructionBuilder::LoadClassNeedsAccessCheck(Handle<mirror::Class> klass) {    if (klass == nullptr) {      return true;    } else if (klass->IsPublic()) {      return false;    } else { -    ObjPtr<mirror::Class> compiling_class = ResolveCompilingClass(soa); +    ObjPtr<mirror::Class> compiling_class = dex_compilation_unit_->GetCompilingClass().Get();      return compiling_class == nullptr || !compiling_class->CanAccess(klass.Get());    }  } @@ -2085,7 +2040,7 @@ void HInstructionBuilder::BuildTypeCheck(const Instruction& instruction,    ScopedObjectAccess soa(Thread::Current());    const DexFile& dex_file = *dex_compilation_unit_->GetDexFile();    Handle<mirror::Class> klass = ResolveClass(soa, type_index); -  bool needs_access_check = LoadClassNeedsAccessCheck(soa, klass); +  bool needs_access_check = LoadClassNeedsAccessCheck(klass);    TypeCheckKind check_kind = HSharpening::ComputeTypeCheckKind(        klass.Get(), code_generator_, needs_access_check); @@ -2103,7 +2058,7 @@ void HInstructionBuilder::BuildTypeCheck(const Instruction& instruction,      bitstring_path_to_root = graph_->GetIntConstant(static_cast<int32_t>(path_to_root), dex_pc);      bitstring_mask = graph_->GetIntConstant(static_cast<int32_t>(mask), dex_pc);    } else { -    class_or_null = BuildLoadClass(soa, type_index, dex_file, klass, dex_pc, needs_access_check); +    class_or_null = BuildLoadClass(type_index, dex_file, klass, dex_pc, needs_access_check);    }    DCHECK(class_or_null != nullptr); diff --git a/compiler/optimizing/instruction_builder.h b/compiler/optimizing/instruction_builder.h index 2ab2139216..d701445946 100644 --- a/compiler/optimizing/instruction_builder.h +++ b/compiler/optimizing/instruction_builder.h @@ -34,7 +34,6 @@ class ArenaBitVector;  class ArtField;  class ArtMethod;  class CodeGenerator; -class CompilerDriver;  class DexCompilationUnit;  class HBasicBlockBuilder;  class Instruction; @@ -59,7 +58,6 @@ class HInstructionBuilder : public ValueObject {                        DataType::Type return_type,                        const DexCompilationUnit* dex_compilation_unit,                        const DexCompilationUnit* outer_compilation_unit, -                      CompilerDriver* compiler_driver,                        CodeGenerator* code_generator,                        ArrayRef<const uint8_t> interpreter_metadata,                        OptimizingCompilerStats* compiler_stats, @@ -222,8 +220,7 @@ class HInstructionBuilder : public ValueObject {    // Builds a `HLoadClass` loading the given `type_index`.    HLoadClass* BuildLoadClass(dex::TypeIndex type_index, uint32_t dex_pc); -  HLoadClass* BuildLoadClass(ScopedObjectAccess& soa, -                             dex::TypeIndex type_index, +  HLoadClass* BuildLoadClass(dex::TypeIndex type_index,                               const DexFile& dex_file,                               Handle<mirror::Class> klass,                               uint32_t dex_pc, @@ -233,7 +230,7 @@ class HInstructionBuilder : public ValueObject {    Handle<mirror::Class> ResolveClass(ScopedObjectAccess& soa, dex::TypeIndex type_index)        REQUIRES_SHARED(Locks::mutator_lock_); -  bool LoadClassNeedsAccessCheck(ScopedObjectAccess& soa, Handle<mirror::Class> klass) +  bool LoadClassNeedsAccessCheck(Handle<mirror::Class> klass)        REQUIRES_SHARED(Locks::mutator_lock_);    // Builds a `HLoadMethodHandle` loading the given `method_handle_index`. @@ -242,17 +239,6 @@ class HInstructionBuilder : public ValueObject {    // Builds a `HLoadMethodType` loading the given `proto_index`.    void BuildLoadMethodType(dex::ProtoIndex proto_index, uint32_t dex_pc); -  // Returns the outer-most compiling method's class. -  ObjPtr<mirror::Class> ResolveOutermostCompilingClass(ScopedObjectAccess& soa) const -      REQUIRES_SHARED(Locks::mutator_lock_); - -  // Returns the class whose method is being compiled. -  ObjPtr<mirror::Class> ResolveCompilingClass(ScopedObjectAccess& soa) const -      REQUIRES_SHARED(Locks::mutator_lock_); - -  // Returns whether `type_index` points to the outer-most compiling method's class. -  bool IsOutermostCompilingClass(dex::TypeIndex type_index) const; -    void PotentiallySimplifyFakeString(uint16_t original_dex_register,                                       uint32_t dex_pc,                                       HInvoke* invoke); @@ -275,7 +261,6 @@ class HInstructionBuilder : public ValueObject {    void HandleStringInitResult(HInvokeStaticOrDirect* invoke);    HClinitCheck* ProcessClinitCheckForInvoke( -      ScopedObjectAccess& soa,        uint32_t dex_pc,        ArtMethod* method,        HInvokeStaticOrDirect::ClinitCheckRequirement* clinit_check_requirement) @@ -289,7 +274,7 @@ class HInstructionBuilder : public ValueObject {    void BuildConstructorFenceForAllocation(HInstruction* allocation);    // Return whether the compiler can assume `cls` is initialized. -  bool IsInitialized(ScopedObjectAccess& soa, Handle<mirror::Class> cls) const +  bool IsInitialized(Handle<mirror::Class> cls) const        REQUIRES_SHARED(Locks::mutator_lock_);    // Try to resolve a method using the class linker. Return null if a method could @@ -320,8 +305,6 @@ class HInstructionBuilder : public ValueObject {    HBasicBlockBuilder* const block_builder_;    SsaBuilder* const ssa_builder_; -  CompilerDriver* const compiler_driver_; -    CodeGenerator* const code_generator_;    // The compilation unit of the current method being compiled. Note that @@ -351,6 +334,10 @@ class HInstructionBuilder : public ValueObject {    ScopedArenaVector<HBasicBlock*> loop_headers_; +  // Cached resolved types for the current compilation unit's DexFile. +  // Handle<>s reference entries in the `handles_`. +  ScopedArenaSafeMap<dex::TypeIndex, Handle<mirror::Class>> class_cache_; +    static constexpr int kDefaultNumberOfLoops = 2;    DISALLOW_COPY_AND_ASSIGN(HInstructionBuilder); diff --git a/compiler/optimizing/instruction_simplifier_arm.cc b/compiler/optimizing/instruction_simplifier_arm.cc index 24fbb6cb4c..f968c19cf9 100644 --- a/compiler/optimizing/instruction_simplifier_arm.cc +++ b/compiler/optimizing/instruction_simplifier_arm.cc @@ -202,6 +202,11 @@ void InstructionSimplifierArmVisitor::VisitArrayGet(HArrayGet* instruction) {      return;    } +  // TODO: Support intermediate address for object arrays on arm. +  if (type == DataType::Type::kReference) { +    return; +  } +    if (type == DataType::Type::kInt64        || type == DataType::Type::kFloat32        || type == DataType::Type::kFloat64) { diff --git a/compiler/optimizing/instruction_simplifier_shared.cc b/compiler/optimizing/instruction_simplifier_shared.cc index ccdcb3532d..0f30f662cd 100644 --- a/compiler/optimizing/instruction_simplifier_shared.cc +++ b/compiler/optimizing/instruction_simplifier_shared.cc @@ -245,11 +245,11 @@ bool TryExtractArrayAccessAddress(HInstruction* access,      return false;    }    if (kEmitCompilerReadBarrier && +      !kUseBakerReadBarrier &&        access->IsArrayGet() &&        access->GetType() == DataType::Type::kReference) { -    // For object arrays, the read barrier instrumentation requires +    // For object arrays, the non-Baker read barrier instrumentation requires      // the original array pointer. -    // TODO: This can be relaxed for Baker CC.      return false;    } diff --git a/compiler/optimizing/intrinsics.h b/compiler/optimizing/intrinsics.h index 8245453ab5..5bd1122698 100644 --- a/compiler/optimizing/intrinsics.h +++ b/compiler/optimizing/intrinsics.h @@ -24,7 +24,6 @@  namespace art { -class CompilerDriver;  class DexFile;  // Positive floating-point infinities. diff --git a/compiler/optimizing/intrinsics_arm64.cc b/compiler/optimizing/intrinsics_arm64.cc index 7684dc79f2..6d04b0e9d9 100644 --- a/compiler/optimizing/intrinsics_arm64.cc +++ b/compiler/optimizing/intrinsics_arm64.cc @@ -2916,6 +2916,40 @@ void IntrinsicLocationsBuilderARM64::VisitReachabilityFence(HInvoke* invoke) {  void IntrinsicCodeGeneratorARM64::VisitReachabilityFence(HInvoke* invoke ATTRIBUTE_UNUSED) { } +void IntrinsicLocationsBuilderARM64::VisitCRC32Update(HInvoke* invoke) { +  if (!codegen_->GetInstructionSetFeatures().HasCRC()) { +    return; +  } + +  LocationSummary* locations = new (allocator_) LocationSummary(invoke, +                                                                LocationSummary::kNoCall, +                                                                kIntrinsified); + +  locations->SetInAt(0, Location::RequiresRegister()); +  locations->SetInAt(1, Location::RequiresRegister()); +  locations->SetOut(Location::RequiresRegister()); +} + +// Lower the invoke of CRC32.update(int crc, int b). +void IntrinsicCodeGeneratorARM64::VisitCRC32Update(HInvoke* invoke) { +  DCHECK(codegen_->GetInstructionSetFeatures().HasCRC()); + +  MacroAssembler* masm = GetVIXLAssembler(); + +  Register crc = InputRegisterAt(invoke, 0); +  Register val = InputRegisterAt(invoke, 1); +  Register out = OutputRegister(invoke); + +  // The general algorithm of the CRC32 calculation is: +  //   crc = ~crc +  //   result = crc32_for_byte(crc, b) +  //   crc = ~result +  // It is directly lowered to three instructions. +  __ Mvn(out, crc); +  __ Crc32b(out, out, val); +  __ Mvn(out, out); +} +  UNIMPLEMENTED_INTRINSIC(ARM64, ReferenceGetReferent)  UNIMPLEMENTED_INTRINSIC(ARM64, StringStringIndexOf); diff --git a/compiler/optimizing/intrinsics_arm_vixl.cc b/compiler/optimizing/intrinsics_arm_vixl.cc index 38e4c8968a..0c463a280e 100644 --- a/compiler/optimizing/intrinsics_arm_vixl.cc +++ b/compiler/optimizing/intrinsics_arm_vixl.cc @@ -3059,6 +3059,7 @@ UNIMPLEMENTED_INTRINSIC(ARMVIXL, MathRoundDouble)   // Could be done by changing  UNIMPLEMENTED_INTRINSIC(ARMVIXL, UnsafeCASLong)     // High register pressure.  UNIMPLEMENTED_INTRINSIC(ARMVIXL, SystemArrayCopyChar)  UNIMPLEMENTED_INTRINSIC(ARMVIXL, ReferenceGetReferent) +UNIMPLEMENTED_INTRINSIC(ARMVIXL, CRC32Update)  UNIMPLEMENTED_INTRINSIC(ARMVIXL, StringStringIndexOf);  UNIMPLEMENTED_INTRINSIC(ARMVIXL, StringStringIndexOfAfter); diff --git a/compiler/optimizing/intrinsics_mips.cc b/compiler/optimizing/intrinsics_mips.cc index 6f7f5e49c1..21fb7d7f1c 100644 --- a/compiler/optimizing/intrinsics_mips.cc +++ b/compiler/optimizing/intrinsics_mips.cc @@ -2696,6 +2696,8 @@ UNIMPLEMENTED_INTRINSIC(MIPS, UnsafeCASLong)  UNIMPLEMENTED_INTRINSIC(MIPS, ReferenceGetReferent)  UNIMPLEMENTED_INTRINSIC(MIPS, SystemArrayCopy) +UNIMPLEMENTED_INTRINSIC(MIPS, CRC32Update) +  UNIMPLEMENTED_INTRINSIC(MIPS, StringStringIndexOf);  UNIMPLEMENTED_INTRINSIC(MIPS, StringStringIndexOfAfter);  UNIMPLEMENTED_INTRINSIC(MIPS, StringBufferAppend); diff --git a/compiler/optimizing/intrinsics_mips64.cc b/compiler/optimizing/intrinsics_mips64.cc index 2eb252908c..4b86f5d423 100644 --- a/compiler/optimizing/intrinsics_mips64.cc +++ b/compiler/optimizing/intrinsics_mips64.cc @@ -2346,6 +2346,7 @@ void IntrinsicCodeGeneratorMIPS64::VisitReachabilityFence(HInvoke* invoke ATTRIB  UNIMPLEMENTED_INTRINSIC(MIPS64, ReferenceGetReferent)  UNIMPLEMENTED_INTRINSIC(MIPS64, SystemArrayCopy) +UNIMPLEMENTED_INTRINSIC(MIPS64, CRC32Update)  UNIMPLEMENTED_INTRINSIC(MIPS64, StringStringIndexOf);  UNIMPLEMENTED_INTRINSIC(MIPS64, StringStringIndexOfAfter); diff --git a/compiler/optimizing/intrinsics_x86.cc b/compiler/optimizing/intrinsics_x86.cc index 3504d7a6f8..6dd4681847 100644 --- a/compiler/optimizing/intrinsics_x86.cc +++ b/compiler/optimizing/intrinsics_x86.cc @@ -2967,6 +2967,7 @@ UNIMPLEMENTED_INTRINSIC(X86, IntegerHighestOneBit)  UNIMPLEMENTED_INTRINSIC(X86, LongHighestOneBit)  UNIMPLEMENTED_INTRINSIC(X86, IntegerLowestOneBit)  UNIMPLEMENTED_INTRINSIC(X86, LongLowestOneBit) +UNIMPLEMENTED_INTRINSIC(X86, CRC32Update)  UNIMPLEMENTED_INTRINSIC(X86, StringStringIndexOf);  UNIMPLEMENTED_INTRINSIC(X86, StringStringIndexOfAfter); diff --git a/compiler/optimizing/intrinsics_x86_64.cc b/compiler/optimizing/intrinsics_x86_64.cc index 96f6eaaf33..7db26dc9be 100644 --- a/compiler/optimizing/intrinsics_x86_64.cc +++ b/compiler/optimizing/intrinsics_x86_64.cc @@ -2732,6 +2732,7 @@ void IntrinsicCodeGeneratorX86_64::VisitReachabilityFence(HInvoke* invoke ATTRIB  UNIMPLEMENTED_INTRINSIC(X86_64, ReferenceGetReferent)  UNIMPLEMENTED_INTRINSIC(X86_64, FloatIsInfinite)  UNIMPLEMENTED_INTRINSIC(X86_64, DoubleIsInfinite) +UNIMPLEMENTED_INTRINSIC(X86_64, CRC32Update)  UNIMPLEMENTED_INTRINSIC(X86_64, StringStringIndexOf);  UNIMPLEMENTED_INTRINSIC(X86_64, StringStringIndexOfAfter); diff --git a/compiler/optimizing/nodes.h b/compiler/optimizing/nodes.h index 6ebe89eae1..21243804fb 100644 --- a/compiler/optimizing/nodes.h +++ b/compiler/optimizing/nodes.h @@ -7402,7 +7402,7 @@ class HMemoryBarrier final : public HExpression<0> {  //     }  //  // See also: -// * CompilerDriver::RequiresConstructorBarrier +// * DexCompilationUnit::RequiresConstructorBarrier  // * QuasiAtomic::ThreadFenceForConstructor  //  class HConstructorFence final : public HVariableInputSizeInstruction { diff --git a/compiler/optimizing/optimizing_compiler.cc b/compiler/optimizing/optimizing_compiler.cc index a95ddff188..4f495b6d81 100644 --- a/compiler/optimizing/optimizing_compiler.cc +++ b/compiler/optimizing/optimizing_compiler.cc @@ -870,7 +870,6 @@ CodeGenerator* OptimizingCompiler::TryCompile(ArenaAllocator* allocator,                            code_item_accessor,                            &dex_compilation_unit,                            &dex_compilation_unit, -                          compiler_driver,                            codegen.get(),                            compilation_stats_.get(),                            interpreter_metadata, @@ -991,7 +990,6 @@ CodeGenerator* OptimizingCompiler::TryCompileIntrinsic(                            CodeItemDebugInfoAccessor(),  // Null code item.                            &dex_compilation_unit,                            &dex_compilation_unit, -                          compiler_driver,                            codegen.get(),                            compilation_stats_.get(),                            /* interpreter_metadata */ ArrayRef<const uint8_t>(), @@ -1056,6 +1054,15 @@ CompiledMethod* OptimizingCompiler::Compile(const DexFile::CodeItem* code_item,      std::unique_ptr<CodeGenerator> codegen;      bool compiled_intrinsic = false;      { +      ScopedObjectAccess soa(Thread::Current()); +      ArtMethod* method = +          runtime->GetClassLinker()->ResolveMethod<ClassLinker::ResolveMode::kCheckICCEAndIAE>( +              method_idx, dex_cache, jclass_loader, /*referrer=*/ nullptr, invoke_type); +      DCHECK_EQ(method == nullptr, soa.Self()->IsExceptionPending()); +      soa.Self()->ClearException();  // Suppress exception if any. +      VariableSizedHandleScope handles(soa.Self()); +      Handle<mirror::Class> compiling_class = +          handles.NewHandle(method != nullptr ? method->GetDeclaringClass() : nullptr);        DexCompilationUnit dex_compilation_unit(            jclass_loader,            runtime->GetClassLinker(), @@ -1064,12 +1071,9 @@ CompiledMethod* OptimizingCompiler::Compile(const DexFile::CodeItem* code_item,            class_def_idx,            method_idx,            access_flags, -          /* verified_method */ nullptr,  // Not needed by the Optimizing compiler. -          dex_cache); -      ScopedObjectAccess soa(Thread::Current()); -      ArtMethod* method = compiler_driver->ResolveMethod( -            soa, dex_cache, jclass_loader, &dex_compilation_unit, method_idx, invoke_type); -      VariableSizedHandleScope handles(soa.Self()); +          /*verified_method=*/ nullptr,  // Not needed by the Optimizing compiler. +          dex_cache, +          compiling_class);        // Go to native so that we don't block GC during compilation.        ScopedThreadSuspension sts(soa.Self(), kNative);        if (method != nullptr && UNLIKELY(method->IsIntrinsic())) { @@ -1171,21 +1175,23 @@ CompiledMethod* OptimizingCompiler::JniCompile(uint32_t access_flags,    if (compiler_options.IsBootImage()) {      ScopedObjectAccess soa(Thread::Current());      ArtMethod* method = runtime->GetClassLinker()->LookupResolvedMethod( -        method_idx, dex_cache.Get(), /* class_loader */ nullptr); +        method_idx, dex_cache.Get(), /*class_loader=*/ nullptr);      if (method != nullptr && UNLIKELY(method->IsIntrinsic())) { +      VariableSizedHandleScope handles(soa.Self());        ScopedNullHandle<mirror::ClassLoader> class_loader;  // null means boot class path loader. +      Handle<mirror::Class> compiling_class = handles.NewHandle(method->GetDeclaringClass());        DexCompilationUnit dex_compilation_unit(            class_loader,            runtime->GetClassLinker(),            dex_file, -          /* code_item */ nullptr, -          /* class_def_idx */ DexFile::kDexNoIndex16, +          /*code_item=*/ nullptr, +          /*class_def_idx=*/ DexFile::kDexNoIndex16,            method_idx,            access_flags, -          /* verified_method */ nullptr, -          dex_cache); +          /*verified_method=*/ nullptr, +          dex_cache, +          compiling_class);        CodeVectorAllocator code_allocator(&allocator); -      VariableSizedHandleScope handles(soa.Self());        // Go to native so that we don't block GC during compilation.        ScopedThreadSuspension sts(soa.Self(), kNative);        std::unique_ptr<CodeGenerator> codegen( @@ -1349,6 +1355,7 @@ bool OptimizingCompiler::JitCompile(Thread* self,    std::unique_ptr<CodeGenerator> codegen;    { +    Handle<mirror::Class> compiling_class = handles.NewHandle(method->GetDeclaringClass());      DexCompilationUnit dex_compilation_unit(          class_loader,          runtime->GetClassLinker(), @@ -1357,8 +1364,9 @@ bool OptimizingCompiler::JitCompile(Thread* self,          class_def_idx,          method_idx,          access_flags, -        /* verified_method */ nullptr, -        dex_cache); +        /*verified_method=*/ nullptr, +        dex_cache, +        compiling_class);      // Go to native so that we don't block GC during compilation.      ScopedThreadSuspension sts(self, kNative); |