compiler_driver.cc revision e0099d4d4839cabfb26270a746580e9d4ec3a6c6
12faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes/*
22faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * Copyright (C) 2011 The Android Open Source Project
32faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes *
42faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * Licensed under the Apache License, Version 2.0 (the "License");
52faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * you may not use this file except in compliance with the License.
62faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * You may obtain a copy of the License at
72faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes *
82faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes *      http://www.apache.org/licenses/LICENSE-2.0
92faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes *
102faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * Unless required by applicable law or agreed to in writing, software
112faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * distributed under the License is distributed on an "AS IS" BASIS,
122faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
132faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * See the License for the specific language governing permissions and
142faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes * limitations under the License.
152faa5f1271587cda765f26bcf2951065300a01ffElliott Hughes */
169ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
171212a022fa5f8ef9585d765b1809521812af882cIan Rogers#include "compiler_driver.h"
189ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
1967f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum#define ATRACE_TAG ATRACE_TAG_DALVIK
2067f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum#include <utils/Trace.h>
21d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
2267f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum#include <vector>
23d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes#include <unistd.h>
2427ec961a1da540ba7f16c07a682585ab167317adBrian Carlstrom
251aa246dec5abe212f699de1413a0c4a191ca364aElliott Hughes#include "base/stl_util.h"
26a84395489098e4531619b1cffd1afc282b14510eSameer Abu Asal#include "base/timing_logger.h"
279ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom#include "class_linker.h"
28b34f69ab43aaf7a6e6045c95f398baf566ef5023Nicolas Geoffray#include "compiler.h"
29be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko#include "compiler_driver-inl.h"
3089756f21c107d96e6d1bbc75811bd33078c8ceeeIan Rogers#include "dex_compilation_unit.h"
314f6ad8ab428038129b2d0d6c40b7fd625cca15e1Ian Rogers#include "dex_file-inl.h"
32c7f832061fea59fd6abd125f26c8ca1faec695a5Vladimir Marko#include "dex/verification_results.h"
332730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko#include "dex/verified_method.h"
342bc47809febcf36369dd40877b8226318642b428Vladimir Marko#include "dex/quick/dex_file_method_inliner.h"
35ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell#include "driver/compiler_options.h"
369baa4aefc370f48774b6104680193d9a7e4fb631Brian Carlstrom#include "jni_internal.h"
376d4d9fcb4f01e287ee29e81cd1c941ee5d11d379Ian Rogers#include "object_utils.h"
381f87008b165d26541d832ff805250afdc89c253dBrian Carlstrom#include "runtime.h"
391d54e73444e017d3a65234e0f193846f3e27472bIan Rogers#include "gc/accounting/card_table-inl.h"
401d54e73444e017d3a65234e0f193846f3e27472bIan Rogers#include "gc/accounting/heap_bitmap.h"
411d54e73444e017d3a65234e0f193846f3e27472bIan Rogers#include "gc/space/space.h"
42ea46f950e7a51585db293cd7f047de190a482414Brian Carlstrom#include "mirror/art_field-inl.h"
43ea46f950e7a51585db293cd7f047de190a482414Brian Carlstrom#include "mirror/art_method-inl.h"
442dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers#include "mirror/class_loader.h"
452dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers#include "mirror/class-inl.h"
4639ebcb800aabedd0ffa6aa4aeac8aa4194c66e61Ian Rogers#include "mirror/dex_cache-inl.h"
472dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers#include "mirror/object-inl.h"
482dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers#include "mirror/object_array-inl.h"
492dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers#include "mirror/throwable.h"
5000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers#include "scoped_thread_state_change.h"
5100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers#include "ScopedLocalRef.h"
52eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier#include "handle_scope-inl.h"
5350b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers#include "thread.h"
540e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier#include "thread_pool.h"
55848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers#include "trampolines/trampoline_compiler.h"
56d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz#include "transaction.h"
57776ac1fa61237db645adb4370a4aab888530caf4Ian Rogers#include "verifier/method_verifier.h"
582bc47809febcf36369dd40877b8226318642b428Vladimir Marko#include "verifier/method_verifier-inl.h"
599ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
6004ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle#ifdef HAVE_ANDROID_OS
6104ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle#include "cutils/properties.h"
6204ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle#endif
6304ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle
649ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstromnamespace art {
659ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
66996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogersstatic double Percentage(size_t x, size_t y) {
67398f64b5805246765b699839b439e18c0dfbf2eeElliott Hughes  return 100.0 * (static_cast<double>(x)) / (static_cast<double>(x + y));
68996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers}
69996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
70996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogersstatic void DumpStat(size_t x, size_t y, const char* str) {
71996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  if (x == 0 && y == 0) {
72996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers    return;
73996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
74e732ef1c0192acd71925bd0ff1ab09640d45531dIan Rogers  LOG(INFO) << Percentage(x, y) << "% of " << str << " for " << (x + y) << " cases";
75996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers}
76996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
77f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Markoclass CompilerDriver::AOTCompilationStats {
78c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers public:
79ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers  AOTCompilationStats()
80ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers      : stats_lock_("AOT compilation statistics lock"),
81ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers        types_in_dex_cache_(0), types_not_in_dex_cache_(0),
82ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers        strings_in_dex_cache_(0), strings_not_in_dex_cache_(0),
83ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers        resolved_types_(0), unresolved_types_(0),
84ca190666fb11820153f74274c495ba1f913d8a69Ian Rogers        resolved_instance_fields_(0), unresolved_instance_fields_(0),
8502c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal        resolved_local_static_fields_(0), resolved_static_fields_(0), unresolved_static_fields_(0),
86fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers        type_based_devirtualization_(0),
87fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers        safe_casts_(0), not_safe_casts_(0) {
882ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    for (size_t i = 0; i <= kMaxInvokeType; i++) {
89c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers      resolved_methods_[i] = 0;
90c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers      unresolved_methods_[i] = 0;
912ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      virtual_made_direct_[i] = 0;
922ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      direct_calls_to_boot_[i] = 0;
932ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      direct_methods_to_boot_[i] = 0;
94b25c3f6a86dc634ce44fb2849385b49465caa84dElliott Hughes    }
95c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
96c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
97c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void Dump() {
98c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(types_in_dex_cache_, types_not_in_dex_cache_, "types known to be in dex cache");
99c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(strings_in_dex_cache_, strings_not_in_dex_cache_, "strings known to be in dex cache");
100c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(resolved_types_, unresolved_types_, "types resolved");
101c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(resolved_instance_fields_, unresolved_instance_fields_, "instance fields resolved");
102c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(resolved_local_static_fields_ + resolved_static_fields_, unresolved_static_fields_,
103c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers             "static fields resolved");
104c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DumpStat(resolved_local_static_fields_, resolved_static_fields_ + unresolved_static_fields_,
105c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers             "static fields local to a class");
106fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    DumpStat(safe_casts_, not_safe_casts_, "check-casts removed based on type information");
107fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    // Note, the code below subtracts the stat value so that when added to the stat value we have
108fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    // 100% of samples. TODO: clean this up.
109fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    DumpStat(type_based_devirtualization_,
110fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers             resolved_methods_[kVirtual] + unresolved_methods_[kVirtual] +
111fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers             resolved_methods_[kInterface] + unresolved_methods_[kInterface] -
112fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers             type_based_devirtualization_,
113fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers             "virtual/interface calls made direct based on type information");
114c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
1152ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    for (size_t i = 0; i <= kMaxInvokeType; i++) {
116c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers      std::ostringstream oss;
1172ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      oss << static_cast<InvokeType>(i) << " methods were AOT resolved";
118c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers      DumpStat(resolved_methods_[i], unresolved_methods_[i], oss.str().c_str());
1192ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      if (virtual_made_direct_[i] > 0) {
1202ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        std::ostringstream oss2;
1212ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        oss2 << static_cast<InvokeType>(i) << " methods made direct";
1222ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        DumpStat(virtual_made_direct_[i],
1232ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 resolved_methods_[i] + unresolved_methods_[i] - virtual_made_direct_[i],
1242ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 oss2.str().c_str());
1252ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      }
1262ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      if (direct_calls_to_boot_[i] > 0) {
1272ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        std::ostringstream oss2;
1282ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        oss2 << static_cast<InvokeType>(i) << " method calls are direct into boot";
1292ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        DumpStat(direct_calls_to_boot_[i],
1302ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 resolved_methods_[i] + unresolved_methods_[i] - direct_calls_to_boot_[i],
1312ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 oss2.str().c_str());
1322ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      }
1332ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      if (direct_methods_to_boot_[i] > 0) {
1342ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        std::ostringstream oss2;
1352ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        oss2 << static_cast<InvokeType>(i) << " method calls have methods in boot";
1362ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers        DumpStat(direct_methods_to_boot_[i],
1372ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 resolved_methods_[i] + unresolved_methods_[i] - direct_methods_to_boot_[i],
1382ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers                 oss2.str().c_str());
1392ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers      }
140c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    }
141c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
142996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
14350b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers// Allow lossy statistics in non-debug builds.
144996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers#ifndef NDEBUG
14550b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers#define STATS_LOCK() MutexLock mu(Thread::Current(), stats_lock_)
146996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers#else
147996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers#define STATS_LOCK()
148996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers#endif
149996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
150c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void TypeInDexCache() {
151c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
152c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    types_in_dex_cache_++;
153c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
154996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
155c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void TypeNotInDexCache() {
156c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
157c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    types_not_in_dex_cache_++;
158c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
159996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
160c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void StringInDexCache() {
161c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
162c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    strings_in_dex_cache_++;
163c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
164996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
165c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void StringNotInDexCache() {
166c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
167c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    strings_not_in_dex_cache_++;
168c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
169996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
170c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void TypeDoesntNeedAccessCheck() {
171c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
172c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    resolved_types_++;
173c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
174996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
175c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void TypeNeedsAccessCheck() {
176c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
177c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    unresolved_types_++;
178c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
179996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
180c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void ResolvedInstanceField() {
181c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
182c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    resolved_instance_fields_++;
183c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
184996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
185b25c3f6a86dc634ce44fb2849385b49465caa84dElliott Hughes  void UnresolvedInstanceField() {
186c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
187c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    unresolved_instance_fields_++;
188c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
189996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
190c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void ResolvedLocalStaticField() {
191c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
192c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    resolved_local_static_fields_++;
193c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
194996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
195c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void ResolvedStaticField() {
196c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
197c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    resolved_static_fields_++;
198c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
199996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
200c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void UnresolvedStaticField() {
201c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
202c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    unresolved_static_fields_++;
203c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  }
204996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
205e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that type information from the verifier led to devirtualization.
20602c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal  void PreciseTypeDevirtualization() {
20702c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal    STATS_LOCK();
20802c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal    type_based_devirtualization_++;
20902c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal  }
210e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers
211e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that a method of the given type was resolved at compile time.
212c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void ResolvedMethod(InvokeType type) {
213c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DCHECK_LE(type, kMaxInvokeType);
214c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
215c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    resolved_methods_[type]++;
216996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
217996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
218e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that a method of the given type was unresolved at compile time as it was in an
219e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // unknown dex file.
220c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  void UnresolvedMethod(InvokeType type) {
221c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    DCHECK_LE(type, kMaxInvokeType);
222c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    STATS_LOCK();
223c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    unresolved_methods_[type]++;
224996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
225c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
226e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that a type of virtual method dispatch has been converted into a direct method
227e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // dispatch.
2282ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  void VirtualMadeDirect(InvokeType type) {
229e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers    DCHECK(type == kVirtual || type == kInterface || type == kSuper);
2302ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    STATS_LOCK();
2312ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    virtual_made_direct_[type]++;
2322ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  }
2332ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers
234e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that a method of the given type was able to call directly into boot.
2352ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  void DirectCallsToBoot(InvokeType type) {
2362ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    DCHECK_LE(type, kMaxInvokeType);
2372ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    STATS_LOCK();
2382ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    direct_calls_to_boot_[type]++;
2392ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  }
2402ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers
241e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  // Indicate that a method of the given type was able to be resolved directly from boot.
2422ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  void DirectMethodsToBoot(InvokeType type) {
2432ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    DCHECK_LE(type, kMaxInvokeType);
244fb6adba0d5d5505610fbd325e7911db700a2f1e8Ian Rogers    STATS_LOCK();
2452ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers    direct_methods_to_boot_[type]++;
246fb6adba0d5d5505610fbd325e7911db700a2f1e8Ian Rogers  }
2472ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers
248f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  void ProcessedInvoke(InvokeType type, int flags) {
249f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    STATS_LOCK();
250f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    if (flags == 0) {
251f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      unresolved_methods_[type]++;
252f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    } else {
253f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      DCHECK_NE((flags & kFlagMethodResolved), 0);
254f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      resolved_methods_[type]++;
255f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      if ((flags & kFlagVirtualMadeDirect) != 0) {
256f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        virtual_made_direct_[type]++;
257f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        if ((flags & kFlagPreciseTypeDevirtualization) != 0) {
258f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko          type_based_devirtualization_++;
259f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        }
260f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      } else {
261f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        DCHECK_EQ((flags & kFlagPreciseTypeDevirtualization), 0);
262f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      }
263f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      if ((flags & kFlagDirectCallToBoot) != 0) {
264f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        direct_calls_to_boot_[type]++;
265f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      }
266f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      if ((flags & kFlagDirectMethodToBoot) != 0) {
267f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        direct_methods_to_boot_[type]++;
268f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      }
269f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    }
270f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  }
271f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko
272fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  // A check-cast could be eliminated due to verifier type analysis.
273fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  void SafeCast() {
274fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    STATS_LOCK();
275fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    safe_casts_++;
276fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  }
277fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers
278fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  // A check-cast couldn't be eliminated due to verifier type analysis.
279fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  void NotASafeCast() {
280fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    STATS_LOCK();
281fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    not_safe_casts_++;
282fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  }
283fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers
284c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers private:
285c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  Mutex stats_lock_;
286c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
287c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t types_in_dex_cache_;
288c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t types_not_in_dex_cache_;
289c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
290c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t strings_in_dex_cache_;
291c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t strings_not_in_dex_cache_;
292c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
293c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t resolved_types_;
294c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t unresolved_types_;
295c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
296c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t resolved_instance_fields_;
297c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t unresolved_instance_fields_;
298c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
299c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t resolved_local_static_fields_;
300c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t resolved_static_fields_;
301c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t unresolved_static_fields_;
30202c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal  // Type based devirtualization for invoke interface and virtual.
30302c42237741b5573f9d790a5a0f17f408dceb543Sameer Abu Asal  size_t type_based_devirtualization_;
304c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
305c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t resolved_methods_[kMaxInvokeType + 1];
306c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers  size_t unresolved_methods_[kMaxInvokeType + 1];
3072ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  size_t virtual_made_direct_[kMaxInvokeType + 1];
3082ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  size_t direct_calls_to_boot_[kMaxInvokeType + 1];
3092ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  size_t direct_methods_to_boot_[kMaxInvokeType + 1];
310c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers
311fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  size_t safe_casts_;
312fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  size_t not_safe_casts_;
313fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers
314b25c3f6a86dc634ce44fb2849385b49465caa84dElliott Hughes  DISALLOW_COPY_AND_ASSIGN(AOTCompilationStats);
315c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers};
316996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers
31751c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom
31851c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstromextern "C" art::CompiledMethod* ArtCompileDEX(art::CompilerDriver& compiler,
31951c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              const art::DexFile::CodeItem* code_item,
32051c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              uint32_t access_flags,
32151c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              art::InvokeType invoke_type,
3228b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                              uint16_t class_def_idx,
32351c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              uint32_t method_idx,
32451c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              jobject class_loader,
32551c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                                              const art::DexFile& dex_file);
32651c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom
3276449c62e40ef3a9bb75f664f922555affb532ee4Brian CarlstromCompilerDriver::CompilerDriver(const CompilerOptions* compiler_options,
3286449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom                               VerificationResults* verification_results,
3295816ed48bc339c983b40dc493e96b97821ce7966Vladimir Marko                               DexFileToMethodInlinerMap* method_inliner_map,
330b34f69ab43aaf7a6e6045c95f398baf566ef5023Nicolas Geoffray                               Compiler::Kind compiler_kind,
331f5df8974173124faddb8e2b6a331959afdb94fdfNicolas Geoffray                               InstructionSet instruction_set,
3327020278bce98a0735dc6abcbd33bdf1ed2634f1dDave Allison                               InstructionSetFeatures instruction_set_features,
333a024a0686c3b0fea13f362bff70d65981e5febc5buzbee                               bool image, DescriptorSet* image_classes, size_t thread_count,
33439c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison                               bool dump_stats, bool dump_passes, CumulativeLogger* timer,
33539c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison                               std::string profile_file)
33639c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison    : profile_ok_(false), compiler_options_(compiler_options),
3376449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom      verification_results_(verification_results),
3385816ed48bc339c983b40dc493e96b97821ce7966Vladimir Marko      method_inliner_map_(method_inliner_map),
33972d32629303f8f39362a4099481f48646aed042fIan Rogers      compiler_(Compiler::Create(this, compiler_kind)),
340c531cefbfb5394413122e9f57d211ba436cff012buzbee      instruction_set_(instruction_set),
3417020278bce98a0735dc6abcbd33bdf1ed2634f1dDave Allison      instruction_set_features_(instruction_set_features),
342fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers      freezing_constructor_lock_("freezing constructor lock"),
343c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes      compiled_classes_lock_("compiled classes lock"),
344c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes      compiled_methods_lock_("compiled method lock"),
345aded5f7ab991f3c1132851599d3bc60ff6707eedBrian Carlstrom      image_(image),
34696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      image_classes_(image_classes),
3475523ee070b005576c6f889415205d49ea77cf243Elliott Hughes      thread_count_(thread_count),
34800f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers      start_ns_(0),
349c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers      stats_(new AOTCompilationStats),
350ba0668ecd2a6459ed7c77012995ad08d27f88725Brian Carlstrom      dump_stats_(dump_stats),
351ea3fa0b4ba13d7bd7f7c1cd85202ccbe141a35aeNicolas Geoffray      dump_passes_(dump_passes),
352ea3fa0b4ba13d7bd7f7c1cd85202ccbe141a35aeNicolas Geoffray      timings_logger_(timer),
353b3bd5f07884f5a1f2b84224363b1372d7c28d447Elliott Hughes      compiler_library_(NULL),
3546f4976c1a9fdaf108974143cd11e6b46037fd24eElliott Hughes      compiler_context_(NULL),
355e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers      compiler_enable_auto_elf_loading_(NULL),
35696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      compiler_get_method_code_addr_(NULL),
35755d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell      support_boot_image_fixup_(instruction_set != kMips),
358ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell      cfi_info_(nullptr),
359d133b97b1ccae88f6ee7040e288fd7a239ee4492Ian Rogers      dedupe_code_("dedupe code"),
360d133b97b1ccae88f6ee7040e288fd7a239ee4492Ian Rogers      dedupe_mapping_table_("dedupe mapping table"),
361d133b97b1ccae88f6ee7040e288fd7a239ee4492Ian Rogers      dedupe_vmap_table_("dedupe vmap table"),
362ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell      dedupe_gc_map_("dedupe gc map"),
363ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell      dedupe_cfi_info_("dedupe cfi info") {
3646449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom  DCHECK(compiler_options_ != nullptr);
3656449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom  DCHECK(verification_results_ != nullptr);
3666449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom  DCHECK(method_inliner_map_ != nullptr);
3672ce745c06271d5223d57dbf08117b20d5b60694aBrian Carlstrom
3684df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  CHECK_PTHREAD_CALL(pthread_key_create, (&tls_key_, NULL), "compiler tls key");
3694df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee
37039c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  // Read the profile file if one is provided.
37139c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  if (profile_file != "") {
3729dae5b4b952824da45e9fd9c12cfcde9858f0974Calin Juravle    profile_ok_ = ProfileHelper::LoadProfileMap(profile_map_, profile_file);
37339c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  }
37439c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison
37575021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  dex_to_dex_compiler_ = reinterpret_cast<DexToDexCompilerFn>(ArtCompileDEX);
3762d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz
37772d32629303f8f39362a4099481f48646aed042fIan Rogers  compiler_->Init();
37800bc1dc4f81268d78d7dfeb298b85c56876425a9Brian Carlstrom
37925c3325bf95036bf325fc7cb21b4fd6d40282857Brian Carlstrom  CHECK(!Runtime::Current()->IsStarted());
380d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root  if (image_) {
381d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root    CHECK(image_classes_.get() != nullptr);
382d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root  } else {
383d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root    CHECK(image_classes_.get() == nullptr);
384ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  }
385ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell
386ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  // Are we generating CFI information?
387ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  if (compiler_options->GetGenerateGDBInformation()) {
388b34f69ab43aaf7a6e6045c95f398baf566ef5023Nicolas Geoffray    cfi_info_.reset(compiler_->GetCallFrameInformationInitialization(*this));
389ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  }
390c486c11a1c71ca9c118d57152427b741229cda49Shih-wei Liao}
391c486c11a1c71ca9c118d57152427b741229cda49Shih-wei Liao
392193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartierstd::vector<uint8_t>* CompilerDriver::DeduplicateCode(const std::vector<uint8_t>& code) {
393193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier  return dedupe_code_.Add(Thread::Current(), code);
394193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier}
395193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier
396193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartierstd::vector<uint8_t>* CompilerDriver::DeduplicateMappingTable(const std::vector<uint8_t>& code) {
397193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier  return dedupe_mapping_table_.Add(Thread::Current(), code);
398193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier}
399193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier
400193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartierstd::vector<uint8_t>* CompilerDriver::DeduplicateVMapTable(const std::vector<uint8_t>& code) {
401193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier  return dedupe_vmap_table_.Add(Thread::Current(), code);
402193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier}
403193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier
404193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartierstd::vector<uint8_t>* CompilerDriver::DeduplicateGCMap(const std::vector<uint8_t>& code) {
405193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier  return dedupe_gc_map_.Add(Thread::Current(), code);
406193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier}
407193bad9b9cfd10642043fa2ebbfc68bd5f9ede4bMathieu Chartier
408ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendellstd::vector<uint8_t>* CompilerDriver::DeduplicateCFIInfo(const std::vector<uint8_t>* cfi_info) {
409ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  if (cfi_info == nullptr) {
410ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell    return nullptr;
411ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  }
412ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell  return dedupe_cfi_info_.Add(Thread::Current(), *cfi_info);
413ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell}
414ae9fd93c39a341e2dffe15c61cc7d9e841fa92c4Mark Mendell
4151212a022fa5f8ef9585d765b1809521812af882cIan RogersCompilerDriver::~CompilerDriver() {
41650b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  Thread* self = Thread::Current();
417c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  {
41850b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers    MutexLock mu(self, compiled_classes_lock_);
419c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes    STLDeleteValues(&compiled_classes_);
420c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  }
421c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  {
42250b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers    MutexLock mu(self, compiled_methods_lock_);
423c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes    STLDeleteValues(&compiled_methods_);
424c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  }
425c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  {
42650b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers    MutexLock mu(self, compiled_methods_lock_);
427f582258f0e296223a091fd64231a203ad71e9649Brian Carlstrom    STLDeleteElements(&code_to_patch_);
428f582258f0e296223a091fd64231a203ad71e9649Brian Carlstrom  }
429f582258f0e296223a091fd64231a203ad71e9649Brian Carlstrom  {
43050b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers    MutexLock mu(self, compiled_methods_lock_);
431f582258f0e296223a091fd64231a203ad71e9649Brian Carlstrom    STLDeleteElements(&methods_to_patch_);
432f582258f0e296223a091fd64231a203ad71e9649Brian Carlstrom  }
433be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  {
434be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    MutexLock mu(self, compiled_methods_lock_);
435be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    STLDeleteElements(&classes_to_patch_);
436be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  }
437ed6d5ed3494648780b9e91422c90d6bc22c16b79Mathieu Chartier  CHECK_PTHREAD_CALL(pthread_key_delete, (tls_key_), "delete tls key");
43872d32629303f8f39362a4099481f48646aed042fIan Rogers  compiler_->UnInit();
4393320cf46afd082398aa401b246e6f301cebdf64dBrian Carlstrom}
4403320cf46afd082398aa401b246e6f301cebdf64dBrian Carlstrom
4411212a022fa5f8ef9585d765b1809521812af882cIan RogersCompilerTls* CompilerDriver::GetTls() {
4424df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  // Lazily create thread-local storage
4434df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  CompilerTls* res = static_cast<CompilerTls*>(pthread_getspecific(tls_key_));
4444df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  if (res == NULL) {
4454df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee    res = new CompilerTls();
4464df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee    CHECK_PTHREAD_CALL(pthread_setspecific, (tls_key_, res), "compiler tls");
4474df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  }
4484df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee  return res;
4494df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee}
4504df2bbdfe6602ce5f141b7b44028b95faa0bd8efbuzbee
451dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers#define CREATE_TRAMPOLINE(type, abi, offset) \
452af13ad9fd18b6f75fe82e7995224c55654594f93Andreas Gampe    if (Is64BitInstructionSet(instruction_set_)) { \
453dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers      return CreateTrampoline64(instruction_set_, abi, \
454dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers                                type ## _ENTRYPOINT_OFFSET(8, offset)); \
455dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers    } else { \
456dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers      return CreateTrampoline32(instruction_set_, abi, \
457dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers                                type ## _ENTRYPOINT_OFFSET(4, offset)); \
458dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers    }
459dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers
460848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreateInterpreterToInterpreterBridge() const {
461dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(INTERPRETER, kInterpreterAbi, pInterpreterToInterpreterBridge)
462848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers}
463848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers
464848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreateInterpreterToCompiledCodeBridge() const {
465dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(INTERPRETER, kInterpreterAbi, pInterpreterToCompiledCodeBridge)
466848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers}
467848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers
468848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreateJniDlsymLookup() const {
469dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(JNI, kJniAbi, pDlsymLookup)
470848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers}
471848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogers
47288474b416eb257078e590bf9bc7957cee604a186Jeff Haoconst std::vector<uint8_t>* CompilerDriver::CreatePortableImtConflictTrampoline() const {
473dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(PORTABLE, kPortableAbi, pPortableImtConflictTrampoline)
47488474b416eb257078e590bf9bc7957cee604a186Jeff Hao}
47588474b416eb257078e590bf9bc7957cee604a186Jeff Hao
4760aba0ba155bef7346bde19e53581200b89ae8a7aJeff Haoconst std::vector<uint8_t>* CompilerDriver::CreatePortableResolutionTrampoline() const {
477dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(PORTABLE, kPortableAbi, pPortableResolutionTrampoline)
4780aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao}
4790aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao
480848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreatePortableToInterpreterBridge() const {
481dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(PORTABLE, kPortableAbi, pPortableToInterpreterBridge)
4820aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao}
4830aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao
4842da882315a61072664f7ce3c212307342e907207Andreas Gampeconst std::vector<uint8_t>* CompilerDriver::CreateQuickGenericJniTrampoline() const {
485dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(QUICK, kQuickAbi, pQuickGenericJniTrampoline)
4862da882315a61072664f7ce3c212307342e907207Andreas Gampe}
4872da882315a61072664f7ce3c212307342e907207Andreas Gampe
48888474b416eb257078e590bf9bc7957cee604a186Jeff Haoconst std::vector<uint8_t>* CompilerDriver::CreateQuickImtConflictTrampoline() const {
489dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(QUICK, kQuickAbi, pQuickImtConflictTrampoline)
49088474b416eb257078e590bf9bc7957cee604a186Jeff Hao}
49188474b416eb257078e590bf9bc7957cee604a186Jeff Hao
492848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreateQuickResolutionTrampoline() const {
493dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(QUICK, kQuickAbi, pQuickResolutionTrampoline)
4940aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao}
4950aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao
496848871b4d8481229c32e0d048a9856e5a9a17ef9Ian Rogersconst std::vector<uint8_t>* CompilerDriver::CreateQuickToInterpreterBridge() const {
497dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers  CREATE_TRAMPOLINE(QUICK, kQuickAbi, pQuickToInterpreterBridge)
4980aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao}
499dd7624d2b9e599d57762d12031b10b89defc9807Ian Rogers#undef CREATE_TRAMPOLINE
5000aba0ba155bef7346bde19e53581200b89ae8a7aJeff Hao
5011212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::CompileAll(jobject class_loader,
5024560248d4c85cade7f4fc7b30c3fb41b95a04a7fBrian Carlstrom                                const std::vector<const DexFile*>& dex_files,
5033d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                TimingLogger* timings) {
50425c3325bf95036bf325fc7cb21b4fd6d40282857Brian Carlstrom  DCHECK(!Runtime::Current()->IsStarted());
505700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers  std::unique_ptr<ThreadPool> thread_pool(new ThreadPool("Compiler driver thread pool", thread_count_ - 1));
5063d504075f7c1204d581923460754bf6d3714b13fIan Rogers  PreCompile(class_loader, dex_files, thread_pool.get(), timings);
5073d504075f7c1204d581923460754bf6d3714b13fIan Rogers  Compile(class_loader, dex_files, thread_pool.get(), timings);
508ba0668ecd2a6459ed7c77012995ad08d27f88725Brian Carlstrom  if (dump_stats_) {
509ba0668ecd2a6459ed7c77012995ad08d27f88725Brian Carlstrom    stats_->Dump();
510ba0668ecd2a6459ed7c77012995ad08d27f88725Brian Carlstrom  }
5118a48741b96ca9cc5835cac72ac133c4ca480930fBrian Carlstrom}
5128a48741b96ca9cc5835cac72ac133c4ca480930fBrian Carlstrom
513590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartierstatic DexToDexCompilationLevel GetDexToDexCompilationlevel(
5140cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier    Thread* self, Handle<mirror::ClassLoader> class_loader, const DexFile& dex_file,
515590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier    const DexFile::ClassDef& class_def) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
5162d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  const char* descriptor = dex_file.GetClassDescriptor(class_def);
5172d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
5189837939678bb5dcba178e5fb00ed59b5d14c8d9bIan Rogers  mirror::Class* klass = class_linker->FindClass(self, descriptor, class_loader);
5192d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  if (klass == NULL) {
5202d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    CHECK(self->IsExceptionPending());
5212d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    self->ClearException();
52275021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    return kDontDexToDexCompile;
52375021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  }
52475021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  // The verifier can only run on "quick" instructions at runtime (see usage of
52575021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  // FindAccessedFieldAtDexPc and FindInvokedMethodAtDexPc in ThrowNullPointerExceptionFromDexPC
52675021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  // function). Since image classes can be verified again while compiling an application,
52775021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  // we must prevent the DEX-to-DEX compiler from introducing them.
52875021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  // TODO: find a way to enable "quick" instructions for image classes and remove this check.
529eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  bool compiling_image_classes = class_loader.Get() == nullptr;
53075021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  if (compiling_image_classes) {
53175021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    return kRequired;
53275021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  } else if (klass->IsVerified()) {
53375021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    // Class is verified so we can enable DEX-to-DEX compilation for performance.
53475021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    return kOptimize;
53575021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  } else if (klass->IsCompileTimeVerified()) {
53675021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    // Class verification has soft-failed. Anyway, ensure at least correctness.
53775021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    DCHECK_EQ(klass->GetStatus(), mirror::Class::kStatusRetryVerificationAtRuntime);
53875021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    return kRequired;
53975021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  } else {
54075021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    // Class verification has failed: do not run DEX-to-DEX compilation.
54175021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz    return kDontDexToDexCompile;
5422d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  }
5432d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz}
5442d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz
5453d504075f7c1204d581923460754bf6d3714b13fIan Rogersvoid CompilerDriver::CompileOne(mirror::ArtMethod* method, TimingLogger* timings) {
54625c3325bf95036bf325fc7cb21b4fd6d40282857Brian Carlstrom  DCHECK(!Runtime::Current()->IsStarted());
54700f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  Thread* self = Thread::Current();
5482d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  jobject jclass_loader;
54900f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  const DexFile* dex_file;
5508b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers  uint16_t class_def_idx;
551ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers  uint32_t method_idx = method->GetDexMethodIndex();
552ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers  uint32_t access_flags = method->GetAccessFlags();
553ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers  InvokeType invoke_type = method->GetInvokeType();
55400f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  {
55500f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    ScopedObjectAccessUnchecked soa(self);
55600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    ScopedLocalRef<jobject>
55700f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers      local_class_loader(soa.Env(),
55800f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers                    soa.AddLocalReference<jobject>(method->GetDeclaringClass()->GetClassLoader()));
5592d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    jclass_loader = soa.Env()->NewGlobalRef(local_class_loader.get());
56000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    // Find the dex_file
561fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers    MethodHelper mh(method);
562fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers    dex_file = &mh.GetDexFile();
563fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers    class_def_idx = mh.GetClassDefIndex();
56400f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  }
565ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers  const DexFile::CodeItem* code_item = dex_file->GetCodeItem(method->GetCodeItemOffset());
56600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  self->TransitionFromRunnableToSuspended(kNative);
567ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
568ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  std::vector<const DexFile*> dex_files;
56900f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  dex_files.push_back(dex_file);
570ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
571700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers  std::unique_ptr<ThreadPool> thread_pool(new ThreadPool("Compiler driver thread pool", 0U));
5723d504075f7c1204d581923460754bf6d3714b13fIan Rogers  PreCompile(jclass_loader, dex_files, thread_pool.get(), timings);
573ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
5742d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  // Can we run DEX-to-DEX compiler on this class ?
57575021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  DexToDexCompilationLevel dex_to_dex_compilation_level = kDontDexToDexCompile;
5762d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  {
5772d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    ScopedObjectAccess soa(Thread::Current());
5782d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    const DexFile::ClassDef& class_def = dex_file->GetClassDef(class_def_idx);
579eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<1> hs(soa.Self());
580eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ClassLoader> class_loader(
581eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(soa.Decode<mirror::ClassLoader*>(jclass_loader)));
5829837939678bb5dcba178e5fb00ed59b5d14c8d9bIan Rogers    dex_to_dex_compilation_level = GetDexToDexCompilationlevel(self, class_loader, *dex_file,
5839837939678bb5dcba178e5fb00ed59b5d14c8d9bIan Rogers                                                               class_def);
5842d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  }
585ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers  CompileMethod(code_item, access_flags, invoke_type, class_def_idx, method_idx, jclass_loader,
586ef7d42fca18c16fbaf103822ad16f23246e2905dIan Rogers                *dex_file, dex_to_dex_compilation_level);
587ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
5882d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  self->GetJniEnv()->DeleteGlobalRef(jclass_loader);
58900f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
59000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  self->TransitionFromSuspendedToRunnable();
5919ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}
5929ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
5931212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::Resolve(jobject class_loader, const std::vector<const DexFile*>& dex_files,
5943d504075f7c1204d581923460754bf6d3714b13fIan Rogers                             ThreadPool* thread_pool, TimingLogger* timings) {
595ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  for (size_t i = 0; i != dex_files.size(); ++i) {
596ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom    const DexFile* dex_file = dex_files[i];
597d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root    CHECK(dex_file != nullptr);
5982f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom    ResolveDexFile(class_loader, *dex_file, thread_pool, timings);
5999ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom  }
6009ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}
6019ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
6021212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::PreCompile(jobject class_loader, const std::vector<const DexFile*>& dex_files,
6033d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                ThreadPool* thread_pool, TimingLogger* timings) {
60496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  LoadImageClasses(timings);
60596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
6064a200f56b7075309316b04d550c9cc50f8314eddJeff Hao  if (!compiler_options_->IsVerificationEnabled()) {
6074a200f56b7075309316b04d550c9cc50f8314eddJeff Hao    VLOG(compiler) << "Verify none mode specified, skipping pre-compilation";
6084a200f56b7075309316b04d550c9cc50f8314eddJeff Hao    return;
6094a200f56b7075309316b04d550c9cc50f8314eddJeff Hao  }
6104a200f56b7075309316b04d550c9cc50f8314eddJeff Hao
6112f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom  Resolve(class_loader, dex_files, thread_pool, timings);
612601a12302407d8199503d2cc6cc0829d5996696dElliott Hughes
6132f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom  Verify(class_loader, dex_files, thread_pool, timings);
614ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
6152f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom  InitializeClasses(class_loader, dex_files, thread_pool, timings);
61696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
61796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  UpdateImageClasses(timings);
618ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom}
619ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
620dfb325e0ddd746cd8f7c2e3723b3a573eb7cc111Ian Rogersbool CompilerDriver::IsImageClass(const char* descriptor) const {
621e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  if (!IsImage()) {
62296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    return true;
623e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  } else {
624dfb325e0ddd746cd8f7c2e3723b3a573eb7cc111Ian Rogers    return image_classes_->find(descriptor) != image_classes_->end();
625ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  }
626ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom}
627ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom
62896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstromstatic void ResolveExceptionsForMethod(MethodHelper* mh,
629700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers    std::set<std::pair<uint16_t, const DexFile*>>& exceptions_to_resolve)
63096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
63196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  const DexFile::CodeItem* code_item = mh->GetCodeItem();
63296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  if (code_item == NULL) {
63396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    return;  // native or abstract method
63496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
63596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  if (code_item->tries_size_ == 0) {
63696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    return;  // nothing to process
63796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
63896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  const byte* encoded_catch_handler_list = DexFile::GetCatchHandlerData(*code_item, 0);
63996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  size_t num_encoded_catch_handlers = DecodeUnsignedLeb128(&encoded_catch_handler_list);
64096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  for (size_t i = 0; i < num_encoded_catch_handlers; i++) {
64196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    int32_t encoded_catch_handler_size = DecodeSignedLeb128(&encoded_catch_handler_list);
64296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    bool has_catch_all = false;
64396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    if (encoded_catch_handler_size <= 0) {
64496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      encoded_catch_handler_size = -encoded_catch_handler_size;
64596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      has_catch_all = true;
64696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
64796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    for (int32_t j = 0; j < encoded_catch_handler_size; j++) {
64896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      uint16_t encoded_catch_handler_handlers_type_idx =
64996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom          DecodeUnsignedLeb128(&encoded_catch_handler_list);
65096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      // Add to set of types to resolve if not already in the dex cache resolved types
65196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      if (!mh->IsResolvedTypeIdx(encoded_catch_handler_handlers_type_idx)) {
65296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom        exceptions_to_resolve.insert(
65396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom            std::pair<uint16_t, const DexFile*>(encoded_catch_handler_handlers_type_idx,
65496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom                                                &mh->GetDexFile()));
65596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      }
65696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      // ignore address associated with catch handler
65796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      DecodeUnsignedLeb128(&encoded_catch_handler_list);
65896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
65996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    if (has_catch_all) {
66096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      // ignore catch all address
66196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      DecodeUnsignedLeb128(&encoded_catch_handler_list);
66296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
66396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
66496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
66596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
66696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstromstatic bool ResolveCatchBlockExceptionsClassVisitor(mirror::Class* c, void* arg)
66796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
668700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers  std::set<std::pair<uint16_t, const DexFile*>>* exceptions_to_resolve =
669700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers      reinterpret_cast<std::set<std::pair<uint16_t, const DexFile*>>*>(arg);
67096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  MethodHelper mh;
67196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  for (size_t i = 0; i < c->NumVirtualMethods(); ++i) {
672ea46f950e7a51585db293cd7f047de190a482414Brian Carlstrom    mirror::ArtMethod* m = c->GetVirtualMethod(i);
67396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    mh.ChangeMethod(m);
67496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    ResolveExceptionsForMethod(&mh, *exceptions_to_resolve);
67596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
67696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  for (size_t i = 0; i < c->NumDirectMethods(); ++i) {
677ea46f950e7a51585db293cd7f047de190a482414Brian Carlstrom    mirror::ArtMethod* m = c->GetDirectMethod(i);
67896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    mh.ChangeMethod(m);
67996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    ResolveExceptionsForMethod(&mh, *exceptions_to_resolve);
68096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
68196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  return true;
68296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
68396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
68496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstromstatic bool RecordImageClassesVisitor(mirror::Class* klass, void* arg)
68596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
68696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  CompilerDriver::DescriptorSet* image_classes =
68796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      reinterpret_cast<CompilerDriver::DescriptorSet*>(arg);
688f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  image_classes->insert(klass->GetDescriptor());
68996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  return true;
69096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
69196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
69296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom// Make a list of descriptors for classes to include in the image
6933d504075f7c1204d581923460754bf6d3714b13fIan Rogersvoid CompilerDriver::LoadImageClasses(TimingLogger* timings)
69496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      LOCKS_EXCLUDED(Locks::mutator_lock_) {
695d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root  CHECK(timings != nullptr);
696e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  if (!IsImage()) {
69796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    return;
69896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
69996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
7003d504075f7c1204d581923460754bf6d3714b13fIan Rogers  timings->NewSplit("LoadImageClasses");
70196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // Make a first class to load all classes explicitly listed in the file
70296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  Thread* self = Thread::Current();
70396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  ScopedObjectAccess soa(self);
70496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
705d5185344e19d9feb7ac268369e0af6a467d1cb48Kenny Root  CHECK(image_classes_.get() != nullptr);
70602e25119b15a6f619f17db99f5d05124a5807ff3Mathieu Chartier  for (auto it = image_classes_->begin(), end = image_classes_->end(); it != end;) {
707e9c36b34efb7460f59c6766e526c9b0de8da70b3Vladimir Marko    const std::string& descriptor(*it);
708eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<1> hs(self);
709eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::Class> klass(
710eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(class_linker->FindSystemClass(self, descriptor.c_str())));
711eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    if (klass.Get() == NULL) {
712e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      VLOG(compiler) << "Failed to find class " << descriptor;
713e9c36b34efb7460f59c6766e526c9b0de8da70b3Vladimir Marko      image_classes_->erase(it++);
714a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers      self->ClearException();
71596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    } else {
71696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      ++it;
71796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
71896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
71996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
72096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // Resolve exception classes referenced by the loaded classes. The catch logic assumes
72196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // exceptions are resolved by the verifier when there is a catch block in an interested method.
72296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // Do this here so that exception classes appear to have been specified image classes.
723700a402244a1a423da4f3ba8032459f4b65fa18fIan Rogers  std::set<std::pair<uint16_t, const DexFile*>> unresolved_exception_types;
724eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  StackHandleScope<1> hs(self);
725eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::Class> java_lang_Throwable(
726eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(class_linker->FindSystemClass(self, "Ljava/lang/Throwable;")));
72796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  do {
72896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    unresolved_exception_types.clear();
72996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    class_linker->VisitClasses(ResolveCatchBlockExceptionsClassVisitor,
73096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom                               &unresolved_exception_types);
73102e25119b15a6f619f17db99f5d05124a5807ff3Mathieu Chartier    for (const std::pair<uint16_t, const DexFile*>& exception_type : unresolved_exception_types) {
73202e25119b15a6f619f17db99f5d05124a5807ff3Mathieu Chartier      uint16_t exception_type_idx = exception_type.first;
73302e25119b15a6f619f17db99f5d05124a5807ff3Mathieu Chartier      const DexFile* dex_file = exception_type.second;
7340cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier      StackHandleScope<2> hs(self);
735eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      Handle<mirror::DexCache> dex_cache(hs.NewHandle(class_linker->FindDexCache(*dex_file)));
736eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      Handle<mirror::Class> klass(hs.NewHandle(
7370cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier          class_linker->ResolveType(*dex_file, exception_type_idx, dex_cache,
7380cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier                                    NullHandle<mirror::ClassLoader>())));
739eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      if (klass.Get() == NULL) {
74096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom        const DexFile::TypeId& type_id = dex_file->GetTypeId(exception_type_idx);
74196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom        const char* descriptor = dex_file->GetTypeDescriptor(type_id);
74296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom        LOG(FATAL) << "Failed to resolve class " << descriptor;
74396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      }
744eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      DCHECK(java_lang_Throwable->IsAssignableFrom(klass.Get()));
74596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
74696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    // Resolving exceptions may load classes that reference more exceptions, iterate until no
74796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    // more are found
74896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  } while (!unresolved_exception_types.empty());
74996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
75096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // We walk the roots looking for classes so that we'll pick up the
75196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // above classes plus any classes them depend on such super
75296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  // classes, interfaces, and the required ClassLinker roots.
75396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  class_linker->VisitClasses(RecordImageClassesVisitor, image_classes_.get());
75496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
75596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  CHECK_NE(image_classes_->size(), 0U);
75696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
75796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
758f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartierstatic void MaybeAddToImageClasses(Handle<mirror::Class> c,
759f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier                                   CompilerDriver::DescriptorSet* image_classes)
76096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
761f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  Thread* self = Thread::Current();
762f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  StackHandleScope<1> hs(self);
763f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  // Make a copy of the handle so that we don't clobber it doing Assign.
764f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  Handle<mirror::Class> klass(hs.NewHandle(c.Get()));
76596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  while (!klass->IsObjectClass()) {
766f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier    std::string descriptor(klass->GetDescriptor());
76796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    std::pair<CompilerDriver::DescriptorSet::iterator, bool> result =
76896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom        image_classes->insert(descriptor);
76996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    if (result.second) {
77075a43f10f55e2aa550de51e969cc1e60d583b632Anwar Ghuloum        VLOG(compiler) << "Adding " << descriptor << " to image classes";
77196391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    } else {
77296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom      return;
77396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
774f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier    for (size_t i = 0; i < klass->NumDirectInterfaces(); ++i) {
775f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier      StackHandleScope<1> hs(self);
776f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier      MaybeAddToImageClasses(hs.NewHandle(mirror::Class::GetDirectInterface(self, klass, i)),
777f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier                             image_classes);
77896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
77996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    if (klass->IsArrayClass()) {
780f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier      StackHandleScope<1> hs(self);
781f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier      MaybeAddToImageClasses(hs.NewHandle(klass->GetComponentType()), image_classes);
78296391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom    }
783f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier    klass.Assign(klass->GetSuperClass());
78496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  }
78596391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
78696391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
78796391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstromvoid CompilerDriver::FindClinitImageClassesCallback(mirror::Object* object, void* arg) {
78896391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  DCHECK(object != NULL);
78996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  DCHECK(arg != NULL);
79096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom  CompilerDriver* compiler_driver = reinterpret_cast<CompilerDriver*>(arg);
791f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  StackHandleScope<1> hs(Thread::Current());
792f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier  MaybeAddToImageClasses(hs.NewHandle(object->GetClass()), compiler_driver->image_classes_.get());
79396391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
79496391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
7953d504075f7c1204d581923460754bf6d3714b13fIan Rogersvoid CompilerDriver::UpdateImageClasses(TimingLogger* timings) {
796e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  if (IsImage()) {
7973d504075f7c1204d581923460754bf6d3714b13fIan Rogers    timings->NewSplit("UpdateImageClasses");
7986f28d91aab952e3244fbb4e707fa38f85538f374Anwar Ghuloum
799e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // Update image_classes_ with classes for objects created by <clinit> methods.
800e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    Thread* self = Thread::Current();
801e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    const char* old_cause = self->StartAssertNoThreadSuspension("ImageWriter");
802e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    gc::Heap* heap = Runtime::Current()->GetHeap();
803e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // TODO: Image spaces only?
804590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier    ScopedObjectAccess soa(Thread::Current());
805e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    WriterMutexLock mu(self, *Locks::heap_bitmap_lock_);
806590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier    heap->VisitObjects(FindClinitImageClassesCallback, this);
807e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    self->EndAssertNoThreadSuspension(old_cause);
808e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  }
80996391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom}
81096391606d8adfc661e1c21703ded1e7a39377a76Brian Carlstrom
811590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartierbool CompilerDriver::CanAssumeTypeIsPresentInDexCache(const DexFile& dex_file, uint32_t type_idx) {
812fc0e94bed3f88ed7e50854fd8dfaf5dcb345250fIan Rogers  if (IsImage() &&
813dfb325e0ddd746cd8f7c2e3723b3a573eb7cc111Ian Rogers      IsImageClass(dex_file.StringDataByIdx(dex_file.GetTypeId(type_idx).descriptor_idx_))) {
8146fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers    if (kIsDebugBuild) {
8156fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers      ScopedObjectAccess soa(Thread::Current());
8166fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers      mirror::DexCache* dex_cache = Runtime::Current()->GetClassLinker()->FindDexCache(dex_file);
8176fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers      mirror::Class* resolved_class = dex_cache->GetResolvedType(type_idx);
8186fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers      CHECK(resolved_class != NULL);
8196fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers    }
820c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeInDexCache();
8216fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers    return true;
822996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  } else {
823c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNotInDexCache();
8246fe568e16cea80dfe9ccb856c138a75da5f2a90dIan Rogers    return false;
825996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
8266d4d9fcb4f01e287ee29e81cd1c941ee5d11d379Ian Rogers}
8276d4d9fcb4f01e287ee29e81cd1c941ee5d11d379Ian Rogers
8281212a022fa5f8ef9585d765b1809521812af882cIan Rogersbool CompilerDriver::CanAssumeStringIsPresentInDexCache(const DexFile& dex_file,
8291212a022fa5f8ef9585d765b1809521812af882cIan Rogers                                                        uint32_t string_idx) {
8301bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // See also Compiler::ResolveDexFile
8311bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
8325f7fa551162b9e755fdaf054ffc89411a0e135e3Ian Rogers  bool result = false;
8335f7fa551162b9e755fdaf054ffc89411a0e135e3Ian Rogers  if (IsImage()) {
8345f7fa551162b9e755fdaf054ffc89411a0e135e3Ian Rogers    // We resolve all const-string strings when building for the image.
83500f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    ScopedObjectAccess soa(Thread::Current());
836eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<1> hs(soa.Self());
837eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::DexCache> dex_cache(
838eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(Runtime::Current()->GetClassLinker()->FindDexCache(dex_file)));
8395f7fa551162b9e755fdaf054ffc89411a0e135e3Ian Rogers    Runtime::Current()->GetClassLinker()->ResolveString(dex_file, string_idx, dex_cache);
8405f7fa551162b9e755fdaf054ffc89411a0e135e3Ian Rogers    result = true;
84100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  }
842996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  if (result) {
843c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->StringInDexCache();
844996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  } else {
845c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->StringNotInDexCache();
846996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
847996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  return result;
8481bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers}
8491bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
8501212a022fa5f8ef9585d765b1809521812af882cIan Rogersbool CompilerDriver::CanAccessTypeWithoutChecks(uint32_t referrer_idx, const DexFile& dex_file,
851c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers                                                uint32_t type_idx,
852c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers                                                bool* type_known_final, bool* type_known_abstract,
853c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers                                                bool* equals_referrers_class) {
854c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  if (type_known_final != NULL) {
855c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    *type_known_final = false;
856c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  }
857c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  if (type_known_abstract != NULL) {
858c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    *type_known_abstract = false;
859c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  }
860c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  if (equals_referrers_class != NULL) {
861c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    *equals_referrers_class = false;
862c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  }
86300f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
8642dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::DexCache* dex_cache = Runtime::Current()->GetClassLinker()->FindDexCache(dex_file);
8651bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // Get type from dex cache assuming it was populated by the verifier
8662dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::Class* resolved_class = dex_cache->GetResolvedType(type_idx);
8671bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  if (resolved_class == NULL) {
868c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
8691bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers    return false;  // Unknown class needs access checks.
8701bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
8711bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  const DexFile::MethodId& method_id = dex_file.GetMethodId(referrer_idx);
872c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  if (equals_referrers_class != NULL) {
873c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    *equals_referrers_class = (method_id.class_idx_ == type_idx);
874c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers  }
8752dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::Class* referrer_class = dex_cache->GetResolvedType(method_id.class_idx_);
8761bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  if (referrer_class == NULL) {
877c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
8781bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers    return false;  // Incomplete referrer knowledge needs access check.
8791bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
8801bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // Perform access check, will return true if access is ok or false if we're going to have to
8811bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // check this at runtime (for example for class loaders).
882996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  bool result = referrer_class->CanAccess(resolved_class);
883996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  if (result) {
884c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeDoesntNeedAccessCheck();
885c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    if (type_known_final != NULL) {
886c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers      *type_known_final = resolved_class->IsFinal() && !resolved_class->IsArrayClass();
887c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    }
888c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    if (type_known_abstract != NULL) {
889b19fa800aab1c2731b12ff24696feca13901db3fIan Rogers      *type_known_abstract = resolved_class->IsAbstract() && !resolved_class->IsArrayClass();
890c9e463c8aa083a5ed20293f42363ebff93de5f84Ian Rogers    }
891996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  } else {
892c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
893996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
894996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  return result;
8951bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers}
8961bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
8971212a022fa5f8ef9585d765b1809521812af882cIan Rogersbool CompilerDriver::CanAccessInstantiableTypeWithoutChecks(uint32_t referrer_idx,
8981212a022fa5f8ef9585d765b1809521812af882cIan Rogers                                                            const DexFile& dex_file,
8991212a022fa5f8ef9585d765b1809521812af882cIan Rogers                                                            uint32_t type_idx) {
90000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
9012dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::DexCache* dex_cache = Runtime::Current()->GetClassLinker()->FindDexCache(dex_file);
9021bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // Get type from dex cache assuming it was populated by the verifier.
9032dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::Class* resolved_class = dex_cache->GetResolvedType(type_idx);
9041bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  if (resolved_class == NULL) {
905c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
9061bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers    return false;  // Unknown class needs access checks.
9071bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
9081bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  const DexFile::MethodId& method_id = dex_file.GetMethodId(referrer_idx);
9092dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::Class* referrer_class = dex_cache->GetResolvedType(method_id.class_idx_);
9101bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  if (referrer_class == NULL) {
911c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
9121bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers    return false;  // Incomplete referrer knowledge needs access check.
9131bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
9141bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // Perform access and instantiable checks, will return true if access is ok or false if we're
9151bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  // going to have to check this at runtime (for example for class loaders).
916996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  bool result = referrer_class->CanAccess(resolved_class) && resolved_class->IsInstantiable();
917996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  if (result) {
918c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeDoesntNeedAccessCheck();
919996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  } else {
920c8b306f5221658c7e4b5516be8917dc8c9288e7eIan Rogers    stats_->TypeNeedsAccessCheck();
921996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  }
922996cc586ce76235ee3efa1eff86de51dabcc5d53Ian Rogers  return result;
9231bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers}
9241bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
925be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchibool CompilerDriver::CanEmbedTypeInCode(const DexFile& dex_file, uint32_t type_idx,
926be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                        bool* is_type_initialized, bool* use_direct_type_ptr,
9278668c3cbdcf9471bd97e0da68a240051f2973074Mathieu Chartier                                        uintptr_t* direct_type_ptr, bool* out_is_finalizable) {
928be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  ScopedObjectAccess soa(Thread::Current());
929be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  mirror::DexCache* dex_cache = Runtime::Current()->GetClassLinker()->FindDexCache(dex_file);
930be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  mirror::Class* resolved_class = dex_cache->GetResolvedType(type_idx);
931be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  if (resolved_class == nullptr) {
932be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    return false;
933be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  }
9348668c3cbdcf9471bd97e0da68a240051f2973074Mathieu Chartier  *out_is_finalizable = resolved_class->IsFinalizable();
935be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  const bool compiling_boot = Runtime::Current()->GetHeap()->IsCompilingBoot();
936be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  if (compiling_boot) {
937be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    // boot -> boot class pointers.
938be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    // True if the class is in the image at boot compiling time.
939be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    const bool is_image_class = IsImage() && IsImageClass(
940be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi        dex_file.StringDataByIdx(dex_file.GetTypeId(type_idx).descriptor_idx_));
941be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    // True if pc relative load works.
942be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    const bool support_boot_image_fixup = GetSupportBootImageFixup();
943be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    if (is_image_class && support_boot_image_fixup) {
944be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *is_type_initialized = resolved_class->IsInitialized();
945be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *use_direct_type_ptr = false;
946be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *direct_type_ptr = 0;
947be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      return true;
948be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    } else {
949be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      return false;
950be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    }
951be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  } else {
952be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    // True if the class is in the image at app compiling time.
953be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    const bool class_in_image =
954be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi        Runtime::Current()->GetHeap()->FindSpaceFromObject(resolved_class, false)->IsImageSpace();
955be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    if (class_in_image) {
956be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      // boot -> app class pointers.
957be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *is_type_initialized = resolved_class->IsInitialized();
958be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *use_direct_type_ptr = true;
959be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      *direct_type_ptr = reinterpret_cast<uintptr_t>(resolved_class);
960be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      return true;
961be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    } else {
962be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      // app -> app class pointers.
963be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      // Give up because app does not have an image and class
964be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      // isn't created at compile time.  TODO: implement this
965be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      // if/when each app gets an image.
966be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi      return false;
967be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi    }
968be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  }
969be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi}
970be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi
971be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Markovoid CompilerDriver::ProcessedInstanceField(bool resolved) {
972be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (!resolved) {
973be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    stats_->UnresolvedInstanceField();
974be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  } else {
975be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    stats_->ResolvedInstanceField();
976be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
977be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko}
978be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko
979be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Markovoid CompilerDriver::ProcessedStaticField(bool resolved, bool local) {
980be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (!resolved) {
981be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    stats_->UnresolvedStaticField();
982be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  } else if (local) {
983be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    stats_->ResolvedLocalStaticField();
984be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  } else {
985be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    stats_->ResolvedStaticField();
986be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
987be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko}
988be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko
989f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Markovoid CompilerDriver::ProcessedInvoke(InvokeType invoke_type, int flags) {
990f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  stats_->ProcessedInvoke(invoke_type, flags);
991a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers}
992a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers
99389756f21c107d96e6d1bbc75811bd33078c8ceeeIan Rogersbool CompilerDriver::ComputeInstanceFieldInfo(uint32_t field_idx, const DexCompilationUnit* mUnit,
994be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko                                              bool is_put, MemberOffset* field_offset,
995be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko                                              bool* is_volatile) {
99600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
997be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  // Try to resolve the field and compiling method's class.
998be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::ArtField* resolved_field;
999be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::Class* referrer_class;
1000be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::DexCache* dex_cache;
1001be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  {
1002eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<3> hs(soa.Self());
1003eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::DexCache> dex_cache_handle(
1004eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(mUnit->GetClassLinker()->FindDexCache(*mUnit->GetDexFile())));
1005eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ClassLoader> class_loader_handle(
1006eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(soa.Decode<mirror::ClassLoader*>(mUnit->GetClassLoader())));
1007eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ArtField> resolved_field_handle(hs.NewHandle(
1008eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        ResolveField(soa, dex_cache_handle, class_loader_handle, mUnit, field_idx, false)));
1009eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    referrer_class = (resolved_field_handle.Get() != nullptr)
1010eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        ? ResolveCompilingMethodsClass(soa, dex_cache_handle, class_loader_handle, mUnit) : nullptr;
1011eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    resolved_field = resolved_field_handle.Get();
1012eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    dex_cache = dex_cache_handle.Get();
10131bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
1014be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  bool result = false;
1015be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (resolved_field != nullptr && referrer_class != nullptr) {
1016be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_volatile = IsFieldVolatile(resolved_field);
1017be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    std::pair<bool, bool> fast_path = IsFastInstanceField(
1018be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko        dex_cache, referrer_class, resolved_field, field_idx, field_offset);
1019be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    result = is_put ? fast_path.second : fast_path.first;
1020be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
1021be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (!result) {
1022be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    // Conservative defaults.
1023be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_volatile = true;
1024be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *field_offset = MemberOffset(static_cast<size_t>(-1));
1025be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
1026be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  ProcessedInstanceField(result);
1027be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  return result;
10281bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers}
10291bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
103089756f21c107d96e6d1bbc75811bd33078c8ceeeIan Rogersbool CompilerDriver::ComputeStaticFieldInfo(uint32_t field_idx, const DexCompilationUnit* mUnit,
1031be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko                                            bool is_put, MemberOffset* field_offset,
1032be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko                                            uint32_t* storage_index, bool* is_referrers_class,
1033be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko                                            bool* is_volatile, bool* is_initialized) {
103400f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
1035be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  // Try to resolve the field and compiling method's class.
1036be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::ArtField* resolved_field;
1037be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::Class* referrer_class;
1038be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  mirror::DexCache* dex_cache;
1039be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  {
1040eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<3> hs(soa.Self());
1041eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::DexCache> dex_cache_handle(
1042eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(mUnit->GetClassLinker()->FindDexCache(*mUnit->GetDexFile())));
1043eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ClassLoader> class_loader_handle(
1044eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(soa.Decode<mirror::ClassLoader*>(mUnit->GetClassLoader())));
1045eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ArtField> resolved_field_handle(hs.NewHandle(
1046eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        ResolveField(soa, dex_cache_handle, class_loader_handle, mUnit, field_idx, true)));
1047eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    referrer_class = (resolved_field_handle.Get() != nullptr)
1048eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        ? ResolveCompilingMethodsClass(soa, dex_cache_handle, class_loader_handle, mUnit) : nullptr;
1049eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    resolved_field = resolved_field_handle.Get();
1050eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    dex_cache = dex_cache_handle.Get();
10511bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers  }
1052be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  bool result = false;
1053be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (resolved_field != nullptr && referrer_class != nullptr) {
1054be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_volatile = IsFieldVolatile(resolved_field);
1055be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    std::pair<bool, bool> fast_path = IsFastStaticField(
1056be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko        dex_cache, referrer_class, resolved_field, field_idx, field_offset,
1057be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko        storage_index, is_referrers_class, is_initialized);
1058be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    result = is_put ? fast_path.second : fast_path.first;
1059be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
1060be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  if (!result) {
1061be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    // Conservative defaults.
1062be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_volatile = true;
1063be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *field_offset = MemberOffset(static_cast<size_t>(-1));
1064be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *storage_index = -1;
1065be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_referrers_class = false;
1066be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko    *is_initialized = false;
1067be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  }
1068be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  ProcessedStaticField(result, *is_referrers_class);
1069be0e546730e532ef0987cd4bde2c6f5a1b14dd2aVladimir Marko  return result;
10701bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers}
10711bddec3a6521f16df37499754000a3b1787a52e9Ian Rogers
107283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogersvoid CompilerDriver::GetCodeAndMethodForDirectCall(InvokeType* type, InvokeType sharp_type,
107383883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers                                                   bool no_guarantee_of_dex_cache_entry,
10744f6ad8ab428038129b2d0d6c40b7fd625cca15e1Ian Rogers                                                   mirror::Class* referrer_class,
1075ea46f950e7a51585db293cd7f047de190a482414Brian Carlstrom                                                   mirror::ArtMethod* method,
1076f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko                                                   int* stats_flags,
107783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers                                                   MethodReference* target_method,
107865ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                                   uintptr_t* direct_code,
107965ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                                   uintptr_t* direct_method) {
1080137e88f798857321f4007631fdf052d2830ec2c4Ian Rogers  // For direct and static methods compute possible direct_code and direct_method values, ie
1081137e88f798857321f4007631fdf052d2830ec2c4Ian Rogers  // an address for the Method* being invoked and an address of the code for that Method*.
1082137e88f798857321f4007631fdf052d2830ec2c4Ian Rogers  // For interface calls compute a value for direct_method that is the interface method being
1083137e88f798857321f4007631fdf052d2830ec2c4Ian Rogers  // invoked, so this can be passed to the out-of-line runtime support code.
108465ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers  *direct_code = 0;
108565ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers  *direct_method = 0;
108683883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  bool use_dex_cache = false;
1087590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier  const bool compiling_boot = Runtime::Current()->GetHeap()->IsCompilingBoot();
1088b34f69ab43aaf7a6e6045c95f398baf566ef5023Nicolas Geoffray  if (compiler_->IsPortable()) {
1089c531cefbfb5394413122e9f57d211ba436cff012buzbee    if (sharp_type != kStatic && sharp_type != kDirect) {
1090c531cefbfb5394413122e9f57d211ba436cff012buzbee      return;
1091c531cefbfb5394413122e9f57d211ba436cff012buzbee    }
109283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    use_dex_cache = true;
1093c531cefbfb5394413122e9f57d211ba436cff012buzbee  } else {
109488474b416eb257078e590bf9bc7957cee604a186Jeff Hao    if (sharp_type != kStatic && sharp_type != kDirect) {
1095c531cefbfb5394413122e9f57d211ba436cff012buzbee      return;
1096c531cefbfb5394413122e9f57d211ba436cff012buzbee    }
109783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    // TODO: support patching on all architectures.
109883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    use_dex_cache = compiling_boot && !support_boot_image_fixup_;
1099b8404a7de94c109e3c17b4205b5f8aaae996eb33TDYa  }
110083883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  bool method_code_in_boot = (method->GetDeclaringClass()->GetClassLoader() == nullptr);
110183883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  if (!use_dex_cache) {
110283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    if (!method_code_in_boot) {
110383883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      use_dex_cache = true;
110483883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    } else {
110583883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      bool has_clinit_trampoline =
110683883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers          method->IsStatic() && !method->GetDeclaringClass()->IsInitialized();
110783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      if (has_clinit_trampoline && (method->GetDeclaringClass() != referrer_class)) {
110883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers        // Ensure we run the clinit trampoline unless we are invoking a static method in the same
110983883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers        // class.
111083883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers        use_dex_cache = true;
111183883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      }
111283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    }
11132ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  }
1114f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  if (method_code_in_boot) {
1115f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    *stats_flags |= kFlagDirectCallToBoot | kFlagDirectMethodToBoot;
11162ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  }
111783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  if (!use_dex_cache && compiling_boot) {
111883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    MethodHelper mh(method);
1119dfb325e0ddd746cd8f7c2e3723b3a573eb7cc111Ian Rogers    if (!IsImageClass(mh.GetDeclaringClassDescriptor())) {
112083883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      // We can only branch directly to Methods that are resolved in the DexCache.
112183883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      // Otherwise we won't invoke the resolution trampoline.
112283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      use_dex_cache = true;
1123e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers    }
1124c468e92d1e18305a5053c8a4a7c88cb297e525afIan Rogers  }
112583883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  // The method is defined not within this dex file. We need a dex cache slot within the current
112683883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  // dex file or direct pointers.
112783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  bool must_use_direct_pointers = false;
112883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  if (target_method->dex_file == method->GetDeclaringClass()->GetDexCache()->GetDexFile()) {
112983883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    target_method->dex_method_index = method->GetDexMethodIndex();
113083883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  } else {
113183883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    if (no_guarantee_of_dex_cache_entry) {
113283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      // See if the method is also declared in this dex cache.
113383883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      uint32_t dex_method_idx = MethodHelper(method).FindDexMethodIndexInOtherDexFile(
1134bbcc0c0a17262f3d2a70fc0a82e1783862f708ccVladimir Marko          *target_method->dex_file, target_method->dex_method_index);
113583883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      if (dex_method_idx != DexFile::kDexNoIndex) {
113683883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers        target_method->dex_method_index = dex_method_idx;
113783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      } else {
1138e0099d4d4839cabfb26270a746580e9d4ec3a6c6Andreas Gampe        if (compiling_boot && !use_dex_cache) {
113949161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao          target_method->dex_method_index = method->GetDexMethodIndex();
114049161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao          target_method->dex_file = method->GetDeclaringClass()->GetDexCache()->GetDexFile();
114149161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao        }
114283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers        must_use_direct_pointers = true;
11433fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers      }
11443fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers    }
114583883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  }
114683883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers  if (use_dex_cache) {
114783883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    if (must_use_direct_pointers) {
114883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      // Fail. Test above showed the only safe dispatch was via the dex cache, however, the direct
114983883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      // pointers are required as the dex cache lacks an appropriate entry.
115083883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      VLOG(compiler) << "Dex cache devirtualization failed for: " << PrettyMethod(method);
115183883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    } else {
115283883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      *type = sharp_type;
115383883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    }
11543fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers  } else {
1155a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko    bool method_in_image = compiling_boot ||
1156a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko        Runtime::Current()->GetHeap()->FindSpaceFromObject(method, false)->IsImageSpace();
1157a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko    if (method_in_image) {
1158a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      CHECK(!method->IsAbstract());
115983883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers      *type = sharp_type;
1160a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      *direct_method = compiling_boot ? -1 : reinterpret_cast<uintptr_t>(method);
1161a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      *direct_code = compiling_boot ? -1 : compiler_->GetEntryPointOf(method);
1162a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      target_method->dex_file = method->GetDeclaringClass()->GetDexCache()->GetDexFile();
1163a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      target_method->dex_method_index = method->GetDexMethodIndex();
1164a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko    } else if (!must_use_direct_pointers) {
1165a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      // Set the code and rely on the dex cache for the method.
1166a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      *type = sharp_type;
1167a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      *direct_code = compiler_->GetEntryPointOf(method);
116883883d7fddf30fdb8b6903560fa1337ab991e74cIan Rogers    } else {
1169a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      // Direct pointers were required but none were available.
1170a51a0b0300268b605e3ad71b0e87ff394032c5e7Vladimir Marko      VLOG(compiler) << "Dex cache devirtualization failed for: " << PrettyMethod(method);
11713fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers    }
11722ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers  }
11732ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers}
11742ed3b9536ccfd7c7321cc18650820b093b22d6c9Ian Rogers
1175e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogersbool CompilerDriver::ComputeInvokeInfo(const DexCompilationUnit* mUnit, const uint32_t dex_pc,
117665ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                       bool update_stats, bool enable_devirtualization,
117765ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                       InvokeType* invoke_type, MethodReference* target_method,
117865ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                       int* vtable_idx, uintptr_t* direct_code,
117965ec92cf13c9d11c83711443a02e4249163d47f1Ian Rogers                                       uintptr_t* direct_method) {
1180f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  InvokeType orig_invoke_type = *invoke_type;
1181f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  int stats_flags = 0;
118200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
1183f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  // Try to resolve the method and compiling method's class.
1184f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  mirror::ArtMethod* resolved_method;
1185f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  mirror::Class* referrer_class;
1186eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  StackHandleScope<3> hs(soa.Self());
1187eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::DexCache> dex_cache(
1188eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(mUnit->GetClassLinker()->FindDexCache(*mUnit->GetDexFile())));
1189eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::ClassLoader> class_loader(hs.NewHandle(
1190eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      soa.Decode<mirror::ClassLoader*>(mUnit->GetClassLoader())));
1191f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  {
1192f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    uint32_t method_idx = target_method->dex_method_index;
1193eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ArtMethod> resolved_method_handle(hs.NewHandle(
1194eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        ResolveMethod(soa, dex_cache, class_loader, mUnit, method_idx, orig_invoke_type)));
1195eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    referrer_class = (resolved_method_handle.Get() != nullptr)
1196f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        ? ResolveCompilingMethodsClass(soa, dex_cache, class_loader, mUnit) : nullptr;
1197eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    resolved_method = resolved_method_handle.Get();
1198f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  }
1199f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  bool result = false;
1200f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  if (resolved_method != nullptr) {
1201f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    *vtable_idx = GetResolvedMethodVTableIndex(resolved_method, orig_invoke_type);
1202f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko
1203f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    if (enable_devirtualization) {
1204f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      DCHECK(mUnit->GetVerifiedMethod() != nullptr);
1205f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      const MethodReference* devirt_target = mUnit->GetVerifiedMethod()->GetDevirtTarget(dex_pc);
1206f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko
1207f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      stats_flags = IsFastInvoke(
1208f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko          soa, dex_cache, class_loader, mUnit, referrer_class, resolved_method,
1209f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko          invoke_type, target_method, devirt_target, direct_code, direct_method);
1210f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      result = stats_flags != 0;
1211f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    } else {
1212f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      // Devirtualization not enabled. Inline IsFastInvoke(), dropping the devirtualization parts.
1213f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      if (UNLIKELY(referrer_class == nullptr) ||
1214f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko          UNLIKELY(!referrer_class->CanAccessResolvedMethod(resolved_method->GetDeclaringClass(),
1215eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier                                                            resolved_method, dex_cache.Get(),
1216f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko                                                            target_method->dex_method_index)) ||
1217f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko          *invoke_type == kSuper) {
1218f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        // Slow path. (Without devirtualization, all super calls go slow path as well.)
1219f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko      } else {
1220f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        // Sharpening failed so generate a regular resolved method dispatch.
1221f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        stats_flags = kFlagMethodResolved;
1222f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        GetCodeAndMethodForDirectCall(invoke_type, *invoke_type, false, referrer_class, resolved_method,
1223f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko                                      &stats_flags, target_method, direct_code, direct_method);
1224f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko        result = true;
1225a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers      }
1226a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers    }
1227a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers  }
1228f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  if (!result) {
1229f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    // Conservative defaults.
1230f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    *vtable_idx = -1;
1231f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    *direct_code = 0u;
1232f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    *direct_method = 0u;
1233a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers  }
1234e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  if (update_stats) {
1235f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko    ProcessedInvoke(orig_invoke_type, stats_flags);
1236e3cd2f0e3c3d976ae9c65c8a731003a5aaf71986Ian Rogers  }
1237f096aad9203d7c50b2f9cbe1c1215a50c265a059Vladimir Marko  return result;
1238a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers}
1239a32a6fd4a781262dff4fec102da053d16b7ef6c0Ian Rogers
12402730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Markoconst VerifiedMethod* CompilerDriver::GetVerifiedMethod(const DexFile* dex_file,
12412730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko                                                        uint32_t method_idx) const {
12422730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko  MethodReference ref(dex_file, method_idx);
12432730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko  return verification_results_->GetVerifiedMethod(ref);
12442730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko}
12452730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko
12462730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Markobool CompilerDriver::IsSafeCast(const DexCompilationUnit* mUnit, uint32_t dex_pc) {
12472730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko  DCHECK(mUnit->GetVerifiedMethod() != nullptr);
12482730db03beee4d6687ddfb5000c33c0370fbc6ebVladimir Marko  bool result = mUnit->GetVerifiedMethod()->IsSafeCast(dex_pc);
1249fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  if (result) {
1250fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    stats_->SafeCast();
1251fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  } else {
1252fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers    stats_->NotASafeCast();
1253fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  }
1254fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers  return result;
1255fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers}
1256fae370a044f5817f69937cccfd2d12a16b374266Ian Rogers
12571212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::AddCodePatch(const DexFile* dex_file,
12588b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  uint16_t referrer_class_def_idx,
12598b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  uint32_t referrer_method_idx,
12608b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  InvokeType referrer_invoke_type,
12618b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  uint32_t target_method_idx,
126249161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                  const DexFile* target_dex_file,
12638b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  InvokeType target_invoke_type,
12648b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                  size_t literal_offset) {
126550b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  MutexLock mu(Thread::Current(), compiled_methods_lock_);
1266be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  code_to_patch_.push_back(new CallPatchInformation(dex_file,
1267be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    referrer_class_def_idx,
1268be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    referrer_method_idx,
1269be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    referrer_invoke_type,
1270be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    target_method_idx,
127149161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                                    target_dex_file,
1272be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    target_invoke_type,
1273be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                    literal_offset));
12743fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers}
127555d0eac918321e0525f6e6491f36a80977e0d416Mark Mendellvoid CompilerDriver::AddRelativeCodePatch(const DexFile* dex_file,
127655d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          uint16_t referrer_class_def_idx,
127755d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          uint32_t referrer_method_idx,
127855d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          InvokeType referrer_invoke_type,
127955d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          uint32_t target_method_idx,
128049161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                          const DexFile* target_dex_file,
128155d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          InvokeType target_invoke_type,
128255d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          size_t literal_offset,
128355d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                          int32_t pc_relative_offset) {
128455d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell  MutexLock mu(Thread::Current(), compiled_methods_lock_);
128555d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell  code_to_patch_.push_back(new RelativeCallPatchInformation(dex_file,
128655d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            referrer_class_def_idx,
128755d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            referrer_method_idx,
128855d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            referrer_invoke_type,
128955d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            target_method_idx,
129049161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                                            target_dex_file,
129155d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            target_invoke_type,
129255d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            literal_offset,
129355d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell                                                            pc_relative_offset));
129455d0eac918321e0525f6e6491f36a80977e0d416Mark Mendell}
12951212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::AddMethodPatch(const DexFile* dex_file,
12968b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    uint16_t referrer_class_def_idx,
12978b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    uint32_t referrer_method_idx,
12988b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    InvokeType referrer_invoke_type,
12998b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    uint32_t target_method_idx,
130049161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                    const DexFile* target_dex_file,
13018b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    InvokeType target_invoke_type,
13028b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                    size_t literal_offset) {
130350b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  MutexLock mu(Thread::Current(), compiled_methods_lock_);
1304be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  methods_to_patch_.push_back(new CallPatchInformation(dex_file,
1305be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       referrer_class_def_idx,
1306be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       referrer_method_idx,
1307be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       referrer_invoke_type,
1308be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       target_method_idx,
130949161cef10a308aedada18e9aa742498d6e6c8c7Jeff Hao                                                       target_dex_file,
1310be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       target_invoke_type,
1311be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       literal_offset));
1312be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi}
1313be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchivoid CompilerDriver::AddClassPatch(const DexFile* dex_file,
1314be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                    uint16_t referrer_class_def_idx,
1315be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                    uint32_t referrer_method_idx,
1316be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                    uint32_t target_type_idx,
1317be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                    size_t literal_offset) {
1318be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  MutexLock mu(Thread::Current(), compiled_methods_lock_);
1319be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi  classes_to_patch_.push_back(new TypePatchInformation(dex_file,
1320be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       referrer_class_def_idx,
1321be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       referrer_method_idx,
1322be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       target_type_idx,
1323be1ca55db3362f5b100c4c65da5342fd299520bbHiroshi Yamauchi                                                       literal_offset));
13243fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers}
13253fa13791c51985d9956d01bc465de6d36c3390d3Ian Rogers
1326219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogersclass ParallelCompilationManager {
1327731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom public:
1328219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  typedef void Callback(const ParallelCompilationManager* manager, size_t index);
13290e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier
1330219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ParallelCompilationManager(ClassLinker* class_linker,
1331219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers                             jobject class_loader,
1332219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers                             CompilerDriver* compiler,
1333219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers                             const DexFile* dex_file,
13343d504075f7c1204d581923460754bf6d3714b13fIan Rogers                             ThreadPool* thread_pool)
13350b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier    : index_(0),
13360b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier      class_linker_(class_linker),
1337731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom      class_loader_(class_loader),
1338731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom      compiler_(compiler),
13390e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier      dex_file_(dex_file),
13403d504075f7c1204d581923460754bf6d3714b13fIan Rogers      thread_pool_(thread_pool) {}
1341731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom
134200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ClassLinker* GetClassLinker() const {
1343731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    CHECK(class_linker_ != NULL);
1344731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    return class_linker_;
1345731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom  }
134600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
134700f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  jobject GetClassLoader() const {
1348731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    return class_loader_;
1349731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom  }
135000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
13511212a022fa5f8ef9585d765b1809521812af882cIan Rogers  CompilerDriver* GetCompiler() const {
1352731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    CHECK(compiler_ != NULL);
1353731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    return compiler_;
1354731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom  }
135500f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
135600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  const DexFile* GetDexFile() const {
1357731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    CHECK(dex_file_ != NULL);
1358731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom    return dex_file_;
1359731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom  }
1360731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom
13610e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  void ForAll(size_t begin, size_t end, Callback callback, size_t work_units) {
13620e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    Thread* self = Thread::Current();
13630e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    self->AssertNoPendingException();
13640e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    CHECK_GT(work_units, 0U);
13659ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
13663e5cf305db800b2989ad57b7cde8fb3cc9fa1b9eIan Rogers    index_.StoreRelaxed(begin);
13670e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    for (size_t i = 0; i < work_units; ++i) {
1368501baec5f2de2156bb5ed7f66d23f1b1ad026267Sebastien Hertz      thread_pool_->AddTask(self, new ForAllClosure(this, end, callback));
13690e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    }
13700e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    thread_pool_->StartWorkers(self);
137100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
13720e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    // Ensure we're suspended while we're blocked waiting for the other threads to finish (worker
13730e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    // thread destructor's called below perform join).
13740e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    CHECK_NE(self->GetState(), kRunnable);
1375d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
13760e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    // Wait for all the worker threads to finish.
13771d54e73444e017d3a65234e0f193846f3e27472bIan Rogers    thread_pool_->Wait(self, true, false);
1378d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  }
1379d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
13800b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier  size_t NextIndex() {
13813e5cf305db800b2989ad57b7cde8fb3cc9fa1b9eIan Rogers    return index_.FetchAndAddSequentiallyConsistent(1);
13820b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier  }
13830b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier
1384d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes private:
138502b6a78038f12c109f95eb31713cfc747f5512f1Mathieu Chartier  class ForAllClosure : public Task {
13860e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier   public:
13870b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier    ForAllClosure(ParallelCompilationManager* manager, size_t end, Callback* callback)
1388219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers        : manager_(manager),
13890e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier          end_(end),
13900b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier          callback_(callback) {}
13919ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
13920e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    virtual void Run(Thread* self) {
13930b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier      while (true) {
13940b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier        const size_t index = manager_->NextIndex();
13950b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier        if (UNLIKELY(index >= end_)) {
13960b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier          break;
13970b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier        }
13980b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier        callback_(manager_, index);
13990e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier        self->AssertNoPendingException();
14000e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier      }
14010e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    }
140202b6a78038f12c109f95eb31713cfc747f5512f1Mathieu Chartier
140302b6a78038f12c109f95eb31713cfc747f5512f1Mathieu Chartier    virtual void Finalize() {
140402b6a78038f12c109f95eb31713cfc747f5512f1Mathieu Chartier      delete this;
140502b6a78038f12c109f95eb31713cfc747f5512f1Mathieu Chartier    }
14060cd7ec2dcd8d7ba30bf3ca420b40dac52849876cBrian Carlstrom
14070e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier   private:
14080b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier    ParallelCompilationManager* const manager_;
14090e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier    const size_t end_;
1410460536209b741bc469f1b0857775449abb2102fbBernhard Rosenkränzer    Callback* const callback_;
14110e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  };
14121e4092589f1400915e6213014da103aab8728ef6Elliott Hughes
14130b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier  AtomicInteger index_;
14140e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  ClassLinker* const class_linker_;
14150e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  const jobject class_loader_;
14161212a022fa5f8ef9585d765b1809521812af882cIan Rogers  CompilerDriver* const compiler_;
14170e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  const DexFile* const dex_file_;
1418219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ThreadPool* const thread_pool_;
14190b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier
14200b3eb39f8bc92f955f875ef50929c439aafe35fbMathieu Chartier  DISALLOW_COPY_AND_ASSIGN(ParallelCompilationManager);
1421d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes};
1422d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
14230e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// Return true if the class should be skipped during compilation.
14240e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao//
14250e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// The first case where we skip is for redundant class definitions in
14260e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// the boot classpath. We skip all but the first definition in that case.
14270e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao//
14280e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// The second case where we skip is when an app bundles classes found
14290e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// in the boot classpath. Since at runtime we will select the class from
14300e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// the boot classpath, we ignore the one from the app.
1431be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogersstatic bool SkipClass(ClassLinker* class_linker, jobject class_loader, const DexFile& dex_file,
1432be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                      const DexFile::ClassDef& class_def) {
14330e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const char* descriptor = dex_file.GetClassDescriptor(class_def);
143400f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  if (class_loader == NULL) {
14350e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    DexFile::ClassPathEntry pair = DexFile::FindInClassPath(descriptor, class_linker->GetBootClassPath());
14360e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    CHECK(pair.second != NULL);
14370e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    if (pair.first != &dex_file) {
14380e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao      LOG(WARNING) << "Skipping class " << descriptor << " from " << dex_file.GetLocation()
14390e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao                   << " previously found in " << pair.first->GetLocation();
14400e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao      return true;
14410e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    }
144200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    return false;
144300f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  }
1444be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  return class_linker->IsInBootClassPath(descriptor);
144500f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers}
144600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
14470e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// A fast version of SkipClass above if the class pointer is available
14480e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao// that avoids the expensive FindInClassPath search.
14490e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Haostatic bool SkipClass(jobject class_loader, const DexFile& dex_file, mirror::Class* klass)
14500e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
14510e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  DCHECK(klass != NULL);
14520e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const DexFile& original_dex_file = *klass->GetDexCache()->GetDexFile();
14530e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  if (&dex_file != &original_dex_file) {
14540e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    if (class_loader == NULL) {
14550e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao      LOG(WARNING) << "Skipping class " << PrettyDescriptor(klass) << " from "
14560e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao                   << dex_file.GetLocation() << " previously found in "
14570e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao                   << original_dex_file.GetLocation();
14580e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    }
14590e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    return true;
14600e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  }
14610e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  return false;
14620e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao}
14630e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao
1464e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogersstatic void ResolveClassFieldsAndMethods(const ParallelCompilationManager* manager,
1465e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers                                         size_t class_def_index)
1466b726dcb581bf72da46527378ccb6889020f0e6e9Ian Rogers    LOCKS_EXCLUDED(Locks::mutator_lock_) {
14678f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  ATRACE_CALL();
1468be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  Thread* self = Thread::Current();
1469be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  jobject jclass_loader = manager->GetClassLoader();
1470219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  const DexFile& dex_file = *manager->GetDexFile();
1471be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  ClassLinker* class_linker = manager->GetClassLinker();
1472d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
1473e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  // If an instance field is final then we need to have a barrier on the return, static final
1474e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  // fields are assigned within the lock held for class initialization. Conservatively assume
1475e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  // constructor barriers are always required.
1476e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  bool requires_constructor_barrier = true;
1477e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers
1478845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // Method and Field are the worst. We can't resolve without either
1479845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // context from the code use (to disambiguate virtual vs direct
1480845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // method and instance vs static field) or from class
1481845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // definitions. While the compiler will resolve what it can as it
1482845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // needs it, here we try to resolve fields and methods used in class
1483845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // definitions, since many of them many never be referenced by
1484845490bda68f7d025ea7f45775c847d2932e00dcBrian Carlstrom  // generated code.
1485d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  const DexFile::ClassDef& class_def = dex_file.GetClassDef(class_def_index);
1486be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  if (!SkipClass(class_linker, jclass_loader, dex_file, class_def)) {
1487cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    ScopedObjectAccess soa(self);
1488eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<2> hs(soa.Self());
1489eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ClassLoader> class_loader(
1490eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(soa.Decode<mirror::ClassLoader*>(jclass_loader)));
1491eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::DexCache> dex_cache(hs.NewHandle(class_linker->FindDexCache(dex_file)));
1492cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    // Resolve the class.
1493cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    mirror::Class* klass = class_linker->ResolveType(dex_file, class_def.class_idx_, dex_cache,
1494cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom                                                     class_loader);
1495cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    bool resolve_fields_and_methods;
1496cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    if (klass == NULL) {
1497cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      // Class couldn't be resolved, for example, super-class is in a different dex file. Don't
1498cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      // attempt to resolve methods and fields when there is no declaring class.
1499cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      CHECK(soa.Self()->IsExceptionPending());
1500cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      soa.Self()->ClearException();
1501cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      resolve_fields_and_methods = false;
1502cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    } else {
1503cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom      resolve_fields_and_methods = manager->GetCompiler()->IsImage();
1504cb5f5e53b580023fa2c1d8235c2e9aa1ff67d1dcBrian Carlstrom    }
1505e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // Note the class_data pointer advances through the headers,
1506e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // static fields, instance fields, direct methods, and virtual
1507e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // methods.
1508e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    const byte* class_data = dex_file.GetClassData(class_def);
1509e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    if (class_data == NULL) {
1510e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      // Empty class such as a marker interface.
1511e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      requires_constructor_barrier = false;
1512e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    } else {
1513e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      ClassDataItemIterator it(dex_file, class_data);
1514e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      while (it.HasNextStaticField()) {
1515e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        if (resolve_fields_and_methods) {
1516e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          mirror::ArtField* field = class_linker->ResolveField(dex_file, it.GetMemberIndex(),
1517e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers                                                               dex_cache, class_loader, true);
1518e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          if (field == NULL) {
1519e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            CHECK(soa.Self()->IsExceptionPending());
1520e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            soa.Self()->ClearException();
1521e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          }
1522e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        }
1523e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        it.Next();
1524e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      }
1525e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      // We require a constructor barrier if there are final instance fields.
1526e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      requires_constructor_barrier = false;
1527e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      while (it.HasNextInstanceField()) {
1528e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        if ((it.GetMemberAccessFlags() & kAccFinal) != 0) {
1529e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          requires_constructor_barrier = true;
1530e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        }
1531e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        if (resolve_fields_and_methods) {
1532e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          mirror::ArtField* field = class_linker->ResolveField(dex_file, it.GetMemberIndex(),
1533e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers                                                               dex_cache, class_loader, false);
1534e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          if (field == NULL) {
1535e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            CHECK(soa.Self()->IsExceptionPending());
1536e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            soa.Self()->ClearException();
1537e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          }
1538e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        }
1539e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        it.Next();
1540e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      }
1541e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      if (resolve_fields_and_methods) {
1542e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        while (it.HasNextDirectMethod()) {
1543e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          mirror::ArtMethod* method = class_linker->ResolveMethod(dex_file, it.GetMemberIndex(),
15440cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier                                                                  dex_cache, class_loader,
15450cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier                                                                  NullHandle<mirror::ArtMethod>(),
1546e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers                                                                  it.GetMethodInvokeType(class_def));
1547e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          if (method == NULL) {
1548e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            CHECK(soa.Self()->IsExceptionPending());
1549e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            soa.Self()->ClearException();
1550e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          }
1551e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          it.Next();
1552e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        }
1553e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        while (it.HasNextVirtualMethod()) {
1554e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          mirror::ArtMethod* method = class_linker->ResolveMethod(dex_file, it.GetMemberIndex(),
15550cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier                                                                  dex_cache, class_loader,
15560cd81352a7c06e381951cea1b104fd73516f4341Mathieu Chartier                                                                  NullHandle<mirror::ArtMethod>(),
1557e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers                                                                  it.GetMethodInvokeType(class_def));
1558e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          if (method == NULL) {
1559e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            CHECK(soa.Self()->IsExceptionPending());
1560e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers            soa.Self()->ClearException();
1561e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          }
1562e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers          it.Next();
1563e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        }
1564e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers        DCHECK(!it.HasNext());
1565e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      }
156620cfffabdc9e02b2df798bc4e6b6035d14bf4e36Brian Carlstrom    }
1567d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  }
1568fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers  if (requires_constructor_barrier) {
1569be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers    manager->GetCompiler()->AddRequiresConstructorBarrier(self, &dex_file, class_def_index);
1570fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers  }
1571d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes}
1572d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
1573219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogersstatic void ResolveType(const ParallelCompilationManager* manager, size_t type_idx)
1574b726dcb581bf72da46527378ccb6889020f0e6e9Ian Rogers    LOCKS_EXCLUDED(Locks::mutator_lock_) {
1575d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  // Class derived values are more complicated, they require the linker and loader.
157600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
1577219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ClassLinker* class_linker = manager->GetClassLinker();
1578219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  const DexFile& dex_file = *manager->GetDexFile();
1579eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  StackHandleScope<2> hs(soa.Self());
1580eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::DexCache> dex_cache(hs.NewHandle(class_linker->FindDexCache(dex_file)));
1581eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::ClassLoader> class_loader(
1582eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(soa.Decode<mirror::ClassLoader*>(manager->GetClassLoader())));
15832dd0e2cea360bc9206eb88ecc40d259e796c239dIan Rogers  mirror::Class* klass = class_linker->ResolveType(dex_file, type_idx, dex_cache, class_loader);
158400f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers
1585d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  if (klass == NULL) {
158600f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    CHECK(soa.Self()->IsExceptionPending());
1587a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers    mirror::Throwable* exception = soa.Self()->GetException(NULL);
1588a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers    VLOG(compiler) << "Exception during type resolution: " << exception->Dump();
1589f832284dd847ff077577bb5712225430bbbb3b67Mathieu Chartier    if (exception->GetClass()->DescriptorEquals("Ljava/lang/OutOfMemoryError;")) {
1590a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers      // There's little point continuing compilation if the heap is exhausted.
1591a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers      LOG(FATAL) << "Out of memory during type resolution for compilation";
1592a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers    }
1593a436fde2762664a3ecdda5eefcadd20b2e104f59Ian Rogers    soa.Self()->ClearException();
1594d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  }
1595d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes}
1596d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
15971212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::ResolveDexFile(jobject class_loader, const DexFile& dex_file,
15983d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                    ThreadPool* thread_pool, TimingLogger* timings) {
1599d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes  ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
1600d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
160100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  // TODO: we could resolve strings here, although the string table is largely filled with class
160200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  //       and method names.
1603d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
1604219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ParallelCompilationManager context(class_linker, class_loader, this, &dex_file, thread_pool);
1605e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  if (IsImage()) {
1606e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // For images we resolve all types, such as array, whereas for applications just those with
1607e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    // classdefs are resolved by ResolveClassFieldsAndMethods.
16083d504075f7c1204d581923460754bf6d3714b13fIan Rogers    timings->NewSplit("Resolve Types");
1609e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    context.ForAll(0, dex_file.NumTypeIds(), ResolveType, thread_count_);
1610e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  }
1611d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
16123d504075f7c1204d581923460754bf6d3714b13fIan Rogers  timings->NewSplit("Resolve MethodsAndFields");
16130e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  context.ForAll(0, dex_file.NumClassDefs(), ResolveClassFieldsAndMethods, thread_count_);
16149ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}
16159ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
16161212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::Verify(jobject class_loader, const std::vector<const DexFile*>& dex_files,
16173d504075f7c1204d581923460754bf6d3714b13fIan Rogers                            ThreadPool* thread_pool, TimingLogger* timings) {
1618ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  for (size_t i = 0; i != dex_files.size(); ++i) {
1619ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom    const DexFile* dex_file = dex_files[i];
162098eacac683b78e60799323e8c7d59e7214808639jeffhao    CHECK(dex_file != NULL);
16212f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom    VerifyDexFile(class_loader, *dex_file, thread_pool, timings);
162298eacac683b78e60799323e8c7d59e7214808639jeffhao  }
162398eacac683b78e60799323e8c7d59e7214808639jeffhao}
162498eacac683b78e60799323e8c7d59e7214808639jeffhao
1625219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogersstatic void VerifyClass(const ParallelCompilationManager* manager, size_t class_def_index)
1626b726dcb581bf72da46527378ccb6889020f0e6e9Ian Rogers    LOCKS_EXCLUDED(Locks::mutator_lock_) {
162767f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum  ATRACE_CALL();
162800f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
16290e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const DexFile& dex_file = *manager->GetDexFile();
16300e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const DexFile::ClassDef& class_def = dex_file.GetClassDef(class_def_index);
16310e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const char* descriptor = dex_file.GetClassDescriptor(class_def);
16320e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  ClassLinker* class_linker = manager->GetClassLinker();
16330e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  jobject jclass_loader = manager->GetClassLoader();
1634eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  StackHandleScope<3> hs(soa.Self());
1635eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::ClassLoader> class_loader(
1636eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(soa.Decode<mirror::ClassLoader*>(jclass_loader)));
1637eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::Class> klass(
1638eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(class_linker->FindClass(soa.Self(), descriptor, class_loader)));
1639eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  if (klass.Get() == nullptr) {
1640e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    CHECK(soa.Self()->IsExceptionPending());
164162d6c772205b8859f0ebf7ad105402ec4c3e2e01Ian Rogers    soa.Self()->ClearException();
1642f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao
1643f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao    /*
1644f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao     * At compile time, we can still structurally verify the class even if FindClass fails.
1645f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao     * This is to ensure the class is structurally sound for compilation. An unsound class
1646f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao     * will be rejected by the verifier and later skipped during compilation in the compiler.
1647f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao     */
1648eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::DexCache> dex_cache(hs.NewHandle(class_linker->FindDexCache(dex_file)));
1649f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao    std::string error_msg;
1650590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier    if (verifier::MethodVerifier::VerifyClass(&dex_file, dex_cache, class_loader, &class_def, true,
1651590fee9e8972f872301c2d16a575d579ee564beeMathieu Chartier                                              &error_msg) ==
165200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers                                                  verifier::MethodVerifier::kHardFailure) {
16530e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao      LOG(ERROR) << "Verification failed on class " << PrettyDescriptor(descriptor)
1654f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao                 << " because: " << error_msg;
1655f56197c5519395fcf8226a7148cc87367e0a7ad4jeffhao    }
1656eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  } else if (!SkipClass(jclass_loader, dex_file, klass.Get())) {
1657eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    CHECK(klass->IsResolved()) << PrettyClass(klass.Get());
16580e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao    class_linker->VerifyClass(klass);
1659d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
1660e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    if (klass->IsErroneous()) {
1661e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      // ClassLinker::VerifyClass throws, which isn't useful in the compiler.
1662e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      CHECK(soa.Self()->IsExceptionPending());
1663e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers      soa.Self()->ClearException();
1664e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    }
166567f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum
1666e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers    CHECK(klass->IsCompileTimeVerified() || klass->IsErroneous())
1667eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        << PrettyDescriptor(klass.Get()) << ": state=" << klass->GetStatus();
1668e6bb3b2ce5a69c31c2adfc7eb2705633b7f966ebIan Rogers  }
166962d6c772205b8859f0ebf7ad105402ec4c3e2e01Ian Rogers  soa.Self()->AssertNoPendingException();
1670d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes}
1671d9c67be7c116875d96b31e640ad47d587b205605Elliott Hughes
16721212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::VerifyDexFile(jobject class_loader, const DexFile& dex_file,
16733d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                   ThreadPool* thread_pool, TimingLogger* timings) {
16743d504075f7c1204d581923460754bf6d3714b13fIan Rogers  timings->NewSplit("Verify Dex File");
1675731b2abfccd8704d129e3b8e46a086660161fef3Brian Carlstrom  ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
1676219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ParallelCompilationManager context(class_linker, class_loader, this, &dex_file, thread_pool);
16770e4627e593bc39f8e3d89c31f8977d55054c07ccMathieu Chartier  context.ForAll(0, dex_file.NumClassDefs(), VerifyClass, thread_count_);
1678a5a97a2bc1dfed70869da34650a5a2a3a3a06ac4Brian Carlstrom}
1679a5a97a2bc1dfed70869da34650a5a2a3a3a06ac4Brian Carlstrom
1680219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogersstatic void InitializeClass(const ParallelCompilationManager* manager, size_t class_def_index)
16813d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers    LOCKS_EXCLUDED(Locks::mutator_lock_) {
16828f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  ATRACE_CALL();
16830e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  jobject jclass_loader = manager->GetClassLoader();
16840e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const DexFile& dex_file = *manager->GetDexFile();
16850e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao  const DexFile::ClassDef& class_def = dex_file.GetClassDef(class_def_index);
1686bcdbbfebc8f32566d4cb3f66405e89cdb7351992Jeff Hao  const DexFile::TypeId& class_type_id = dex_file.GetTypeId(class_def.class_idx_);
1687bcdbbfebc8f32566d4cb3f66405e89cdb7351992Jeff Hao  const char* descriptor = dex_file.StringDataByIdx(class_type_id.descriptor_idx_);
1688fc0e94bed3f88ed7e50854fd8dfaf5dcb345250fIan Rogers
168900f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  ScopedObjectAccess soa(Thread::Current());
1690eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  StackHandleScope<3> hs(soa.Self());
1691eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::ClassLoader> class_loader(
1692eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(soa.Decode<mirror::ClassLoader*>(jclass_loader)));
1693eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  Handle<mirror::Class> klass(
1694eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier      hs.NewHandle(manager->GetClassLinker()->FindClass(soa.Self(), descriptor, class_loader)));
16950e49b42e03af56521d8ce2c9c84ac5b79e6241c9Jeff Hao
1696eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier  if (klass.Get() != nullptr && !SkipClass(jclass_loader, dex_file, klass.Get())) {
169764b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers    // Only try to initialize classes that were successfully verified.
16983d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers    if (klass->IsVerified()) {
16998f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      // Attempt to initialize the class but bail if we either need to initialize the super-class
17008f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      // or static fields.
17018f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      manager->GetClassLinker()->EnsureInitialized(klass, false, false);
170264b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers      if (!klass->IsInitialized()) {
17038f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // We don't want non-trivial class initialization occurring on multiple threads due to
17048f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // deadlock problems. For example, a parent class is initialized (holding its lock) that
17058f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // refers to a sub-class in its static/class initializer causing it to try to acquire the
17068f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // sub-class' lock. While on a second thread the sub-class is initialized (holding its lock)
17078f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // after first initializing its parents, whose locks are acquired. This leads to a
17088f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // parent-to-child and a child-to-parent lock ordering and consequent potential deadlock.
17098f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // We need to use an ObjectLock due to potential suspension in the interpreting code. Rather
17108f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // than use a special Object for the purpose we use the Class of java.lang.Class.
1711eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        Handle<mirror::Class> h_klass(hs.NewHandle(klass->GetClass()));
1712db2633ce0358c704f97130a94b582602cb01d14aMathieu Chartier        ObjectLock<mirror::Class> lock(soa.Self(), h_klass);
17138f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // Attempt to initialize allowing initialization of parent classes but still not static
17148f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        // fields.
17158f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        manager->GetClassLinker()->EnsureInitialized(klass, false, true);
17168f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        if (!klass->IsInitialized()) {
17178f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers          // We need to initialize static fields, we only do this for image classes that aren't
1718d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz          // marked with the $NoPreloadHolder (which implies this should not be initialized early).
17198f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers          bool can_init_static_fields = manager->GetCompiler()->IsImage() &&
1720d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz              manager->GetCompiler()->IsImageClass(descriptor) &&
1721d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz              !StringPiece(descriptor).ends_with("$NoPreloadHolder;");
17228f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers          if (can_init_static_fields) {
1723d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz            VLOG(compiler) << "Initializing: " << descriptor;
1724c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // TODO multithreading support. We should ensure the current compilation thread has
1725c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // exclusive access to the runtime and the transaction. To achieve this, we could use
1726c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // a ReaderWriterMutex but we're holding the mutator lock so we fail mutex sanity
1727c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // checks in Thread::AssertThreadSuspensionIsAllowable.
1728c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            Runtime* const runtime = Runtime::Current();
1729c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            Transaction transaction;
1730c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers
1731c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // Run the class initializer in transaction mode.
1732c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            runtime->EnterTransactionMode(&transaction);
1733c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            const mirror::Class::Status old_status = klass->GetStatus();
1734c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            bool success = manager->GetClassLinker()->EnsureInitialized(klass, true, true);
1735c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // TODO we detach transaction from runtime to indicate we quit the transactional
1736c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // mode which prevents the GC from visiting objects modified during the transaction.
1737c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            // Ensure GC is not run so don't access freed objects when aborting transaction.
1738c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            const char* old_casue = soa.Self()->StartAssertNoThreadSuspension("Transaction end");
1739c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            runtime->ExitTransactionMode();
1740c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers
1741c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            if (!success) {
1742c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              CHECK(soa.Self()->IsExceptionPending());
1743c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              ThrowLocation throw_location;
1744c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              mirror::Throwable* exception = soa.Self()->GetException(&throw_location);
1745c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              VLOG(compiler) << "Initialization of " << descriptor << " aborted because of "
1746c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers                  << exception->Dump();
1747c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              soa.Self()->ClearException();
1748c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              transaction.Abort();
1749c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers              CHECK_EQ(old_status, klass->GetStatus()) << "Previous class status not restored";
175064b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers            }
1751c45b8b582be5c98941ca3869fcdc9a08d520da41Ian Rogers            soa.Self()->EndAssertNoThreadSuspension(old_casue);
175264b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers          }
175364b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers        }
17548f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        soa.Self()->AssertNoPendingException();
175564b6d145fa53b8dfb07a8fc2426af13f155d5a4dIan Rogers      }
175627ec961a1da540ba7f16c07a682585ab167317adBrian Carlstrom    }
17573d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers    // Record the final class status if necessary.
175851c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom    ClassReference ref(manager->GetDexFile(), class_def_index);
17598f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    manager->GetCompiler()->RecordClassStatus(ref, klass->GetStatus());
176098eacac683b78e60799323e8c7d59e7214808639jeffhao  }
17611f5393447b9f45be7918042d9ee7b521376de866Ian Rogers  // Clear any class not found or verification exceptions.
176262d6c772205b8859f0ebf7ad105402ec4c3e2e01Ian Rogers  soa.Self()->ClearException();
176398eacac683b78e60799323e8c7d59e7214808639jeffhao}
176498eacac683b78e60799323e8c7d59e7214808639jeffhao
17651212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::InitializeClasses(jobject jni_class_loader, const DexFile& dex_file,
17663d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                       ThreadPool* thread_pool, TimingLogger* timings) {
17673d504075f7c1204d581923460754bf6d3714b13fIan Rogers  timings->NewSplit("InitializeNoClinit");
17683d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers  ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
1769219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  ParallelCompilationManager context(class_linker, jni_class_loader, this, &dex_file, thread_pool);
1770d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  size_t thread_count;
1771d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  if (IsImage()) {
1772d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz    // TODO: remove this when transactional mode supports multithreading.
1773d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz    thread_count = 1U;
1774d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  } else {
1775d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz    thread_count = thread_count_;
1776d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  }
1777d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  context.ForAll(0, dex_file.NumClassDefs(), InitializeClass, thread_count);
1778d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  if (IsImage()) {
1779d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz    // Prune garbage objects created during aborted transactions.
1780d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz    Runtime::Current()->GetHeap()->CollectGarbage(true);
1781d2fe10a3a34af171bf1631219cd2d6ff6b7778b5Sebastien Hertz  }
17823d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers}
178390dc30f4b9967e850d0594e57dfa8e7cb0369575Shih-wei Liao
17841212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::InitializeClasses(jobject class_loader,
17851212a022fa5f8ef9585d765b1809521812af882cIan Rogers                                       const std::vector<const DexFile*>& dex_files,
17863d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                       ThreadPool* thread_pool, TimingLogger* timings) {
17873d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers  for (size_t i = 0; i != dex_files.size(); ++i) {
17883d1548debdaf8a1915fa432a4d267f7c0145654dIan Rogers    const DexFile* dex_file = dex_files[i];
178990dc30f4b9967e850d0594e57dfa8e7cb0369575Shih-wei Liao    CHECK(dex_file != NULL);
17902f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom    InitializeClasses(class_loader, *dex_file, thread_pool, timings);
179100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  }
179290dc30f4b9967e850d0594e57dfa8e7cb0369575Shih-wei Liao}
179390dc30f4b9967e850d0594e57dfa8e7cb0369575Shih-wei Liao
17941212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::Compile(jobject class_loader, const std::vector<const DexFile*>& dex_files,
17953d504075f7c1204d581923460754bf6d3714b13fIan Rogers                             ThreadPool* thread_pool, TimingLogger* timings) {
1796ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom  for (size_t i = 0; i != dex_files.size(); ++i) {
1797ae826983f7903bc0a6bbbe8426bf393fb2f6d747Brian Carlstrom    const DexFile* dex_file = dex_files[i];
179883db7721aef15df6919c0ec072e087bef6041e2dBrian Carlstrom    CHECK(dex_file != NULL);
17992f66382fdb5e98537f724eba43ef1c7162c71b0eBrian Carlstrom    CompileDexFile(class_loader, *dex_file, thread_pool, timings);
180083db7721aef15df6919c0ec072e087bef6041e2dBrian Carlstrom  }
180183db7721aef15df6919c0ec072e087bef6041e2dBrian Carlstrom}
180283db7721aef15df6919c0ec072e087bef6041e2dBrian Carlstrom
1803219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogersvoid CompilerDriver::CompileClass(const ParallelCompilationManager* manager, size_t class_def_index) {
180467f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum  ATRACE_CALL();
18052d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  jobject jclass_loader = manager->GetClassLoader();
1806219b5a847ef74be5d3de4c16a29ec6413cc42af1Ian Rogers  const DexFile& dex_file = *manager->GetDexFile();
1807c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes  const DexFile::ClassDef& class_def = dex_file.GetClassDef(class_def_index);
1808be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  ClassLinker* class_linker = manager->GetClassLinker();
1809be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  if (SkipClass(class_linker, jclass_loader, dex_file, class_def)) {
1810be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers    return;
18115ead0950c661761e90e04aefd6ea2205532ce874Brian Carlstrom  }
1812d1224c79631bd1801b067a0f212b91afa961a362jeffhao  ClassReference ref(&dex_file, class_def_index);
1813d1224c79631bd1801b067a0f212b91afa961a362jeffhao  // Skip compiling classes with generic verifier failures since they will still fail at runtime
1814c7f832061fea59fd6abd125f26c8ca1faec695a5Vladimir Marko  if (manager->GetCompiler()->verification_results_->IsClassRejected(ref)) {
1815d1224c79631bd1801b067a0f212b91afa961a362jeffhao    return;
1816d1224c79631bd1801b067a0f212b91afa961a362jeffhao  }
18170571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  const byte* class_data = dex_file.GetClassData(class_def);
18180571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  if (class_data == NULL) {
18190571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    // empty class, probably a marker interface
18200571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    return;
18210571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  }
182267f99418f648c3a95256ed3dcd8e8b64eef0b372Anwar Ghuloum
18232d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  // Can we run DEX-to-DEX compiler on this class ?
182475021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz  DexToDexCompilationLevel dex_to_dex_compilation_level = kDontDexToDexCompile;
18252d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  {
18262d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz    ScopedObjectAccess soa(Thread::Current());
1827eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    StackHandleScope<1> hs(soa.Self());
1828eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier    Handle<mirror::ClassLoader> class_loader(
1829eb8167a4f4d27fce0530f6724ab8032610cd146bMathieu Chartier        hs.NewHandle(soa.Decode<mirror::ClassLoader*>(jclass_loader)));
18309837939678bb5dcba178e5fb00ed59b5d14c8d9bIan Rogers    dex_to_dex_compilation_level = GetDexToDexCompilationlevel(soa.Self(), class_loader, dex_file,
18319837939678bb5dcba178e5fb00ed59b5d14c8d9bIan Rogers                                                               class_def);
18322d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz  }
18330571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  ClassDataItemIterator it(dex_file, class_data);
18340571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  // Skip fields
18350571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  while (it.HasNextStaticField()) {
18360571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    it.Next();
18370571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  }
18380571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  while (it.HasNextInstanceField()) {
18390571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    it.Next();
18400571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  }
1841be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  CompilerDriver* driver = manager->GetCompiler();
18420571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  // Compile direct methods
184368adbe41c7d9295da2bfc521d737ba6dabd36c98Brian Carlstrom  int64_t previous_direct_method_idx = -1;
18440571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  while (it.HasNextDirectMethod()) {
18456f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    uint32_t method_idx = it.GetMemberIndex();
18466f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    if (method_idx == previous_direct_method_idx) {
18476f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      // smali can create dex files with two encoded_methods sharing the same method_idx
18486f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      // http://code.google.com/p/smali/issues/detail?id=119
18496f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      it.Next();
18506f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      continue;
18516f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    }
18526f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    previous_direct_method_idx = method_idx;
1853be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers    driver->CompileMethod(it.GetMethodCodeItem(), it.GetMemberAccessFlags(),
1854be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                          it.GetMethodInvokeType(class_def), class_def_index,
1855be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                          method_idx, jclass_loader, dex_file, dex_to_dex_compilation_level);
18560571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    it.Next();
18579ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom  }
18580571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  // Compile virtual methods
185968adbe41c7d9295da2bfc521d737ba6dabd36c98Brian Carlstrom  int64_t previous_virtual_method_idx = -1;
18600571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  while (it.HasNextVirtualMethod()) {
18616f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    uint32_t method_idx = it.GetMemberIndex();
18626f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    if (method_idx == previous_virtual_method_idx) {
18636f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      // smali can create dex files with two encoded_methods sharing the same method_idx
18646f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      // http://code.google.com/p/smali/issues/detail?id=119
18656f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      it.Next();
18666f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom      continue;
18676f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    }
18686f29d0e6d5444ff84157c922c23c221567dcc6c5Brian Carlstrom    previous_virtual_method_idx = method_idx;
1869be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers    driver->CompileMethod(it.GetMethodCodeItem(), it.GetMemberAccessFlags(),
1870be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                          it.GetMethodInvokeType(class_def), class_def_index,
1871be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                          method_idx, jclass_loader, dex_file, dex_to_dex_compilation_level);
18720571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    it.Next();
18739ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom  }
18740571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  DCHECK(!it.HasNext());
18759ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}
18769ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
18771212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::CompileDexFile(jobject class_loader, const DexFile& dex_file,
18783d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                    ThreadPool* thread_pool, TimingLogger* timings) {
18793d504075f7c1204d581923460754bf6d3714b13fIan Rogers  timings->NewSplit("Compile Dex File");
1880be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers  ParallelCompilationManager context(Runtime::Current()->GetClassLinker(), class_loader, this,
1881be7149fc2e7cc607937209f2819e3c1d672e2668Ian Rogers                                     &dex_file, thread_pool);
18821212a022fa5f8ef9585d765b1809521812af882cIan Rogers  context.ForAll(0, dex_file.NumClassDefs(), CompilerDriver::CompileClass, thread_count_);
1883c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes}
1884c225caa9715eeaeff87f27d5b6a3e7d4f6b7efadElliott Hughes
18851212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::CompileMethod(const DexFile::CodeItem* code_item, uint32_t access_flags,
18868b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                   InvokeType invoke_type, uint16_t class_def_idx,
18871212a022fa5f8ef9585d765b1809521812af882cIan Rogers                                   uint32_t method_idx, jobject class_loader,
18882d6ba5158d7fd459db2870df47300b517dc4d08cSebastien Hertz                                   const DexFile& dex_file,
188975021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz                                   DexToDexCompilationLevel dex_to_dex_compilation_level) {
1890f09afe8fbd78943df6a8b10f03c36dcd190dd054Elliott Hughes  CompiledMethod* compiled_method = NULL;
1891bb551fa68ffc57f679b8c914ac856666f0348b77Elliott Hughes  uint64_t start_ns = NanoTime();
18924dd96f56909ec35c83a3d468b0e47769988c1a1dLogan Chien
1893169c9a7f46776b235d0a37d5e0ff27682deffe06Ian Rogers  if ((access_flags & kAccNative) != 0) {
18940188ab739c3ef8bcd521d88f84894db8c0f012c5Ian Rogers    // Are we interpreting only and have support for generic JNI down calls?
18954a200f56b7075309316b04d550c9cc50f8314eddJeff Hao    if (!compiler_options_->IsCompilationEnabled() &&
1896b95a5345ae4217b70ca36f0cced92f68dda7caf5Stuart Monteith        (instruction_set_ == kX86_64 || instruction_set_ == kArm64)) {
18975b27149ce3cbf39f99c9d9c8ba5ebf1657c297e1Ian Rogers      // Leaving this empty will trigger the generic JNI version
18985b27149ce3cbf39f99c9d9c8ba5ebf1657c297e1Ian Rogers    } else {
189972d32629303f8f39362a4099481f48646aed042fIan Rogers      compiled_method = compiler_->JniCompile(access_flags, method_idx, dex_file);
19005b27149ce3cbf39f99c9d9c8ba5ebf1657c297e1Ian Rogers      CHECK(compiled_method != NULL);
19015b27149ce3cbf39f99c9d9c8ba5ebf1657c297e1Ian Rogers    }
1902169c9a7f46776b235d0a37d5e0ff27682deffe06Ian Rogers  } else if ((access_flags & kAccAbstract) != 0) {
19032cc022b653e1e84eed2522254ec684bd097572b8Brian Carlstrom  } else {
190490af14d2743614e3e1453984b14258a6f145501dDragos Sbirlea    MethodReference method_ref(&dex_file, method_idx);
19056449c62e40ef3a9bb75f664f922555affb532ee4Brian Carlstrom    bool compile = verification_results_->IsCandidateForCompilation(method_ref, access_flags);
19064d4adb1dae07bb7421e863732ab789413a3b43f0Sebastien Hertz    if (compile) {
1907a024a0686c3b0fea13f362bff70d65981e5febc5buzbee      // NOTE: if compiler declines to compile this method, it will return NULL.
190872d32629303f8f39362a4099481f48646aed042fIan Rogers      compiled_method = compiler_->Compile(code_item, access_flags, invoke_type, class_def_idx,
190972d32629303f8f39362a4099481f48646aed042fIan Rogers                                           method_idx, class_loader, dex_file);
191017965ed6ec29407599a0bcd4ea92532424294fedSebastien Hertz    }
191117965ed6ec29407599a0bcd4ea92532424294fedSebastien Hertz    if (compiled_method == nullptr && dex_to_dex_compilation_level != kDontDexToDexCompile) {
191217965ed6ec29407599a0bcd4ea92532424294fedSebastien Hertz      // TODO: add a command-line option to disable DEX-to-DEX compilation ?
191375021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz      (*dex_to_dex_compiler_)(*this, code_item, access_flags,
191475021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz                              invoke_type, class_def_idx,
191575021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz                              method_idx, class_loader, dex_file,
191675021222d9c03a80fa5c136db0d5fb8d82d04031Sebastien Hertz                              dex_to_dex_compilation_level);
1917f3e9855ee2000106b54fd479f7a46da2dc2ad079Ian Rogers    }
1918bb551fa68ffc57f679b8c914ac856666f0348b77Elliott Hughes  }
19193bb17a644e2945c3913cfbde245d2f520d62a3ffIan Rogers  uint64_t duration_ns = NanoTime() - start_ns;
1920b34f69ab43aaf7a6e6045c95f398baf566ef5023Nicolas Geoffray  if (duration_ns > MsToNs(compiler_->GetMaximumCompilationTimeBeforeWarning())) {
1921bb551fa68ffc57f679b8c914ac856666f0348b77Elliott Hughes    LOG(WARNING) << "Compilation of " << PrettyMethod(method_idx, dex_file)
19223bb17a644e2945c3913cfbde245d2f520d62a3ffIan Rogers                 << " took " << PrettyDuration(duration_ns);
1923f09afe8fbd78943df6a8b10f03c36dcd190dd054Elliott Hughes  }
1924f09afe8fbd78943df6a8b10f03c36dcd190dd054Elliott Hughes
192550b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  Thread* self = Thread::Current();
1926f09afe8fbd78943df6a8b10f03c36dcd190dd054Elliott Hughes  if (compiled_method != NULL) {
19270571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers    MethodReference ref(&dex_file, method_idx);
19288f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    DCHECK(GetCompiledMethod(ref) == NULL) << PrettyMethod(method_idx, dex_file);
192900f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    {
193050b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers      MutexLock mu(self, compiled_methods_lock_);
193100f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers      compiled_methods_.Put(ref, compiled_method);
193200f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    }
19330755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom    DCHECK(GetCompiledMethod(ref) != NULL) << PrettyMethod(method_idx, dex_file);
19342cc022b653e1e84eed2522254ec684bd097572b8Brian Carlstrom  }
19359baa4aefc370f48774b6104680193d9a7e4fb631Brian Carlstrom
193650b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  if (self->IsExceptionPending()) {
193750b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers    ScopedObjectAccess soa(self);
193800f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers    LOG(FATAL) << "Unexpected exception compiling: " << PrettyMethod(method_idx, dex_file) << "\n"
193962d6c772205b8859f0ebf7ad105402ec4c3e2e01Ian Rogers        << self->GetException(NULL)->Dump();
194000f7d0eaa6bd93d33bf0c1429bf4ba0b3f28abacIan Rogers  }
19410571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers}
194228ad40dc3ec2f09b0ffd4f6d6787bf1b532ccd5dIan Rogers
19431212a022fa5f8ef9585d765b1809521812af882cIan RogersCompiledClass* CompilerDriver::GetCompiledClass(ClassReference ref) const {
194450b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  MutexLock mu(Thread::Current(), compiled_classes_lock_);
19450755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom  ClassTable::const_iterator it = compiled_classes_.find(ref);
19460755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom  if (it == compiled_classes_.end()) {
19470755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom    return NULL;
19480755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom  }
19490755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom  CHECK(it->second != NULL);
19500755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom  return it->second;
19510755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom}
19520755ec5ea1dce0b549fc1adefeb52d89f119ebecBrian Carlstrom
19538f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogersvoid CompilerDriver::RecordClassStatus(ClassReference ref, mirror::Class::Status status) {
19548f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  MutexLock mu(Thread::Current(), compiled_classes_lock_);
19558f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  auto it = compiled_classes_.find(ref);
19568f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  if (it == compiled_classes_.end() || it->second->GetStatus() != status) {
19578f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    // An entry doesn't exist or the status is lower than the new status.
19588f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    if (it != compiled_classes_.end()) {
19598f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      CHECK_GT(status, it->second->GetStatus());
19608f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      delete it->second;
19618f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    }
19628f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    switch (status) {
19638f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      case mirror::Class::kStatusNotReady:
19648f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      case mirror::Class::kStatusError:
19658f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      case mirror::Class::kStatusRetryVerificationAtRuntime:
19668f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      case mirror::Class::kStatusVerified:
19678f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      case mirror::Class::kStatusInitialized:
19688f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        break;  // Expected states.
19698f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers      default:
19708f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers        LOG(FATAL) << "Unexpected class status for class "
19718f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers            << PrettyDescriptor(ref.first->GetClassDescriptor(ref.first->GetClassDef(ref.second)))
19728f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers            << " of " << status;
19738f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    }
19748f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    CompiledClass* compiled_class = new CompiledClass(status);
19758f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers    compiled_classes_.Overwrite(ref, compiled_class);
19768f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  }
19778f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers}
19788f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers
19791212a022fa5f8ef9585d765b1809521812af882cIan RogersCompiledMethod* CompilerDriver::GetCompiledMethod(MethodReference ref) const {
198050b35e2fd1a68cd1240e4a9d9f363e11764957d1Ian Rogers  MutexLock mu(Thread::Current(), compiled_methods_lock_);
19810571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  MethodTable::const_iterator it = compiled_methods_.find(ref);
19820571d357843c53e042f370f5f2c2e9aa3fe803a9Ian Rogers  if (it == compiled_methods_.end()) {
19833320cf46afd082398aa401b246e6f301cebdf64dBrian Carlstrom    return NULL;
19842c8f653c98d658419f464b6147c10e11a664d2e6Ian Rogers  }
19853320cf46afd082398aa401b246e6f301cebdf64dBrian Carlstrom  CHECK(it->second != NULL);
19863320cf46afd082398aa401b246e6f301cebdf64dBrian Carlstrom  return it->second;
19879ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}
19889ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom
19891212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::AddRequiresConstructorBarrier(Thread* self, const DexFile* dex_file,
19908b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                                   uint16_t class_def_index) {
19918f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  WriterMutexLock mu(self, freezing_constructor_lock_);
1992fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers  freezing_constructor_classes_.insert(ClassReference(dex_file, class_def_index));
1993fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers}
1994fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers
19951212a022fa5f8ef9585d765b1809521812af882cIan Rogersbool CompilerDriver::RequiresConstructorBarrier(Thread* self, const DexFile* dex_file,
19968b2c0b9abc3f520495f4387ea040132ba85cae69Ian Rogers                                                uint16_t class_def_index) {
19978f3c9ae38df2460940a26dff889a84430b6c38d3Ian Rogers  ReaderMutexLock mu(self, freezing_constructor_lock_);
1998fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers  return freezing_constructor_classes_.count(ClassReference(dex_file, class_def_index)) != 0;
1999fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers}
2000fffdb023275613612a22ec62b3421ffe4d2b73feIan Rogers
20013f47c12487250f61f3be95e9f275e3b08e2c49fbBrian Carlstrombool CompilerDriver::WriteElf(const std::string& android_root,
2002265091e581c9f643b37e7966890911f09e223269Brian Carlstrom                              bool is_host,
200351c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                              const std::vector<const art::DexFile*>& dex_files,
20043d504075f7c1204d581923460754bf6d3714b13fIan Rogers                              OatWriter* oat_writer,
200551c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom                              art::File* file)
200651c2467e8771b56e25ae4f17f66522f979f57a7eBrian Carlstrom    SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {
200772d32629303f8f39362a4099481f48646aed042fIan Rogers  return compiler_->WriteElf(file, oat_writer, dex_files, android_root, is_host);
2008265091e581c9f643b37e7966890911f09e223269Brian Carlstrom}
20091212a022fa5f8ef9585d765b1809521812af882cIan Rogersvoid CompilerDriver::InstructionSetToLLVMTarget(InstructionSet instruction_set,
20103d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                                std::string* target_triple,
20113d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                                std::string* target_cpu,
20123d504075f7c1204d581923460754bf6d3714b13fIan Rogers                                                std::string* target_attr) {
2013265091e581c9f643b37e7966890911f09e223269Brian Carlstrom  switch (instruction_set) {
2014700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    case kThumb2:
20153d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_triple = "thumb-none-linux-gnueabi";
20163d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_cpu = "cortex-a9";
20173d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_attr = "+thumb2,+neon,+neonfp,+vfp3,+db";
2018700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      break;
2019700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom
2020700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    case kArm:
20213d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_triple = "armv7-none-linux-gnueabi";
2022700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      // TODO: Fix for Nexus S.
20233d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_cpu = "cortex-a9";
2024700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      // TODO: Fix for Xoom.
20253d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_attr = "+v7,+neon,+neonfp,+vfp3,+db";
2026700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      break;
2027700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom
2028700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    case kX86:
20293d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_triple = "i386-pc-linux-gnu";
20303d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_attr = "";
2031700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      break;
2032700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom
20336a58cb16d803c9a7b3a75ccac8be19dd9d4e520dDmitry Petrochenko    case kX86_64:
20346a58cb16d803c9a7b3a75ccac8be19dd9d4e520dDmitry Petrochenko      *target_triple = "x86_64-pc-linux-gnu";
20356a58cb16d803c9a7b3a75ccac8be19dd9d4e520dDmitry Petrochenko      *target_attr = "";
20366a58cb16d803c9a7b3a75ccac8be19dd9d4e520dDmitry Petrochenko      break;
20376a58cb16d803c9a7b3a75ccac8be19dd9d4e520dDmitry Petrochenko
2038700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    case kMips:
20393d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_triple = "mipsel-unknown-linux";
20403d504075f7c1204d581923460754bf6d3714b13fIan Rogers      *target_attr = "mips32r2";
2041700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      break;
2042700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom
2043700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    default:
2044700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom      LOG(FATAL) << "Unknown instruction set: " << instruction_set;
2045700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom    }
2046700c8d31733534a3d978b75a03f6f7e177dc7e81Brian Carlstrom  }
204739c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison
204839c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allisonbool CompilerDriver::SkipCompilation(const std::string& method_name) {
204939c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  if (!profile_ok_) {
2050644789fa0e807a9aa6d6e32d36ce039ec7318a7fDave Allison    return false;
205139c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  }
205204ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  // Methods that comprise topKPercentThreshold % of the total samples will be compiled.
205304ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  double topKPercentThreshold = 90.0;
205404ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle#ifdef HAVE_ANDROID_OS
205504ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  char buf[PROP_VALUE_MAX];
205604ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  property_get("dalvik.vm.profile.compile_thr", buf, "90.0");
205704ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  topKPercentThreshold = strtod(buf, nullptr);
205804ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle#endif
205904ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  // Test for reasonable thresholds.
206004ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  if (topKPercentThreshold < 10.0 || topKPercentThreshold > 90.0) {
206104ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle    topKPercentThreshold = 90.0;
206204ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  }
206339c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison
206439c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  // First find the method in the profile map.
206539c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  ProfileMap::iterator i = profile_map_.find(method_name);
206639c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  if (i == profile_map_.end()) {
206739c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison    // Not in profile, no information can be determined.
206839c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison    VLOG(compiler) << "not compiling " << method_name << " because it's not in the profile";
206939c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison    return true;
207039c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  }
207139c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  const ProfileData& data = i->second;
207204ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle
2073f6a4cee66a173ee7ef48af5503d9899aa93b3aebCalin Juravle  // Compare against the start of the topK percentage bucket just in case the threshold
207404ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  // falls inside a bucket.
207504ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle  bool compile = data.GetTopKUsedPercentage() - data.GetUsedPercent() <= topKPercentThreshold;
207639c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  if (compile) {
2077f6a4cee66a173ee7ef48af5503d9899aa93b3aebCalin Juravle    LOG(INFO) << "compiling method " << method_name << " because its usage is part of top "
2078f6a4cee66a173ee7ef48af5503d9899aa93b3aebCalin Juravle        << data.GetTopKUsedPercentage() << "% with a percent of " << data.GetUsedPercent() << "%";
207939c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  } else {
208004ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle    VLOG(compiler) << "not compiling method " << method_name << " because it's not part of leading "
208104ff226764b125c79a0bf41e571c8311bac8a868Calin Juravle        << topKPercentThreshold << "% samples)";
208239c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  }
208339c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison  return !compile;
208439c3bfbd03d85c63cfbe69f17ce5800ccc7d6c13Dave Allison}
20859ea1cb1a22be5b85dc2622e3836c46a1c48e3f25Brian Carlstrom}  // namespace art
2086