diff options
Diffstat (limited to 'compiler/optimizing/builder.cc')
| -rw-r--r-- | compiler/optimizing/builder.cc | 100 | 
1 files changed, 69 insertions, 31 deletions
| diff --git a/compiler/optimizing/builder.cc b/compiler/optimizing/builder.cc index 1f9287cbfc..cafa554177 100644 --- a/compiler/optimizing/builder.cc +++ b/compiler/optimizing/builder.cc @@ -603,7 +603,12 @@ bool HGraphBuilder::BuildInvoke(const Instruction& instruction,    const char* descriptor = dex_file_->StringDataByIdx(proto_id.shorty_idx_);    Primitive::Type return_type = Primitive::GetType(descriptor[0]);    bool is_instance_call = invoke_type != kStatic; -  size_t number_of_arguments = strlen(descriptor) - (is_instance_call ? 0 : 1); +  // Remove the return type from the 'proto'. +  size_t number_of_arguments = strlen(descriptor) - 1; +  if (is_instance_call) { +    // One extra argument for 'this'. +    ++number_of_arguments; +  }    MethodReference target_method(dex_file_, method_idx);    uintptr_t direct_code; @@ -614,7 +619,8 @@ bool HGraphBuilder::BuildInvoke(const Instruction& instruction,    if (!compiler_driver_->ComputeInvokeInfo(dex_compilation_unit_, dex_pc, true, true,                                             &optimized_invoke_type, &target_method, &table_index,                                             &direct_code, &direct_method)) { -    VLOG(compiler) << "Did not compile " << PrettyMethod(method_idx, *dex_file_) +    VLOG(compiler) << "Did not compile " +                   << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_)                     << " because a method call could not be resolved";      MaybeRecordStat(MethodCompilationStat::kNotCompiledUnresolvedMethod);      return false; @@ -746,26 +752,45 @@ bool HGraphBuilder::BuildInvoke(const Instruction& instruction,      start_index = 1;    } -  uint32_t descriptor_index = 1; +  uint32_t descriptor_index = 1;  // Skip the return type.    uint32_t argument_index = start_index;    if (is_string_init) {      start_index = 1;    } -  for (size_t i = start_index; i < number_of_vreg_arguments; i++, argument_index++) { +  for (size_t i = start_index; +       // Make sure we don't go over the expected arguments or over the number of +       // dex registers given. If the instruction was seen as dead by the verifier, +       // it hasn't been properly checked. +       (i < number_of_vreg_arguments) && (argument_index < number_of_arguments); +       i++, argument_index++) {      Primitive::Type type = Primitive::GetType(descriptor[descriptor_index++]);      bool is_wide = (type == Primitive::kPrimLong) || (type == Primitive::kPrimDouble); -    // Longs and doubles should be in pairs, that is, sequential registers. The verifier should -    // reject any class where this is violated. -    DCHECK(is_range || !is_wide || (args[i] + 1 == args[i + 1])) -        << "Non sequential register pair in " << dex_compilation_unit_->GetSymbol() -        << " at " << dex_pc; +    if (!is_range +        && is_wide +        && ((i + 1 == number_of_vreg_arguments) || (args[i] + 1 != args[i + 1]))) { +      // Longs and doubles should be in pairs, that is, sequential registers. The verifier should +      // reject any class where this is violated. However, the verifier only does these checks +      // on non trivially dead instructions, so we just bailout the compilation. +      VLOG(compiler) << "Did not compile " +                     << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_) +                     << " because of non-sequential dex register pair in wide argument"; +      MaybeRecordStat(MethodCompilationStat::kNotCompiledMalformedOpcode); +      return false; +    }      HInstruction* arg = LoadLocal(is_range ? register_index + i : args[i], type);      invoke->SetArgumentAt(argument_index, arg);      if (is_wide) {        i++;      }    } -  DCHECK_EQ(argument_index, number_of_arguments); + +  if (argument_index != number_of_arguments) { +    VLOG(compiler) << "Did not compile " +                   << PrettyMethod(dex_compilation_unit_->GetDexMethodIndex(), *dex_file_) +                   << " because of wrong number of arguments in invoke instruction"; +    MaybeRecordStat(MethodCompilationStat::kNotCompiledMalformedOpcode); +    return false; +  }    if (invoke->IsInvokeStaticOrDirect()) {      invoke->SetArgumentAt(argument_index, graph_->GetCurrentMethod()); @@ -853,17 +878,25 @@ bool HGraphBuilder::BuildInstanceFieldAccess(const Instruction& instruction,    return true;  } -mirror::Class* HGraphBuilder::GetOutermostCompilingClass() const { +static mirror::Class* GetClassFrom(CompilerDriver* driver, +                                   const DexCompilationUnit& compilation_unit) {    ScopedObjectAccess soa(Thread::Current());    StackHandleScope<2> hs(soa.Self()); -  const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile(); +  const DexFile& dex_file = *compilation_unit.GetDexFile();    Handle<mirror::ClassLoader> class_loader(hs.NewHandle( -      soa.Decode<mirror::ClassLoader*>(dex_compilation_unit_->GetClassLoader()))); -  Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle( -      outer_compilation_unit_->GetClassLinker()->FindDexCache(outer_dex_file))); +      soa.Decode<mirror::ClassLoader*>(compilation_unit.GetClassLoader()))); +  Handle<mirror::DexCache> dex_cache(hs.NewHandle( +      compilation_unit.GetClassLinker()->FindDexCache(dex_file))); + +  return driver->ResolveCompilingMethodsClass(soa, dex_cache, class_loader, &compilation_unit); +} + +mirror::Class* HGraphBuilder::GetOutermostCompilingClass() const { +  return GetClassFrom(compiler_driver_, *outer_compilation_unit_); +} -  return compiler_driver_->ResolveCompilingMethodsClass( -      soa, outer_dex_cache, class_loader, outer_compilation_unit_); +mirror::Class* HGraphBuilder::GetCompilingClass() const { +  return GetClassFrom(compiler_driver_, *dex_compilation_unit_);  }  bool HGraphBuilder::IsOutermostCompilingClass(uint16_t type_index) const { @@ -903,20 +936,20 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,    const DexFile& outer_dex_file = *outer_compilation_unit_->GetDexFile();    Handle<mirror::DexCache> outer_dex_cache(hs.NewHandle(        outer_compilation_unit_->GetClassLinker()->FindDexCache(outer_dex_file))); -  Handle<mirror::Class> referrer_class(hs.NewHandle(GetOutermostCompilingClass())); +  Handle<mirror::Class> outer_class(hs.NewHandle(GetOutermostCompilingClass()));    // The index at which the field's class is stored in the DexCache's type array.    uint32_t storage_index; -  bool is_referrer_class = (referrer_class.Get() == resolved_field->GetDeclaringClass()); -  if (is_referrer_class) { -    storage_index = referrer_class->GetDexTypeIndex(); +  bool is_outer_class = (outer_class.Get() == resolved_field->GetDeclaringClass()); +  if (is_outer_class) { +    storage_index = outer_class->GetDexTypeIndex();    } else if (outer_dex_cache.Get() != dex_cache.Get()) {      // The compiler driver cannot currently understand multiple dex caches involved. Just bailout.      return false;    } else {      std::pair<bool, bool> pair = compiler_driver_->IsFastStaticField(          outer_dex_cache.Get(), -        referrer_class.Get(), +        GetCompilingClass(),          resolved_field,          field_index,          &storage_index); @@ -934,12 +967,12 @@ bool HGraphBuilder::BuildStaticFieldAccess(const Instruction& instruction,    HLoadClass* constant = new (arena_) HLoadClass(graph_->GetCurrentMethod(),                                                   storage_index,                                                   *dex_compilation_unit_->GetDexFile(), -                                                 is_referrer_class, +                                                 is_outer_class,                                                   dex_pc);    current_block_->AddInstruction(constant);    HInstruction* cls = constant; -  if (!is_initialized && !is_referrer_class) { +  if (!is_initialized && !is_outer_class) {      cls = new (arena_) HClinitCheck(constant, dex_pc);      current_block_->AddInstruction(cls);    } @@ -1210,14 +1243,20 @@ bool HGraphBuilder::NeedsAccessCheck(uint32_t type_index) const {  }  void HGraphBuilder::BuildPackedSwitch(const Instruction& instruction, uint32_t dex_pc) { +  // Verifier guarantees that the payload for PackedSwitch contains: +  //   (a) number of entries (may be zero) +  //   (b) first and lowest switch case value (entry 0, always present) +  //   (c) list of target pcs (entries 1 <= i <= N)    SwitchTable table(instruction, dex_pc, false);    // Value to test against.    HInstruction* value = LoadLocal(instruction.VRegA(), Primitive::kPrimInt); +  // Retrieve number of entries.    uint16_t num_entries = table.GetNumEntries(); -  // There should be at least one entry here. -  DCHECK_GT(num_entries, 0U); +  if (num_entries == 0) { +    return; +  }    // Chained cmp-and-branch, starting from starting_key.    int32_t starting_key = table.GetEntryAt(0); @@ -1229,6 +1268,10 @@ void HGraphBuilder::BuildPackedSwitch(const Instruction& instruction, uint32_t d  }  void HGraphBuilder::BuildSparseSwitch(const Instruction& instruction, uint32_t dex_pc) { +  // Verifier guarantees that the payload for SparseSwitch contains: +  //   (a) number of entries (may be zero) +  //   (b) sorted key values (entries 0 <= i < N) +  //   (c) target pcs corresponding to the switch values (entries N <= i < 2*N)    SwitchTable table(instruction, dex_pc, true);    // Value to test against. @@ -1428,21 +1471,16 @@ bool HGraphBuilder::AnalyzeDexInstruction(const Instruction& instruction, uint32      }      case Instruction::RETURN: { -      DCHECK_NE(return_type_, Primitive::kPrimNot); -      DCHECK_NE(return_type_, Primitive::kPrimLong); -      DCHECK_NE(return_type_, Primitive::kPrimDouble);        BuildReturn(instruction, return_type_);        break;      }      case Instruction::RETURN_OBJECT: { -      DCHECK(return_type_ == Primitive::kPrimNot);        BuildReturn(instruction, return_type_);        break;      }      case Instruction::RETURN_WIDE: { -      DCHECK(return_type_ == Primitive::kPrimDouble || return_type_ == Primitive::kPrimLong);        BuildReturn(instruction, return_type_);        break;      } |