diff options
Diffstat (limited to 'compiler/optimizing/builder.cc')
| -rw-r--r-- | compiler/optimizing/builder.cc | 161 | 
1 files changed, 101 insertions, 60 deletions
| diff --git a/compiler/optimizing/builder.cc b/compiler/optimizing/builder.cc index cb36f62235..5acc5fda71 100644 --- a/compiler/optimizing/builder.cc +++ b/compiler/optimizing/builder.cc @@ -940,7 +940,8 @@ HClinitCheck* HGraphBuilder::ProcessClinitCheckForInvoke(            storage_index,            *dex_compilation_unit_->GetDexFile(),            is_outer_class, -          dex_pc); +          dex_pc, +          /*needs_access_check*/ false);        current_block_->AddInstruction(load_class);        clinit_check = new (arena_) HClinitCheck(load_class, dex_pc);        current_block_->AddInstruction(clinit_check); @@ -1186,6 +1187,12 @@ void HGraphBuilder::PotentiallySimplifyFakeString(uint16_t original_dex_register    }  } +static Primitive::Type GetFieldAccessType(const DexFile& dex_file, uint16_t field_index) { +  const DexFile::FieldId& field_id = dex_file.GetFieldId(field_index); +  const char* type = dex_file.GetFieldTypeDescriptor(field_id); +  return Primitive::GetType(type[0]); +} +  bool HGraphBuilder::BuildInstanceFieldAccess(const Instruction& instruction,                                               uint32_t dex_pc,                                               bool is_put) { @@ -1205,44 +1212,61 @@ bool HGraphBuilder::BuildInstanceFieldAccess(const Instruction& instruction,    ArtField* resolved_field =        compiler_driver_->ComputeInstanceFieldInfo(field_index, dex_compilation_unit_, is_put, soa); -  if (resolved_field == nullptr) { -    MaybeRecordStat(MethodCompilationStat::kNotCompiledUnresolvedField); -    return false; -  } - -  Primitive::Type field_type = resolved_field->GetTypeAsPrimitiveType();    HInstruction* object = LoadLocal(obj_reg, Primitive::kPrimNot, dex_pc); -  current_block_->AddInstruction(new (arena_) HNullCheck(object, dex_pc)); +  HInstruction* null_check = new (arena_) HNullCheck(object, dex_pc); +  current_block_->AddInstruction(null_check); + +  Primitive::Type field_type = (resolved_field == nullptr) +      ? GetFieldAccessType(*dex_file_, field_index) +      : resolved_field->GetTypeAsPrimitiveType();    if (is_put) {      Temporaries temps(graph_); -    HInstruction* null_check = current_block_->GetLastInstruction();      // We need one temporary for the null check.      temps.Add(null_check);      HInstruction* value = LoadLocal(source_or_dest_reg, field_type, dex_pc); -    current_block_->AddInstruction(new (arena_) HInstanceFieldSet( -        null_check, -        value, -        field_type, -        resolved_field->GetOffset(), -        resolved_field->IsVolatile(), -        field_index, -        *dex_file_, -        dex_compilation_unit_->GetDexCache(), -        dex_pc)); +    HInstruction* field_set = nullptr; +    if (resolved_field == nullptr) { +      MaybeRecordStat(MethodCompilationStat::kUnresolvedField); +      field_set = new (arena_) HUnresolvedInstanceFieldSet(null_check, +                                                           value, +                                                           field_type, +                                                           field_index, +                                                           dex_pc); +    } else { +      field_set = new (arena_) HInstanceFieldSet(null_check, +                                                 value, +                                                 field_type, +                                                 resolved_field->GetOffset(), +                                                 resolved_field->IsVolatile(), +                                                 field_index, +                                                 *dex_file_, +                                                 dex_compilation_unit_->GetDexCache(), +                                                 dex_pc); +    } +    current_block_->AddInstruction(field_set);    } else { -    current_block_->AddInstruction(new (arena_) HInstanceFieldGet( -        current_block_->GetLastInstruction(), -        field_type, -        resolved_field->GetOffset(), -        resolved_field->IsVolatile(), -        field_index, -        *dex_file_, -        dex_compilation_unit_->GetDexCache(), -        dex_pc)); - -    UpdateLocal(source_or_dest_reg, current_block_->GetLastInstruction(), dex_pc); +    HInstruction* field_get = nullptr; +    if (resolved_field == nullptr) { +      MaybeRecordStat(MethodCompilationStat::kUnresolvedField); +      field_get = new (arena_) HUnresolvedInstanceFieldGet(null_check, +                                                           field_type, +                                                           field_index, +                                                           dex_pc); +    } else { +      field_get = new (arena_) HInstanceFieldGet(null_check, +                                                 field_type, +                                                 resolved_field->GetOffset(), +                                                 resolved_field->IsVolatile(), +                                                 field_index, +                                                 *dex_file_, +                                                 dex_compilation_unit_->GetDexCache(), +                                                 dex_pc); +    } +    current_block_->AddInstruction(field_get); +    UpdateLocal(source_or_dest_reg, field_get, dex_pc);    } +    return true;  } @@ -1282,6 +1306,23 @@ bool HGraphBuilder::IsOutermostCompilingClass(uint16_t type_index) const {    return outer_class.Get() == cls.Get();  } +void HGraphBuilder::BuildUnresolvedStaticFieldAccess(const Instruction& instruction, +                                                     uint32_t dex_pc, +                                                     bool is_put, +                                                     Primitive::Type field_type) { +  uint32_t source_or_dest_reg = instruction.VRegA_21c(); +  uint16_t field_index = instruction.VRegB_21c(); + +  if (is_put) { +    HInstruction* value = LoadLocal(source_or_dest_reg, field_type, dex_pc); +    current_block_->AddInstruction( +        new (arena_) HUnresolvedStaticFieldSet(value, field_type, field_index, dex_pc)); +  } else { +    current_block_->AddInstruction( +        new (arena_) HUnresolvedStaticFieldGet(field_type, field_index, dex_pc)); +    UpdateLocal(source_or_dest_reg, current_block_->GetLastInstruction(), dex_pc); +  } +}  bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,                                             uint32_t dex_pc,                                             bool is_put) { @@ -1299,10 +1340,13 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,        soa, dex_cache, class_loader, dex_compilation_unit_, field_index, true);    if (resolved_field == nullptr) { -    MaybeRecordStat(MethodCompilationStat::kNotCompiledUnresolvedField); -    return false; +    MaybeRecordStat(MethodCompilationStat::kUnresolvedField); +    Primitive::Type field_type = GetFieldAccessType(*dex_file_, field_index); +    BuildUnresolvedStaticFieldAccess(instruction, dex_pc, is_put, field_type); +    return true;    } +  Primitive::Type field_type = resolved_field->GetTypeAsPrimitiveType();    const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile();    Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle(        outer_compilation_unit_->GetClassLinker()->FindDexCache(soa.Self(), outer_dex_file))); @@ -1317,6 +1361,7 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,      // The compiler driver cannot currently understand multiple dex caches involved. Just bailout.      return false;    } else { +    // TODO: This is rather expensive. Perf it and cache the results if needed.      std::pair<bool, bool> pair = compiler_driver_->IsFastStaticField(          outer_dex_cache.Get(),          GetCompilingClass(), @@ -1325,7 +1370,9 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,          &storage_index);      bool can_easily_access = is_put ? pair.second : pair.first;      if (!can_easily_access) { -      return false; +      MaybeRecordStat(MethodCompilationStat::kUnresolvedFieldNotAFastAccess); +      BuildUnresolvedStaticFieldAccess(instruction, dex_pc, is_put, field_type); +      return true;      }    } @@ -1338,7 +1385,8 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,                                                   storage_index,                                                   *dex_compilation_unit_->GetDexFile(),                                                   is_outer_class, -                                                 dex_pc); +                                                 dex_pc, +                                                 /*needs_access_check*/ false);    current_block_->AddInstruction(constant);    HInstruction* cls = constant; @@ -1346,8 +1394,6 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,      cls = new (arena_) HClinitCheck(constant, dex_pc);      current_block_->AddInstruction(cls);    } - -  Primitive::Type field_type = resolved_field->GetTypeAsPrimitiveType();    if (is_put) {      // We need to keep the class alive before loading the value.      Temporaries temps(graph_); @@ -1571,7 +1617,9 @@ void HGraphBuilder::BuildFillWideArrayData(HInstruction* object,  static TypeCheckKind ComputeTypeCheckKind(Handle<mirror::Class> cls)      SHARED_REQUIRES(Locks::mutator_lock_) { -  if (cls->IsInterface()) { +  if (cls.Get() == nullptr) { +    return TypeCheckKind::kUnresolvedCheck; +  } else if (cls->IsInterface()) {      return TypeCheckKind::kInterfaceCheck;    } else if (cls->IsArrayClass()) {      if (cls->GetComponentType()->IsObjectClass()) { @@ -1590,11 +1638,20 @@ static TypeCheckKind ComputeTypeCheckKind(Handle<mirror::Class> cls)    }  } -bool HGraphBuilder::BuildTypeCheck(const Instruction& instruction, +void HGraphBuilder::BuildTypeCheck(const Instruction& instruction,                                     uint8_t destination,                                     uint8_t reference,                                     uint16_t type_index,                                     uint32_t dex_pc) { +  bool type_known_final, type_known_abstract, use_declaring_class; +  bool can_access = compiler_driver_->CanAccessTypeWithoutChecks( +      dex_compilation_unit_->GetDexMethodIndex(), +      *dex_compilation_unit_->GetDexFile(), +      type_index, +      &type_known_final, +      &type_known_abstract, +      &use_declaring_class); +    ScopedObjectAccess soa(Thread::Current());    StackHandleScope<2> hs(soa.Self());    Handle<mirror::DexCache> dex_cache(hs.NewHandle( @@ -1602,22 +1659,14 @@ bool HGraphBuilder::BuildTypeCheck(const Instruction& instruction,            soa.Self(), *dex_compilation_unit_->GetDexFile())));    Handle<mirror::Class> resolved_class(hs.NewHandle(dex_cache->GetResolvedType(type_index))); -  if ((resolved_class.Get() == nullptr) || -       // TODO: Remove this check once the compiler actually knows which -       // ArtMethod it is compiling. -      (GetCompilingClass() == nullptr) || -      !GetCompilingClass()->CanAccess(resolved_class.Get())) { -    MaybeRecordStat(MethodCompilationStat::kNotCompiledCantAccesType); -    return false; -  } -    HInstruction* object = LoadLocal(reference, Primitive::kPrimNot, dex_pc);    HLoadClass* cls = new (arena_) HLoadClass(        graph_->GetCurrentMethod(),        type_index,        *dex_compilation_unit_->GetDexFile(),        IsOutermostCompilingClass(type_index), -      dex_pc); +      dex_pc, +      !can_access);    current_block_->AddInstruction(cls);    // The class needs a temporary before being used by the type check. @@ -1632,7 +1681,6 @@ bool HGraphBuilder::BuildTypeCheck(const Instruction& instruction,      DCHECK_EQ(instruction.Opcode(), Instruction::CHECK_CAST);      current_block_->AddInstruction(new (arena_) HCheckCast(object, cls, check_kind, dex_pc));    } -  return true;  }  bool HGraphBuilder::NeedsAccessCheck(uint32_t type_index) const { @@ -2747,16 +2795,13 @@ bool HGraphBuilder::AnalyzeDexInstruction(const Instruction& instruction, uint32        bool can_access = compiler_driver_->CanAccessTypeWithoutChecks(            dex_compilation_unit_->GetDexMethodIndex(), *dex_file_, type_index,            &type_known_final, &type_known_abstract, &dont_use_is_referrers_class); -      if (!can_access) { -        MaybeRecordStat(MethodCompilationStat::kNotCompiledCantAccesType); -        return false; -      }        current_block_->AddInstruction(new (arena_) HLoadClass(            graph_->GetCurrentMethod(),            type_index,            *dex_compilation_unit_->GetDexFile(),            IsOutermostCompilingClass(type_index), -          dex_pc)); +          dex_pc, +          !can_access));        UpdateLocal(instruction.VRegA_21c(), current_block_->GetLastInstruction(), dex_pc);        break;      } @@ -2783,18 +2828,14 @@ bool HGraphBuilder::AnalyzeDexInstruction(const Instruction& instruction, uint32        uint8_t destination = instruction.VRegA_22c();        uint8_t reference = instruction.VRegB_22c();        uint16_t type_index = instruction.VRegC_22c(); -      if (!BuildTypeCheck(instruction, destination, reference, type_index, dex_pc)) { -        return false; -      } +      BuildTypeCheck(instruction, destination, reference, type_index, dex_pc);        break;      }      case Instruction::CHECK_CAST: {        uint8_t reference = instruction.VRegA_21c();        uint16_t type_index = instruction.VRegB_21c(); -      if (!BuildTypeCheck(instruction, -1, reference, type_index, dex_pc)) { -        return false; -      } +      BuildTypeCheck(instruction, -1, reference, type_index, dex_pc);        break;      } |