blob: 0281eeedb97d1bcfcf9179887baa83b01ca19b20 [file] [log] [blame]
Mathieu Chartier1ca68902017-04-18 11:26:22 -07001/*
2 * Copyright (C) 2017 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
17#include "verification.h"
18
19#include <iomanip>
20#include <sstream>
21
Andreas Gampe5e36c2f2017-04-21 19:11:15 -070022#include "art_field-inl.h"
David Sehr891a50e2017-10-27 17:01:07 -070023#include "base/file_utils.h"
Mathieu Chartier1ca68902017-04-18 11:26:22 -070024#include "mirror/class-inl.h"
Mathieu Chartier4f5e3cb2017-06-12 13:10:01 -070025#include "mirror/object-refvisitor-inl.h"
Mathieu Chartier1ca68902017-04-18 11:26:22 -070026
27namespace art {
28namespace gc {
29
Mathieu Chartierb814ef52017-06-27 12:56:00 -070030std::string Verification::DumpRAMAroundAddress(uintptr_t addr, uintptr_t bytes) const {
31 const uintptr_t dump_start = addr - bytes;
32 const uintptr_t dump_end = addr + bytes;
33 std::ostringstream oss;
34 if (dump_start < dump_end &&
35 IsAddressInHeapSpace(reinterpret_cast<const void*>(dump_start)) &&
36 IsAddressInHeapSpace(reinterpret_cast<const void*>(dump_end - 1))) {
37 oss << " adjacent_ram=";
38 for (uintptr_t p = dump_start; p < dump_end; ++p) {
39 if (p == addr) {
40 // Marker of where the address is.
41 oss << "|";
42 }
43 uint8_t* ptr = reinterpret_cast<uint8_t*>(p);
44 oss << std::hex << std::setfill('0') << std::setw(2) << static_cast<uintptr_t>(*ptr);
45 }
46 } else {
47 oss << " <invalid address>";
48 }
49 return oss.str();
50}
51
Mathieu Chartier1ca68902017-04-18 11:26:22 -070052std::string Verification::DumpObjectInfo(const void* addr, const char* tag) const {
53 std::ostringstream oss;
54 oss << tag << "=" << addr;
55 if (IsValidHeapObjectAddress(addr)) {
56 mirror::Object* obj = reinterpret_cast<mirror::Object*>(const_cast<void*>(addr));
57 mirror::Class* klass = obj->GetClass<kVerifyNone, kWithoutReadBarrier>();
58 oss << " klass=" << klass;
59 if (IsValidClass(klass)) {
60 oss << "(" << klass->PrettyClass() << ")";
Vladimir Marko98db89c2018-09-07 11:45:46 +010061 if (klass->IsArrayClass<kVerifyNone>()) {
62 oss << " length=" << obj->AsArray<kVerifyNone>()->GetLength();
Mathieu Chartier1ca68902017-04-18 11:26:22 -070063 }
64 } else {
65 oss << " <invalid address>";
66 }
67 space::Space* const space = heap_->FindSpaceFromAddress(addr);
68 if (space != nullptr) {
69 oss << " space=" << *space;
70 }
71 accounting::CardTable* card_table = heap_->GetCardTable();
72 if (card_table->AddrIsInCardTable(addr)) {
73 oss << " card=" << static_cast<size_t>(
74 card_table->GetCard(reinterpret_cast<const mirror::Object*>(addr)));
75 }
76 // Dump adjacent RAM.
Mathieu Chartierb814ef52017-06-27 12:56:00 -070077 oss << DumpRAMAroundAddress(reinterpret_cast<uintptr_t>(addr), 4 * kObjectAlignment);
Mathieu Chartier1ca68902017-04-18 11:26:22 -070078 } else {
79 oss << " <invalid address>";
80 }
81 return oss.str();
82}
83
84void Verification::LogHeapCorruption(ObjPtr<mirror::Object> holder,
85 MemberOffset offset,
86 mirror::Object* ref,
87 bool fatal) const {
88 // Lowest priority logging first:
Andreas Gampe170331f2017-12-07 18:41:03 -080089 PrintFileToLog("/proc/self/maps", android::base::LogSeverity::FATAL_WITHOUT_ABORT);
Roland Levillain680e0992018-08-24 15:41:26 +010090 MemMap::DumpMaps(LOG_STREAM(FATAL_WITHOUT_ABORT), /* terse */ true);
Mathieu Chartier3008ea42018-10-16 13:44:50 -070091 Runtime::Current()->GetHeap()->DumpSpaces(LOG_STREAM(FATAL_WITHOUT_ABORT));
Mathieu Chartier1ca68902017-04-18 11:26:22 -070092 // Buffer the output in the string stream since it is more important than the stack traces
93 // and we want it to have log priority. The stack traces are printed from Runtime::Abort
94 // which is called from LOG(FATAL) but before the abort message.
95 std::ostringstream oss;
96 oss << "GC tried to mark invalid reference " << ref << std::endl;
97 oss << DumpObjectInfo(ref, "ref") << "\n";
Mathieu Chartier4ce0c762017-05-18 10:01:07 -070098 oss << DumpObjectInfo(holder.Ptr(), "holder");
Mathieu Chartier1ca68902017-04-18 11:26:22 -070099 if (holder != nullptr) {
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700100 mirror::Class* holder_klass = holder->GetClass<kVerifyNone, kWithoutReadBarrier>();
101 if (IsValidClass(holder_klass)) {
Mathieu Chartierb814ef52017-06-27 12:56:00 -0700102 oss << " field_offset=" << offset.Uint32Value();
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700103 ArtField* field = holder->FindFieldByOffset(offset);
104 if (field != nullptr) {
105 oss << " name=" << field->GetName();
106 }
107 }
Mathieu Chartierb814ef52017-06-27 12:56:00 -0700108 mirror::HeapReference<mirror::Object>* addr = holder->GetFieldObjectReferenceAddr(offset);
109 oss << " reference addr"
110 << DumpRAMAroundAddress(reinterpret_cast<uintptr_t>(addr), 4 * kObjectAlignment);
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700111 }
112
113 if (fatal) {
114 LOG(FATAL) << oss.str();
115 } else {
116 LOG(FATAL_WITHOUT_ABORT) << oss.str();
117 }
118}
119
Mathieu Chartierb814ef52017-06-27 12:56:00 -0700120bool Verification::IsAddressInHeapSpace(const void* addr, space::Space** out_space) const {
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700121 space::Space* const space = heap_->FindSpaceFromAddress(addr);
122 if (space != nullptr) {
123 if (out_space != nullptr) {
124 *out_space = space;
125 }
126 return true;
127 }
128 return false;
129}
130
Mathieu Chartierb814ef52017-06-27 12:56:00 -0700131bool Verification::IsValidHeapObjectAddress(const void* addr, space::Space** out_space) const {
132 return IsAligned<kObjectAlignment>(addr) && IsAddressInHeapSpace(addr, out_space);
133}
134
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700135bool Verification::IsValidClass(const void* addr) const {
136 if (!IsValidHeapObjectAddress(addr)) {
137 return false;
138 }
139 mirror::Class* klass = reinterpret_cast<mirror::Class*>(const_cast<void*>(addr));
140 mirror::Class* k1 = klass->GetClass<kVerifyNone, kWithoutReadBarrier>();
141 if (!IsValidHeapObjectAddress(k1)) {
142 return false;
143 }
Roland Levillain001eff92018-01-24 14:24:33 +0000144 // `k1` should be class class, take the class again to verify.
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700145 // Note that this check may not be valid for the no image space since the class class might move
146 // around from moving GC.
147 mirror::Class* k2 = k1->GetClass<kVerifyNone, kWithoutReadBarrier>();
148 if (!IsValidHeapObjectAddress(k2)) {
149 return false;
150 }
151 return k1 == k2;
152}
153
Mathieu Chartier4f5e3cb2017-06-12 13:10:01 -0700154using ObjectSet = std::set<mirror::Object*>;
155using WorkQueue = std::deque<std::pair<mirror::Object*, std::string>>;
156
157// Use for visiting the GcRoots held live by ArtFields, ArtMethods, and ClassLoaders.
158class Verification::BFSFindReachable {
159 public:
160 explicit BFSFindReachable(ObjectSet* visited) : visited_(visited) {}
161
162 void operator()(mirror::Object* obj, MemberOffset offset, bool is_static ATTRIBUTE_UNUSED) const
163 REQUIRES_SHARED(Locks::mutator_lock_) {
164 ArtField* field = obj->FindFieldByOffset(offset);
165 Visit(obj->GetFieldObject<mirror::Object>(offset),
166 field != nullptr ? field->GetName() : "");
167 }
168
169 void VisitRootIfNonNull(mirror::CompressedReference<mirror::Object>* root) const
170 REQUIRES_SHARED(Locks::mutator_lock_) {
171 if (!root->IsNull()) {
172 VisitRoot(root);
173 }
174 }
175
176 void VisitRoot(mirror::CompressedReference<mirror::Object>* root) const
177 REQUIRES_SHARED(Locks::mutator_lock_) {
178 Visit(root->AsMirrorPtr(), "!nativeRoot");
179 }
180
181 void Visit(mirror::Object* ref, const std::string& field_name) const
182 REQUIRES_SHARED(Locks::mutator_lock_) {
183 if (ref != nullptr && visited_->insert(ref).second) {
184 new_visited_.emplace_back(ref, field_name);
185 }
186 }
187
188 const WorkQueue& NewlyVisited() const {
189 return new_visited_;
190 }
191
192 private:
193 ObjectSet* visited_;
194 mutable WorkQueue new_visited_;
195};
196
197class Verification::CollectRootVisitor : public SingleRootVisitor {
198 public:
199 CollectRootVisitor(ObjectSet* visited, WorkQueue* work) : visited_(visited), work_(work) {}
200
201 void VisitRoot(mirror::Object* obj, const RootInfo& info)
Roland Levillainbbc6e7e2018-08-24 16:58:47 +0100202 override REQUIRES_SHARED(Locks::mutator_lock_) {
Mathieu Chartier4f5e3cb2017-06-12 13:10:01 -0700203 if (obj != nullptr && visited_->insert(obj).second) {
204 std::ostringstream oss;
205 oss << info.ToString() << " = " << obj << "(" << obj->PrettyTypeOf() << ")";
206 work_->emplace_back(obj, oss.str());
207 }
208 }
209
210 private:
211 ObjectSet* const visited_;
212 WorkQueue* const work_;
213};
214
215std::string Verification::FirstPathFromRootSet(ObjPtr<mirror::Object> target) const {
216 Runtime* const runtime = Runtime::Current();
217 std::set<mirror::Object*> visited;
218 std::deque<std::pair<mirror::Object*, std::string>> work;
219 {
220 CollectRootVisitor root_visitor(&visited, &work);
221 runtime->VisitRoots(&root_visitor, kVisitRootFlagAllRoots);
222 }
223 while (!work.empty()) {
224 auto pair = work.front();
225 work.pop_front();
226 if (pair.first == target) {
227 return pair.second;
228 }
229 BFSFindReachable visitor(&visited);
230 pair.first->VisitReferences(visitor, VoidFunctor());
231 for (auto&& pair2 : visitor.NewlyVisited()) {
232 std::ostringstream oss;
233 mirror::Object* obj = pair2.first;
234 oss << pair.second << " -> " << obj << "(" << obj->PrettyTypeOf() << ")." << pair2.second;
235 work.emplace_back(obj, oss.str());
236 }
237 }
238 return "<no path found>";
239}
240
Mathieu Chartier1ca68902017-04-18 11:26:22 -0700241} // namespace gc
242} // namespace art