blob: f8061bcfd8f903d50fa4c67942bb898eab853c61 [file] [log] [blame]
Calin Juravle31f2c152015-10-23 17:56:15 +01001/*
2 * Copyright (C) 2015 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 */
16
Calin Juravle33083d62017-01-18 15:29:12 -080017#ifndef ART_RUNTIME_JIT_PROFILE_COMPILATION_INFO_H_
18#define ART_RUNTIME_JIT_PROFILE_COMPILATION_INFO_H_
Calin Juravle31f2c152015-10-23 17:56:15 +010019
20#include <set>
Calin Juravle4d77b6a2015-12-01 18:38:09 +000021#include <vector>
Calin Juravle31f2c152015-10-23 17:56:15 +010022
23#include "atomic.h"
Mathieu Chartierc5dd3192015-12-09 16:38:30 -080024#include "dex_cache_resolved_classes.h"
Calin Juravle31f2c152015-10-23 17:56:15 +010025#include "dex_file.h"
Andreas Gampea5b09a62016-11-17 15:21:22 -080026#include "dex_file_types.h"
Calin Juravle226501b2015-12-11 14:41:31 +000027#include "method_reference.h"
Calin Juravle31f2c152015-10-23 17:56:15 +010028#include "safe_map.h"
29
30namespace art {
31
Calin Juravle31f2c152015-10-23 17:56:15 +010032/**
Calin Juravle998c2162015-12-21 15:39:33 +020033 * Profile information in a format suitable to be queried by the compiler and
34 * performing profile guided compilation.
35 * It is a serialize-friendly format based on information collected by the
36 * interpreter (ProfileInfo).
Calin Juravle31f2c152015-10-23 17:56:15 +010037 * Currently it stores only the hot compiled methods.
38 */
Calin Juravle226501b2015-12-11 14:41:31 +000039class ProfileCompilationInfo {
40 public:
Calin Juravle64142952016-03-21 14:37:55 +000041 static const uint8_t kProfileMagic[];
42 static const uint8_t kProfileVersion[];
43
Calin Juravle67265462016-03-18 16:23:40 +000044 // Add the given methods and classes to the current profile object.
Calin Juravle99629622016-04-19 16:33:46 +010045 bool AddMethodsAndClasses(const std::vector<MethodReference>& methods,
Calin Juravle67265462016-03-18 16:23:40 +000046 const std::set<DexCacheResolvedClasses>& resolved_classes);
Calin Juravle877fd962016-01-05 14:29:29 +000047 // Loads profile information from the given file descriptor.
Calin Juravle2e2db782016-02-23 12:00:03 +000048 bool Load(int fd);
Calin Juravle67265462016-03-18 16:23:40 +000049 // Merge the data from another ProfileCompilationInfo into the current object.
50 bool MergeWith(const ProfileCompilationInfo& info);
Calin Juravle877fd962016-01-05 14:29:29 +000051 // Saves the profile data to the given file descriptor.
Calin Juravle2e2db782016-02-23 12:00:03 +000052 bool Save(int fd);
Calin Juravle67265462016-03-18 16:23:40 +000053 // Loads and merges profile information from the given file into the current
54 // object and tries to save it back to disk.
Calin Juravle5d1bd0a2016-03-24 20:33:22 +000055 // If `force` is true then the save will go through even if the given file
56 // has bad data or its version does not match. In this cases the profile content
57 // is ignored.
58 bool MergeAndSave(const std::string& filename, uint64_t* bytes_written, bool force);
Calin Juravle67265462016-03-18 16:23:40 +000059
Calin Juravle998c2162015-12-21 15:39:33 +020060 // Returns the number of methods that were profiled.
61 uint32_t GetNumberOfMethods() const;
Calin Juravle67265462016-03-18 16:23:40 +000062 // Returns the number of resolved classes that were profiled.
63 uint32_t GetNumberOfResolvedClasses() const;
Calin Juravle226501b2015-12-11 14:41:31 +000064
65 // Returns true if the method reference is present in the profiling info.
66 bool ContainsMethod(const MethodReference& method_ref) const;
67
Jeff Hao54b58552016-11-16 15:15:04 -080068 // Returns true if the class's type is present in the profiling info.
Andreas Gampea5b09a62016-11-17 15:21:22 -080069 bool ContainsClass(const DexFile& dex_file, dex::TypeIndex type_idx) const;
Mathieu Chartiera8077802016-03-16 19:08:31 -070070
Calin Juravle226501b2015-12-11 14:41:31 +000071 // Dumps all the loaded profile info into a string and returns it.
Calin Juravle998c2162015-12-21 15:39:33 +020072 // If dex_files is not null then the method indices will be resolved to their
73 // names.
Calin Juravle226501b2015-12-11 14:41:31 +000074 // This is intended for testing and debugging.
Calin Juravle998c2162015-12-21 15:39:33 +020075 std::string DumpInfo(const std::vector<const DexFile*>* dex_files,
76 bool print_full_dex_location = true) const;
Calin Juravle226501b2015-12-11 14:41:31 +000077
Calin Juravle2e2db782016-02-23 12:00:03 +000078 bool Equals(const ProfileCompilationInfo& other);
Calin Juravle67265462016-03-18 16:23:40 +000079
Calin Juravle31708b72016-02-05 19:44:05 +000080 static std::string GetProfileDexFileKey(const std::string& dex_location);
Calin Juravle877fd962016-01-05 14:29:29 +000081
Mathieu Chartierc5dd3192015-12-09 16:38:30 -080082 // Returns the class descriptors for all of the classes in the profiles' class sets.
83 // Note the dex location is actually the profile key, the caller needs to call back in to the
84 // profile info stuff to generate a map back to the dex location.
85 std::set<DexCacheResolvedClasses> GetResolvedClasses() const;
Calin Juravle226501b2015-12-11 14:41:31 +000086
Calin Juravle67265462016-03-18 16:23:40 +000087 // Clears the resolved classes from the current object.
88 void ClearResolvedClasses();
89
Calin Juravle7bcdb532016-06-07 16:14:47 +010090 static bool GenerateTestProfile(int fd,
91 uint16_t number_of_dex_files,
92 uint16_t method_ratio,
93 uint16_t class_ratio);
94
Mathieu Chartierc5dd3192015-12-09 16:38:30 -080095 private:
Calin Juravle64142952016-03-21 14:37:55 +000096 enum ProfileLoadSatus {
97 kProfileLoadIOError,
98 kProfileLoadVersionMismatch,
99 kProfileLoadBadData,
100 kProfileLoadSuccess
101 };
102
Calin Juravle998c2162015-12-21 15:39:33 +0200103 struct DexFileData {
104 explicit DexFileData(uint32_t location_checksum) : checksum(location_checksum) {}
105 uint32_t checksum;
106 std::set<uint16_t> method_set;
Andreas Gampea5b09a62016-11-17 15:21:22 -0800107 std::set<dex::TypeIndex> class_set;
Calin Juravle877fd962016-01-05 14:29:29 +0000108
109 bool operator==(const DexFileData& other) const {
110 return checksum == other.checksum && method_set == other.method_set;
111 }
Calin Juravle998c2162015-12-21 15:39:33 +0200112 };
Calin Juravle226501b2015-12-11 14:41:31 +0000113
Calin Juravle998c2162015-12-21 15:39:33 +0200114 using DexFileToProfileInfoMap = SafeMap<const std::string, DexFileData>;
115
Mathieu Chartierc5dd3192015-12-09 16:38:30 -0800116 DexFileData* GetOrAddDexFileData(const std::string& dex_location, uint32_t checksum);
117 bool AddMethodIndex(const std::string& dex_location, uint32_t checksum, uint16_t method_idx);
Andreas Gampea5b09a62016-11-17 15:21:22 -0800118 bool AddClassIndex(const std::string& dex_location, uint32_t checksum, dex::TypeIndex type_idx);
Calin Juravle99629622016-04-19 16:33:46 +0100119 bool AddResolvedClasses(const DexCacheResolvedClasses& classes);
Calin Juravle64142952016-03-21 14:37:55 +0000120
121 // Parsing functionality.
122
123 struct ProfileLineHeader {
124 std::string dex_location;
125 uint16_t method_set_size;
126 uint16_t class_set_size;
127 uint32_t checksum;
128 };
129
130 // A helper structure to make sure we don't read past our buffers in the loops.
131 struct SafeBuffer {
132 public:
133 explicit SafeBuffer(size_t size) : storage_(new uint8_t[size]) {
134 ptr_current_ = storage_.get();
135 ptr_end_ = ptr_current_ + size;
136 }
137
138 // Reads the content of the descriptor at the current position.
139 ProfileLoadSatus FillFromFd(int fd,
140 const std::string& source,
141 /*out*/std::string* error);
142
143 // Reads an uint value (high bits to low bits) and advances the current pointer
144 // with the number of bits read.
145 template <typename T> T ReadUintAndAdvance();
146
147 // Compares the given data with the content current pointer. If the contents are
148 // equal it advances the current pointer by data_size.
149 bool CompareAndAdvance(const uint8_t* data, size_t data_size);
150
151 // Get the underlying raw buffer.
152 uint8_t* Get() { return storage_.get(); }
153
154 private:
Andreas Gampe7b8a2652016-11-11 17:11:25 -0800155 std::unique_ptr<uint8_t[]> storage_;
Calin Juravle64142952016-03-21 14:37:55 +0000156 uint8_t* ptr_current_;
157 uint8_t* ptr_end_;
158 };
159
160 ProfileLoadSatus LoadInternal(int fd, std::string* error);
161
162 ProfileLoadSatus ReadProfileHeader(int fd,
163 /*out*/uint16_t* number_of_lines,
164 /*out*/std::string* error);
165
166 ProfileLoadSatus ReadProfileLineHeader(int fd,
167 /*out*/ProfileLineHeader* line_header,
168 /*out*/std::string* error);
169 ProfileLoadSatus ReadProfileLine(int fd,
170 const ProfileLineHeader& line_header,
171 /*out*/std::string* error);
172
173 bool ProcessLine(SafeBuffer& line_buffer,
174 uint16_t method_set_size,
175 uint16_t class_set_size,
176 uint32_t checksum,
177 const std::string& dex_location);
Mathieu Chartierc5dd3192015-12-09 16:38:30 -0800178
Calin Juravle877fd962016-01-05 14:29:29 +0000179 friend class ProfileCompilationInfoTest;
180 friend class CompilerDriverProfileTest;
181 friend class ProfileAssistantTest;
Jeff Hao41fba6a2016-11-28 11:53:33 -0800182 friend class Dex2oatLayoutTest;
Calin Juravle877fd962016-01-05 14:29:29 +0000183
Calin Juravle226501b2015-12-11 14:41:31 +0000184 DexFileToProfileInfoMap info_;
185};
186
Calin Juravle31f2c152015-10-23 17:56:15 +0100187} // namespace art
188
Calin Juravle33083d62017-01-18 15:29:12 -0800189#endif // ART_RUNTIME_JIT_PROFILE_COMPILATION_INFO_H_