blob: c0732aab9461aa10065869c4e4e5168dba97df7b [file] [log] [blame]
Elliott Hughes2faa5f12012-01-30 14:42:07 -08001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070016
17#ifndef ART_SRC_COMPILER_H_
18#define ART_SRC_COMPILER_H_
19
Elliott Hughes8add92d2012-01-18 18:18:43 -080020#include <set>
21#include <string>
Logan Chiendf576142012-03-20 17:36:32 +080022#include <vector>
Elliott Hughese5448b52012-01-18 16:44:06 -080023
Brian Carlstrom0755ec52012-01-11 15:19:46 -080024#include "compiled_class.h"
Brian Carlstrom3320cf42011-10-04 14:58:28 -070025#include "compiled_method.h"
Ian Rogersa3760aa2011-11-14 14:32:37 -080026#include "dex_cache.h"
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070027#include "dex_file.h"
Logan Chiendf576142012-03-20 17:36:32 +080028#include "elf_image.h"
Elliott Hughes0f3c5532012-03-30 14:51:51 -070029#include "instruction_set.h"
30#include "invoke_type.h"
Brian Carlstrom3320cf42011-10-04 14:58:28 -070031#include "oat_file.h"
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070032#include "object.h"
Ian Rogers1cb0a1d2011-10-06 15:24:35 -070033#include "runtime.h"
Elliott Hughesa0e18062012-04-13 15:59:59 -070034#include "safe_map.h"
Ian Rogers0571d352011-11-03 19:51:38 -070035
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070036namespace art {
37
Ian Rogersc8b306f2012-02-17 21:34:44 -080038class AOTCompilationStats;
Elliott Hughesc225caa2012-02-03 15:43:37 -080039class Context;
Logan Chien4dd96f52012-02-29 01:26:58 +080040class OatCompilationUnit;
Elliott Hughes601a1232012-02-02 17:47:38 -080041class TimingLogger;
Ian Rogers996cc582012-02-14 22:23:29 -080042
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070043class Compiler {
44 public:
Brian Carlstromaded5f72011-10-07 17:15:04 -070045 // Create a compiler targeting the requested "instruction_set".
Brian Carlstromae826982011-11-09 01:33:42 -080046 // "image" should be true if image specific optimizations should be
47 // enabled. "image_classes" lets the compiler know what classes it
48 // can assume will be in the image, with NULL implying all available
49 // classes.
Elliott Hughes5523ee02012-02-03 18:18:34 -080050 explicit Compiler(InstructionSet instruction_set, bool image, size_t thread_count,
Brian Carlstromba0668e2012-03-26 13:14:07 -070051 bool support_debugging, const std::set<std::string>* image_classes,
52 bool dump_stats, bool dump_timings);
Brian Carlstrom3320cf42011-10-04 14:58:28 -070053
54 ~Compiler();
Ian Rogers2c8f6532011-09-02 17:16:34 -070055
Jesse Wilson254db0f2011-11-16 16:44:11 -050056 void CompileAll(const ClassLoader* class_loader,
Brian Carlstromae826982011-11-09 01:33:42 -080057 const std::vector<const DexFile*>& dex_files);
Brian Carlstrom8a487412011-08-29 20:08:52 -070058
59 // Compile a single Method
Brian Carlstrom3320cf42011-10-04 14:58:28 -070060 void CompileOne(const Method* method);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -070061
Elliott Hughesde6e4cf2012-02-27 14:46:06 -080062 bool IsDebuggingSupported() {
63 return support_debugging_;
64 }
65
Brian Carlstrom3320cf42011-10-04 14:58:28 -070066 InstructionSet GetInstructionSet() const {
67 return instruction_set_;
68 }
69
Brian Carlstromaded5f72011-10-07 17:15:04 -070070 bool IsImage() const {
71 return image_;
72 }
73
Brian Carlstrome24fa612011-09-29 00:53:55 -070074 // Stub to throw AbstractMethodError
Brian Carlstrome24fa612011-09-29 00:53:55 -070075 static ByteArray* CreateAbstractMethodErrorStub(InstructionSet instruction_set);
76
Brian Carlstrom3320cf42011-10-04 14:58:28 -070077
Ian Rogersad25ac52011-10-04 19:13:33 -070078 // Generate the trampoline that's invoked by unresolved direct methods
Ian Rogers1cb0a1d2011-10-06 15:24:35 -070079 static ByteArray* CreateResolutionStub(InstructionSet instruction_set,
80 Runtime::TrampolineType type);
Ian Rogersad25ac52011-10-04 19:13:33 -070081
Elliott Hughes8add92d2012-01-18 18:18:43 -080082 static ByteArray* CreateJniDlsymLookupStub(InstructionSet instruction_set);
Ian Rogers169c9a72011-11-13 20:13:17 -080083
Brian Carlstrom0755ec52012-01-11 15:19:46 -080084 // A class is uniquely located by its DexFile and the class_defs_ table index into that DexFile
85 typedef std::pair<const DexFile*, uint32_t> ClassReference;
Elliott Hughes8add92d2012-01-18 18:18:43 -080086
Brian Carlstrom0755ec52012-01-11 15:19:46 -080087 CompiledClass* GetCompiledClass(ClassReference ref) const;
Ian Rogers0571d352011-11-03 19:51:38 -070088
Brian Carlstrom0755ec52012-01-11 15:19:46 -080089 // A method is uniquely located by its DexFile and the method_ids_ table index into that DexFile
90 typedef std::pair<const DexFile*, uint32_t> MethodReference;
Elliott Hughes8add92d2012-01-18 18:18:43 -080091
Ian Rogers0571d352011-11-03 19:51:38 -070092 CompiledMethod* GetCompiledMethod(MethodReference ref) const;
Brian Carlstrom0755ec52012-01-11 15:19:46 -080093
Ian Rogers0571d352011-11-03 19:51:38 -070094 const CompiledInvokeStub* FindInvokeStub(bool is_static, const char* shorty) const;
Elliott Hughesa0e18062012-04-13 15:59:59 -070095 const CompiledInvokeStub* FindInvokeStub(const std::string& key) const;
Brian Carlstrom3320cf42011-10-04 14:58:28 -070096
Ian Rogers28ad40d2011-10-27 15:19:26 -070097 // Callbacks from OAT/ART compiler to see what runtime checks must be generated
Ian Rogers0571d352011-11-03 19:51:38 -070098
Ian Rogers996cc582012-02-14 22:23:29 -080099 bool CanAssumeTypeIsPresentInDexCache(const DexCache* dex_cache, uint32_t type_idx);
Ian Rogers1bddec32012-02-04 12:27:34 -0800100
Ian Rogers996cc582012-02-14 22:23:29 -0800101 bool CanAssumeStringIsPresentInDexCache(const DexCache* dex_cache, uint32_t string_idx);
Ian Rogers1bddec32012-02-04 12:27:34 -0800102
103 // Are runtime access checks necessary in the compiled code?
104 bool CanAccessTypeWithoutChecks(uint32_t referrer_idx, const DexCache* dex_cache,
Ian Rogers996cc582012-02-14 22:23:29 -0800105 const DexFile& dex_file, uint32_t type_idx);
Ian Rogers1bddec32012-02-04 12:27:34 -0800106
107 // Are runtime access and instantiable checks necessary in the code?
Ian Rogersd4135902012-02-03 18:05:08 -0800108 bool CanAccessInstantiableTypeWithoutChecks(uint32_t referrer_idx, const DexCache* dex_cache,
Ian Rogers996cc582012-02-14 22:23:29 -0800109 const DexFile& dex_file, uint32_t type_idx);
Ian Rogers1bddec32012-02-04 12:27:34 -0800110
111 // Can we fast path instance field access? Computes field's offset and volatility
Logan Chien4dd96f52012-02-29 01:26:58 +0800112 bool ComputeInstanceFieldInfo(uint32_t field_idx, OatCompilationUnit* mUnit,
jeffhao8cd6dda2012-02-22 10:15:34 -0800113 int& field_offset, bool& is_volatile, bool is_put);
Ian Rogers1bddec32012-02-04 12:27:34 -0800114
115 // Can we fastpath static field access? Computes field's offset, volatility and whether the
116 // field is within the referrer (which can avoid checking class initialization)
Logan Chien4dd96f52012-02-29 01:26:58 +0800117 bool ComputeStaticFieldInfo(uint32_t field_idx, OatCompilationUnit* mUnit,
Ian Rogers1bddec32012-02-04 12:27:34 -0800118 int& field_offset, int& ssb_index,
jeffhao8cd6dda2012-02-22 10:15:34 -0800119 bool& is_referrers_class, bool& is_volatile, bool is_put);
Ian Rogers1bddec32012-02-04 12:27:34 -0800120
Ian Rogersa32a6fd2012-02-06 20:18:44 -0800121 // Can we fastpath a interface, super class or virtual method call? Computes method's vtable index
Ian Rogersfb6adba2012-03-04 21:51:51 -0800122 bool ComputeInvokeInfo(uint32_t method_idx, OatCompilationUnit* mUnit, InvokeType& type,
Ian Rogers2ed3b952012-03-17 11:49:39 -0700123 int& vtable_idx, uintptr_t& direct_code, uintptr_t& direct_method);
Ian Rogersa32a6fd2012-02-06 20:18:44 -0800124
Ian Rogers3fa13792012-03-18 15:53:45 -0700125 // Record patch information for later fix up
Brian Carlstromf5822582012-03-19 22:34:31 -0700126 void AddCodePatch(DexCache* dex_cache,
127 const DexFile* dex_file,
128 uint32_t referrer_method_idx,
129 uint32_t referrer_access_flags,
130 uint32_t target_method_idx,
131 bool target_is_direct,
Ian Rogers3fa13792012-03-18 15:53:45 -0700132 size_t literal_offset);
Brian Carlstromf5822582012-03-19 22:34:31 -0700133 void AddMethodPatch(DexCache* dex_cache,
134 const DexFile* dex_file,
135 uint32_t referrer_method_idx,
136 uint32_t referrer_access_flags,
137 uint32_t target_method_idx,
138 bool target_is_direct,
Ian Rogers3fa13792012-03-18 15:53:45 -0700139 size_t literal_offset);
140
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800141#if defined(ART_USE_LLVM_COMPILER)
Logan Chien8b977d32012-02-21 19:14:55 +0800142 void SetBitcodeFileName(std::string const& filename);
Logan Chienf7015fd2012-03-18 01:19:37 +0800143
144 void EnableAutoElfLoading();
145
146 const void* GetMethodCodeAddr(const CompiledMethod* cm,
147 const Method* method) const;
148
149 const Method::InvokeStub* GetMethodInvokeStubAddr(const CompiledInvokeStub* cm,
150 const Method* method) const;
Logan Chiendf576142012-03-20 17:36:32 +0800151
152 std::vector<ElfImage> GetElfImages() const;
Logan Chienf7015fd2012-03-18 01:19:37 +0800153#else
154 void EnableAutoElfLoader() { }
155
156 const void* GetMethodCodeAddr(const CompiledMethod*,
157 const Method*) const {
158 return NULL;
159 }
160
161 const Method::InvokeStub* GetMethodInvokeStubAddr(const CompiledInvokeStub*,
162 const Method*) const {
163 return NULL;
164 }
Logan Chiendf576142012-03-20 17:36:32 +0800165
166 std::vector<ElfImage> GetElfImages() const {
167 return std::vector<ElfImage>();
168 }
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800169#endif
170
Elliott Hughes6f4976c2012-03-13 21:19:01 -0700171 void SetCompilerContext(void* compiler_context) {
172 compiler_context_ = compiler_context;
173 }
174
175 void* GetCompilerContext() const {
176 return compiler_context_;
177 }
178
Logan Chien08e1ba32012-05-08 15:08:51 +0800179 size_t GetThreadCount() const {
180 return thread_count_;
181 }
182
Brian Carlstromf5822582012-03-19 22:34:31 -0700183 class PatchInformation {
184 public:
185 DexCache* GetDexCache() const {
186 return dex_cache_;
187 }
188 const DexFile& GetDexFile() const {
189 return *dex_file_;
190 }
191 uint32_t GetReferrerMethodIdx() const {
192 return referrer_method_idx_;
193 }
194 bool GetReferrerIsDirect() const {
195 return referrer_is_direct_;
196 }
197 uint32_t GetTargetMethodIdx() const {
198 return target_method_idx_;
199 }
200 bool GetTargetIsDirect() const {
201 return target_is_direct_;
202 }
203 size_t GetLiteralOffset() const {;
204 return literal_offset_;
205 }
206
207 private:
208 PatchInformation(DexCache* dex_cache,
209 const DexFile* dex_file,
210 uint32_t referrer_method_idx,
211 uint32_t referrer_access_flags,
212 uint32_t target_method_idx,
213 uint32_t target_is_direct,
214 size_t literal_offset)
215 : dex_cache_(dex_cache),
216 dex_file_(dex_file),
217 referrer_method_idx_(referrer_method_idx),
218 referrer_is_direct_(Method::IsDirect(referrer_access_flags)),
219 target_method_idx_(target_method_idx),
220 target_is_direct_(target_is_direct),
221 literal_offset_(literal_offset) {
222 CHECK(dex_file_ != NULL);
223 }
224
225 DexCache* dex_cache_;
226 const DexFile* dex_file_;
227 uint32_t referrer_method_idx_;
228 bool referrer_is_direct_;
229 uint32_t target_method_idx_;
230 bool target_is_direct_;
231 size_t literal_offset_;
232
233 friend class Compiler;
Brian Carlstrom6c403e12012-03-20 13:22:55 -0700234 DISALLOW_COPY_AND_ASSIGN(PatchInformation);
Brian Carlstromf5822582012-03-19 22:34:31 -0700235 };
236
237 const std::vector<const PatchInformation*>& GetCodeToPatch() const {
238 return code_to_patch_;
239 }
240 const std::vector<const PatchInformation*>& GetMethodsToPatch() const {
241 return methods_to_patch_;
242 }
243
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700244 private:
Brian Carlstromae826982011-11-09 01:33:42 -0800245
Ian Rogers2ed3b952012-03-17 11:49:39 -0700246 // Compute constant code and method pointers when possible
Elliott Hughesb25c3f62012-03-26 16:35:06 -0700247 void GetCodeAndMethodForDirectCall(InvokeType type, InvokeType sharp_type, Method* method,
Ian Rogers2ed3b952012-03-17 11:49:39 -0700248 uintptr_t& direct_code, uintptr_t& direct_method);
249
Brian Carlstromae826982011-11-09 01:33:42 -0800250 // Checks if class specified by type_idx is one of the image_classes_
251 bool IsImageClass(const std::string& descriptor) const;
252
Elliott Hughesd9c67be2012-02-02 19:54:06 -0800253 void PreCompile(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files, TimingLogger& timings);
Brian Carlstromae826982011-11-09 01:33:42 -0800254 void PostCompile(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files);
255
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700256 // Attempt to resolve all type, methods, fields, and strings
257 // referenced from code in the dex file following PathClassLoader
258 // ordering semantics.
Elliott Hughesd9c67be2012-02-02 19:54:06 -0800259 void Resolve(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files, TimingLogger& timings);
260 void ResolveDexFile(const ClassLoader* class_loader, const DexFile& dex_file, TimingLogger& timings);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700261
Brian Carlstromae826982011-11-09 01:33:42 -0800262 void Verify(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files);
jeffhao98eacac2011-09-14 16:11:53 -0700263 void VerifyDexFile(const ClassLoader* class_loader, const DexFile& dex_file);
264
Brian Carlstromae826982011-11-09 01:33:42 -0800265 void InitializeClassesWithoutClinit(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files);
Brian Carlstroma5a97a22011-09-15 14:08:49 -0700266 void InitializeClassesWithoutClinit(const ClassLoader* class_loader, const DexFile& dex_file);
267
Brian Carlstromae826982011-11-09 01:33:42 -0800268 void Compile(const ClassLoader* class_loader,
269 const std::vector<const DexFile*>& dex_files);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700270 void CompileDexFile(const ClassLoader* class_loader, const DexFile& dex_file);
Ian Rogers0571d352011-11-03 19:51:38 -0700271 void CompileClass(const DexFile::ClassDef& class_def, const ClassLoader* class_loader,
272 const DexFile& dex_file);
Ian Rogersa3760aa2011-11-14 14:32:37 -0800273 void CompileMethod(const DexFile::CodeItem* code_item, uint32_t access_flags, uint32_t method_idx,
274 const ClassLoader* class_loader, const DexFile& dex_file);
Brian Carlstrom83db7722011-08-26 17:32:56 -0700275
Elliott Hughesc225caa2012-02-03 15:43:37 -0800276 static void CompileClass(Context* context, size_t class_def_index);
277
Brian Carlstrome7d856b2012-01-11 18:10:55 -0800278 void SetGcMaps(const ClassLoader* class_loader, const std::vector<const DexFile*>& dex_files);
279 void SetGcMapsDexFile(const ClassLoader* class_loader, const DexFile& dex_file);
280 void SetGcMapsMethod(const DexFile& dex_file, Method* method);
281
Elliott Hughesa0e18062012-04-13 15:59:59 -0700282 void InsertInvokeStub(const std::string& key, const CompiledInvokeStub* compiled_invoke_stub);
Ian Rogers0571d352011-11-03 19:51:38 -0700283
Brian Carlstromf5822582012-03-19 22:34:31 -0700284 std::vector<const PatchInformation*> code_to_patch_;
285 std::vector<const PatchInformation*> methods_to_patch_;
Ian Rogers3fa13792012-03-18 15:53:45 -0700286
Ian Rogers2c8f6532011-09-02 17:16:34 -0700287 InstructionSet instruction_set_;
Ian Rogers2c8f6532011-09-02 17:16:34 -0700288
Elliott Hughesa0e18062012-04-13 15:59:59 -0700289 typedef SafeMap<const ClassReference, CompiledClass*> ClassTable;
Brian Carlstrom0755ec52012-01-11 15:19:46 -0800290 // All class references that this compiler has compiled
Elliott Hughesc225caa2012-02-03 15:43:37 -0800291 mutable Mutex compiled_classes_lock_;
Brian Carlstrom0755ec52012-01-11 15:19:46 -0800292 ClassTable compiled_classes_;
293
Elliott Hughesa0e18062012-04-13 15:59:59 -0700294 typedef SafeMap<const MethodReference, CompiledMethod*> MethodTable;
Ian Rogers0571d352011-11-03 19:51:38 -0700295 // All method references that this compiler has compiled
Elliott Hughesc225caa2012-02-03 15:43:37 -0800296 mutable Mutex compiled_methods_lock_;
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700297 MethodTable compiled_methods_;
298
Elliott Hughesa0e18062012-04-13 15:59:59 -0700299 typedef SafeMap<std::string, const CompiledInvokeStub*> InvokeStubTable;
Ian Rogers0571d352011-11-03 19:51:38 -0700300 // Invocation stubs created to allow invocation of the compiled methods
Elliott Hughesc225caa2012-02-03 15:43:37 -0800301 mutable Mutex compiled_invoke_stubs_lock_;
Brian Carlstrom3320cf42011-10-04 14:58:28 -0700302 InvokeStubTable compiled_invoke_stubs_;
303
Brian Carlstromaded5f72011-10-07 17:15:04 -0700304 bool image_;
Elliott Hughes5523ee02012-02-03 18:18:34 -0800305 size_t thread_count_;
Elliott Hughesde6e4cf2012-02-27 14:46:06 -0800306 bool support_debugging_;
Elliott Hughesbb551fa2012-01-25 16:35:29 -0800307 uint64_t start_ns_;
308
Ian Rogersc8b306f2012-02-17 21:34:44 -0800309 UniquePtr<AOTCompilationStats> stats_;
Ian Rogers996cc582012-02-14 22:23:29 -0800310
Brian Carlstromba0668e2012-03-26 13:14:07 -0700311 bool dump_stats_;
312 bool dump_timings_;
313
Brian Carlstromae826982011-11-09 01:33:42 -0800314 const std::set<std::string>* image_classes_;
315
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800316#if defined(ART_USE_LLVM_COMPILER)
Shih-wei Liaoc4c98812012-03-10 21:55:51 -0800317 typedef void (*CompilerCallbackFn)(Compiler& compiler);
318 typedef MutexLock* (*CompilerMutexLockFn)(Compiler& compiler);
319#endif
Logan Chienf7015fd2012-03-18 01:19:37 +0800320
Elliott Hughesb3bd5f02012-03-08 21:05:27 -0800321 void* compiler_library_;
Elliott Hughes46f060a2012-03-09 17:36:50 -0800322
Elliott Hughesb3bd5f02012-03-08 21:05:27 -0800323 typedef CompiledMethod* (*CompilerFn)(Compiler& compiler,
324 const DexFile::CodeItem* code_item,
325 uint32_t access_flags, uint32_t method_idx,
326 const ClassLoader* class_loader,
327 const DexFile& dex_file);
328 CompilerFn compiler_;
Elliott Hughes46f060a2012-03-09 17:36:50 -0800329
Elliott Hughes6f4976c2012-03-13 21:19:01 -0700330 void* compiler_context_;
331
Elliott Hughes46f060a2012-03-09 17:36:50 -0800332 typedef CompiledMethod* (*JniCompilerFn)(Compiler& compiler,
333 uint32_t access_flags, uint32_t method_idx,
Elliott Hughes46f060a2012-03-09 17:36:50 -0800334 const DexFile& dex_file);
335 JniCompilerFn jni_compiler_;
Shih-wei Liaoc4c98812012-03-10 21:55:51 -0800336 typedef CompiledInvokeStub* (*CreateInvokeStubFn)(Compiler& compiler, bool is_static,
337 const char* shorty, uint32_t shorty_len);
Shih-wei Liaoc4c98812012-03-10 21:55:51 -0800338 CreateInvokeStubFn create_invoke_stub_;
Shih-wei Liaod1fec812012-02-13 09:51:10 -0800339
Logan Chienf7015fd2012-03-18 01:19:37 +0800340#if defined(ART_USE_LLVM_COMPILER)
341 typedef void (*CompilerEnableAutoElfLoadingFn)(Compiler& compiler);
342 CompilerEnableAutoElfLoadingFn compiler_enable_auto_elf_loading_;
343
344 typedef const void* (*CompilerGetMethodCodeAddrFn)
345 (const Compiler& compiler, const CompiledMethod* cm, const Method* method);
346 CompilerGetMethodCodeAddrFn compiler_get_method_code_addr_;
347
348 typedef const Method::InvokeStub* (*CompilerGetMethodInvokeStubAddrFn)
349 (const Compiler& compiler, const CompiledInvokeStub* cm, const Method* method);
350 CompilerGetMethodInvokeStubAddrFn compiler_get_method_invoke_stub_addr_;
351#endif
352
353
Ian Rogers2c8f6532011-09-02 17:16:34 -0700354 DISALLOW_COPY_AND_ASSIGN(Compiler);
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700355};
356
Elliott Hughes8add92d2012-01-18 18:18:43 -0800357inline bool operator<(const Compiler::ClassReference& lhs, const Compiler::ClassReference& rhs) {
358 if (lhs.second < rhs.second) {
359 return true;
360 } else if (lhs.second > rhs.second) {
361 return false;
362 } else {
363 return (lhs.first < rhs.first);
364 }
365}
366
Brian Carlstrom9ea1cb12011-08-24 23:18:18 -0700367} // namespace art
368
369#endif // ART_SRC_COMPILER_H_