blob: f6a73a866083228022519701a236e5247735996b [file] [log] [blame]
Elliott Hughes9d5ccec2011-09-19 13:19:50 -07001/*
2 * Copyright (C) 2008 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
Andreas Gampe277ccbd2014-11-03 21:36:10 -080017#include "dalvik_system_VMDebug.h"
18
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070019#include <string.h>
20#include <unistd.h>
21
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -070022#include <sstream>
23
24#include "base/histogram-inl.h"
Vladimir Marko80afd022015-05-19 18:08:00 +010025#include "base/time_utils.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070026#include "class_linker.h"
Ian Rogers62d6c772013-02-27 08:32:07 -080027#include "common_throws.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070028#include "debugger.h"
Mathieu Chartier7410f292013-11-24 13:17:35 -080029#include "gc/space/bump_pointer_space.h"
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -070030#include "gc/space/dlmalloc_space.h"
31#include "gc/space/large_object_space.h"
32#include "gc/space/space-inl.h"
Mathieu Chartier1f3b5352014-02-03 14:00:42 -080033#include "gc/space/zygote_space.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070034#include "hprof/hprof.h"
35#include "jni_internal.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080036#include "mirror/class.h"
Ian Rogersdd157d72014-05-15 14:47:50 -070037#include "ScopedLocalRef.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070038#include "ScopedUtfChars.h"
Mathieu Chartier0795f232016-09-27 18:43:30 -070039#include "scoped_fast_native_object_access-inl.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070040#include "trace.h"
Ian Rogersdd157d72014-05-15 14:47:50 -070041#include "well_known_classes.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070042
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070043namespace art {
44
Elliott Hughes0512f022012-03-15 22:10:52 -070045static jobjectArray VMDebug_getVmFeatureList(JNIEnv* env, jclass) {
Ian Rogersdd157d72014-05-15 14:47:50 -070046 static const char* features[] = {
47 "method-trace-profiling",
48 "method-trace-profiling-streaming",
49 "method-sample-profiling",
50 "hprof-heap-dump",
51 "hprof-heap-dump-streaming",
52 };
53 jobjectArray result = env->NewObjectArray(arraysize(features),
54 WellKnownClasses::java_lang_String,
55 nullptr);
56 if (result != nullptr) {
57 for (size_t i = 0; i < arraysize(features); ++i) {
58 ScopedLocalRef<jstring> jfeature(env, env->NewStringUTF(features[i]));
59 if (jfeature.get() == nullptr) {
60 return nullptr;
61 }
62 env->SetObjectArrayElement(result, i, jfeature.get());
63 }
64 }
65 return result;
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070066}
67
Elliott Hughes0512f022012-03-15 22:10:52 -070068static void VMDebug_startAllocCounting(JNIEnv*, jclass) {
Mathieu Chartier9ef78b52014-09-25 17:03:12 -070069 Runtime::Current()->SetStatsEnabled(true);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070070}
71
Elliott Hughes0512f022012-03-15 22:10:52 -070072static void VMDebug_stopAllocCounting(JNIEnv*, jclass) {
Mathieu Chartier9ef78b52014-09-25 17:03:12 -070073 Runtime::Current()->SetStatsEnabled(false);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070074}
75
Elliott Hughes1bac54f2012-03-16 12:48:31 -070076static jint VMDebug_getAllocCount(JNIEnv*, jclass, jint kind) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070077 return Runtime::Current()->GetStat(kind);
78}
79
Elliott Hughes0512f022012-03-15 22:10:52 -070080static void VMDebug_resetAllocCount(JNIEnv*, jclass, jint kinds) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070081 Runtime::Current()->ResetStats(kinds);
82}
83
Jeff Hao23009dc2013-08-22 15:36:42 -070084static void VMDebug_startMethodTracingDdmsImpl(JNIEnv*, jclass, jint bufferSize, jint flags,
85 jboolean samplingEnabled, jint intervalUs) {
Andreas Gampe7e7e0f42015-03-29 15:26:23 -070086 Trace::Start("[DDMS]", -1, bufferSize, flags, Trace::TraceOutputMode::kDDMS,
87 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
88 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070089}
90
Ian Rogers00f7d0e2012-07-19 15:28:27 -070091static void VMDebug_startMethodTracingFd(JNIEnv* env, jclass, jstring javaTraceFilename,
Jeff Hao4044bda2014-01-06 15:50:45 -080092 jobject javaFd, jint bufferSize, jint flags,
Shukang Zhouca6e14e2017-01-25 00:35:21 -080093 jboolean samplingEnabled, jint intervalUs,
94 jboolean streamingOutput) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070095 int originalFd = jniGetFDFromFileDescriptor(env, javaFd);
96 if (originalFd < 0) {
97 return;
98 }
99
100 int fd = dup(originalFd);
101 if (fd < 0) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700102 ScopedObjectAccess soa(env);
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +0000103 soa.Self()->ThrowNewExceptionF("Ljava/lang/RuntimeException;",
Ian Rogers62d6c772013-02-27 08:32:07 -0800104 "dup(%d) failed: %s", originalFd, strerror(errno));
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700105 return;
106 }
107
108 ScopedUtfChars traceFilename(env, javaTraceFilename);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700109 if (traceFilename.c_str() == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700110 return;
111 }
Shukang Zhouca6e14e2017-01-25 00:35:21 -0800112 Trace::TraceOutputMode outputMode = streamingOutput
113 ? Trace::TraceOutputMode::kStreaming
114 : Trace::TraceOutputMode::kFile;
115 Trace::Start(traceFilename.c_str(), fd, bufferSize, flags, outputMode,
Andreas Gampe7e7e0f42015-03-29 15:26:23 -0700116 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
117 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700118}
119
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700120static void VMDebug_startMethodTracingFilename(JNIEnv* env, jclass, jstring javaTraceFilename,
Jeff Hao4044bda2014-01-06 15:50:45 -0800121 jint bufferSize, jint flags,
122 jboolean samplingEnabled, jint intervalUs) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700123 ScopedUtfChars traceFilename(env, javaTraceFilename);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700124 if (traceFilename.c_str() == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700125 return;
126 }
Andreas Gampe7e7e0f42015-03-29 15:26:23 -0700127 Trace::Start(traceFilename.c_str(), -1, bufferSize, flags, Trace::TraceOutputMode::kFile,
128 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
129 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700130}
131
Jeff Hao64caa7d2013-08-29 11:18:01 -0700132static jint VMDebug_getMethodTracingMode(JNIEnv*, jclass) {
133 return Trace::GetMethodTracingMode();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700134}
135
Elliott Hughes0512f022012-03-15 22:10:52 -0700136static void VMDebug_stopMethodTracing(JNIEnv*, jclass) {
jeffhaoe343b762011-12-05 16:36:44 -0800137 Trace::Stop();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700138}
139
Elliott Hughes0512f022012-03-15 22:10:52 -0700140static void VMDebug_startEmulatorTracing(JNIEnv*, jclass) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700141 UNIMPLEMENTED(WARNING);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700142 // dvmEmulatorTraceStart();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700143}
144
Elliott Hughes0512f022012-03-15 22:10:52 -0700145static void VMDebug_stopEmulatorTracing(JNIEnv*, jclass) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700146 UNIMPLEMENTED(WARNING);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700147 // dvmEmulatorTraceStop();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700148}
149
Elliott Hughes0512f022012-03-15 22:10:52 -0700150static jboolean VMDebug_isDebuggerConnected(JNIEnv*, jclass) {
Elliott Hughesc0f09332012-03-26 13:27:06 -0700151 return Dbg::IsDebuggerActive();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700152}
153
Elliott Hughes0512f022012-03-15 22:10:52 -0700154static jboolean VMDebug_isDebuggingEnabled(JNIEnv*, jclass) {
Elliott Hughesc0f09332012-03-26 13:27:06 -0700155 return Dbg::IsJdwpConfigured();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700156}
157
Elliott Hughes0512f022012-03-15 22:10:52 -0700158static jlong VMDebug_lastDebuggerActivity(JNIEnv*, jclass) {
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700159 return Dbg::LastDebuggerActivity();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700160}
161
Ian Rogers62d6c772013-02-27 08:32:07 -0800162static void ThrowUnsupportedOperationException(JNIEnv* env) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700163 ScopedObjectAccess soa(env);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700164 soa.Self()->ThrowNewException("Ljava/lang/UnsupportedOperationException;", nullptr);
Ian Rogers62d6c772013-02-27 08:32:07 -0800165}
166
167static void VMDebug_startInstructionCounting(JNIEnv* env, jclass) {
168 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700169}
170
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700171static void VMDebug_stopInstructionCounting(JNIEnv* env, jclass) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800172 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700173}
174
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700175static void VMDebug_getInstructionCount(JNIEnv* env, jclass, jintArray /*javaCounts*/) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800176 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700177}
178
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700179static void VMDebug_resetInstructionCount(JNIEnv* env, jclass) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800180 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700181}
182
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700183static void VMDebug_printLoadedClasses(JNIEnv* env, jclass, jint flags) {
Andreas Gampecc1b5352016-12-01 16:58:38 -0800184 class DumpClassVisitor : public ClassVisitor {
185 public:
186 explicit DumpClassVisitor(int dump_flags) : flags_(dump_flags) {}
187
188 bool operator()(ObjPtr<mirror::Class> klass) OVERRIDE REQUIRES_SHARED(Locks::mutator_lock_) {
189 klass->DumpClass(LOG_STREAM(ERROR), flags_);
190 return true;
191 }
192
193 private:
194 const int flags_;
195 };
196 DumpClassVisitor visitor(flags);
197
Ian Rogers53b8b092014-03-13 23:45:53 -0700198 ScopedFastNativeObjectAccess soa(env);
Andreas Gampecc1b5352016-12-01 16:58:38 -0800199 return Runtime::Current()->GetClassLinker()->VisitClasses(&visitor);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700200}
201
Ian Rogers7dfb28c2013-08-22 08:18:36 -0700202static jint VMDebug_getLoadedClassCount(JNIEnv* env, jclass) {
Ian Rogers53b8b092014-03-13 23:45:53 -0700203 ScopedFastNativeObjectAccess soa(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700204 return Runtime::Current()->GetClassLinker()->NumLoadedClasses();
205}
206
207/*
208 * Returns the thread-specific CPU-time clock value for the current thread,
209 * or -1 if the feature isn't supported.
210 */
Elliott Hughes0512f022012-03-15 22:10:52 -0700211static jlong VMDebug_threadCpuTimeNanos(JNIEnv*, jclass) {
212 return ThreadCpuNanoTime();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700213}
214
215/*
216 * static void dumpHprofData(String fileName, FileDescriptor fd)
217 *
218 * Cause "hprof" data to be dumped. We can throw an IOException if an
219 * error occurs during file handling.
220 */
Elliott Hughes0512f022012-03-15 22:10:52 -0700221static void VMDebug_dumpHprofData(JNIEnv* env, jclass, jstring javaFilename, jobject javaFd) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700222 // Only one of these may be null.
223 if (javaFilename == nullptr && javaFd == nullptr) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700224 ScopedObjectAccess soa(env);
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +0000225 ThrowNullPointerException("fileName == null && fd == null");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700226 return;
227 }
228
229 std::string filename;
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700230 if (javaFilename != nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700231 ScopedUtfChars chars(env, javaFilename);
232 if (env->ExceptionCheck()) {
233 return;
234 }
235 filename = chars.c_str();
236 } else {
237 filename = "[fd]";
238 }
239
240 int fd = -1;
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700241 if (javaFd != nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700242 fd = jniGetFDFromFileDescriptor(env, javaFd);
243 if (fd < 0) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700244 ScopedObjectAccess soa(env);
Ian Rogers62d6c772013-02-27 08:32:07 -0800245 ThrowRuntimeException("Invalid file descriptor");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700246 return;
247 }
248 }
249
Elliott Hughes622a6982012-06-08 17:58:54 -0700250 hprof::DumpHeap(filename.c_str(), fd, false);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700251}
252
Elliott Hugheseac76672012-05-24 21:56:51 -0700253static void VMDebug_dumpHprofDataDdms(JNIEnv*, jclass) {
Elliott Hughes622a6982012-06-08 17:58:54 -0700254 hprof::DumpHeap("[DDMS]", -1, true);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700255}
256
Elliott Hughes0512f022012-03-15 22:10:52 -0700257static void VMDebug_dumpReferenceTables(JNIEnv* env, jclass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700258 ScopedObjectAccess soa(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700259 LOG(INFO) << "--- reference table dump ---";
260
Andreas Gampe3fec9ac2016-09-13 10:47:28 -0700261 soa.Env()->DumpReferenceTables(LOG_STREAM(INFO));
262 soa.Vm()->DumpReferenceTables(LOG_STREAM(INFO));
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700263
264 LOG(INFO) << "---";
265}
266
Elliott Hughes0512f022012-03-15 22:10:52 -0700267static void VMDebug_crash(JNIEnv*, jclass) {
Elliott Hughes81ff3182012-03-23 20:35:56 -0700268 LOG(FATAL) << "Crashing runtime on request";
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700269}
270
Elliott Hughes0512f022012-03-15 22:10:52 -0700271static void VMDebug_infopoint(JNIEnv*, jclass, jint id) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700272 LOG(INFO) << "VMDebug infopoint " << id << " hit";
273}
274
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700275static jlong VMDebug_countInstancesOfClass(JNIEnv* env,
276 jclass,
277 jclass javaClass,
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700278 jboolean countAssignable) {
279 ScopedObjectAccess soa(env);
Mathieu Chartierf1820852015-07-10 13:19:51 -0700280 gc::Heap* const heap = Runtime::Current()->GetHeap();
281 // Caller's responsibility to do GC if desired.
Mathieu Chartier0795f232016-09-27 18:43:30 -0700282 ObjPtr<mirror::Class> c = soa.Decode<mirror::Class>(javaClass);
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800283 if (c == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700284 return 0;
285 }
Mathieu Chartiere8a3c572016-10-11 16:52:17 -0700286 VariableSizedHandleScope hs(soa.Self());
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700287 std::vector<Handle<mirror::Class>> classes {hs.NewHandle(c)};
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800288 uint64_t count = 0;
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800289 heap->CountInstances(classes, countAssignable, &count);
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800290 return count;
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700291}
292
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700293static jlongArray VMDebug_countInstancesOfClasses(JNIEnv* env,
294 jclass,
295 jobjectArray javaClasses,
Mathieu Chartierf1820852015-07-10 13:19:51 -0700296 jboolean countAssignable) {
297 ScopedObjectAccess soa(env);
298 gc::Heap* const heap = Runtime::Current()->GetHeap();
299 // Caller's responsibility to do GC if desired.
Mathieu Chartier0795f232016-09-27 18:43:30 -0700300 ObjPtr<mirror::ObjectArray<mirror::Class>> decoded_classes =
301 soa.Decode<mirror::ObjectArray<mirror::Class>>(javaClasses);
Mathieu Chartierf1820852015-07-10 13:19:51 -0700302 if (decoded_classes == nullptr) {
303 return nullptr;
304 }
Mathieu Chartiere8a3c572016-10-11 16:52:17 -0700305 VariableSizedHandleScope hs(soa.Self());
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700306 std::vector<Handle<mirror::Class>> classes;
Mathieu Chartierf1820852015-07-10 13:19:51 -0700307 for (size_t i = 0, count = decoded_classes->GetLength(); i < count; ++i) {
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700308 classes.push_back(hs.NewHandle(decoded_classes->Get(i)));
Mathieu Chartierf1820852015-07-10 13:19:51 -0700309 }
310 std::vector<uint64_t> counts(classes.size(), 0u);
311 // Heap::CountInstances can handle null and will put 0 for these classes.
312 heap->CountInstances(classes, countAssignable, &counts[0]);
Mathieu Chartier9d156d52016-10-06 17:44:26 -0700313 ObjPtr<mirror::LongArray> long_counts = mirror::LongArray::Alloc(soa.Self(), counts.size());
Mathieu Chartierf1820852015-07-10 13:19:51 -0700314 if (long_counts == nullptr) {
315 soa.Self()->AssertPendingOOMException();
316 return nullptr;
317 }
318 for (size_t i = 0; i < counts.size(); ++i) {
319 long_counts->Set(i, counts[i]);
320 }
321 return soa.AddLocalReference<jlongArray>(long_counts);
322}
323
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700324// We export the VM internal per-heap-space size/alloc/free metrics
325// for the zygote space, alloc space (application heap), and the large
326// object space for dumpsys meminfo. The other memory region data such
327// as PSS, private/shared dirty/shared data are available via
328// /proc/<pid>/smaps.
329static void VMDebug_getHeapSpaceStats(JNIEnv* env, jclass, jlongArray data) {
Ian Rogers6b99dd12013-07-25 12:11:32 -0700330 jlong* arr = reinterpret_cast<jlong*>(env->GetPrimitiveArrayCritical(data, 0));
Mathieu Chartier7410f292013-11-24 13:17:35 -0800331 if (arr == nullptr || env->GetArrayLength(data) < 9) {
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700332 return;
333 }
334
335 size_t allocSize = 0;
336 size_t allocUsed = 0;
337 size_t zygoteSize = 0;
338 size_t zygoteUsed = 0;
339 size_t largeObjectsSize = 0;
340 size_t largeObjectsUsed = 0;
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700341 gc::Heap* heap = Runtime::Current()->GetHeap();
Mathieu Chartiera9d82fe2016-01-25 20:06:11 -0800342 {
343 ScopedObjectAccess soa(env);
344 for (gc::space::ContinuousSpace* space : heap->GetContinuousSpaces()) {
345 if (space->IsImageSpace()) {
346 // Currently don't include the image space.
347 } else if (space->IsZygoteSpace()) {
348 gc::space::ZygoteSpace* zygote_space = space->AsZygoteSpace();
349 zygoteSize += zygote_space->Size();
350 zygoteUsed += zygote_space->GetBytesAllocated();
351 } else if (space->IsMallocSpace()) {
352 // This is a malloc space.
353 gc::space::MallocSpace* malloc_space = space->AsMallocSpace();
354 allocSize += malloc_space->GetFootprint();
355 allocUsed += malloc_space->GetBytesAllocated();
356 } else if (space->IsBumpPointerSpace()) {
357 gc::space::BumpPointerSpace* bump_pointer_space = space->AsBumpPointerSpace();
358 allocSize += bump_pointer_space->Size();
359 allocUsed += bump_pointer_space->GetBytesAllocated();
360 }
361 }
362 for (gc::space::DiscontinuousSpace* space : heap->GetDiscontinuousSpaces()) {
363 if (space->IsLargeObjectSpace()) {
364 largeObjectsSize += space->AsLargeObjectSpace()->GetBytesAllocated();
365 largeObjectsUsed += largeObjectsSize;
366 }
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700367 }
368 }
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700369 size_t allocFree = allocSize - allocUsed;
370 size_t zygoteFree = zygoteSize - zygoteUsed;
371 size_t largeObjectsFree = largeObjectsSize - largeObjectsUsed;
372
373 int j = 0;
374 arr[j++] = allocSize;
375 arr[j++] = allocUsed;
376 arr[j++] = allocFree;
377 arr[j++] = zygoteSize;
378 arr[j++] = zygoteUsed;
379 arr[j++] = zygoteFree;
380 arr[j++] = largeObjectsSize;
381 arr[j++] = largeObjectsUsed;
382 arr[j++] = largeObjectsFree;
383 env->ReleasePrimitiveArrayCritical(data, arr, 0);
384}
385
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -0700386// The runtime stat names for VMDebug.getRuntimeStat().
387enum class VMDebugRuntimeStatId {
388 kArtGcGcCount = 0,
389 kArtGcGcTime,
390 kArtGcBytesAllocated,
391 kArtGcBytesFreed,
392 kArtGcBlockingGcCount,
393 kArtGcBlockingGcTime,
394 kArtGcGcCountRateHistogram,
395 kArtGcBlockingGcCountRateHistogram,
396 kNumRuntimeStats,
397};
398
399static jobject VMDebug_getRuntimeStatInternal(JNIEnv* env, jclass, jint statId) {
400 gc::Heap* heap = Runtime::Current()->GetHeap();
401 switch (static_cast<VMDebugRuntimeStatId>(statId)) {
402 case VMDebugRuntimeStatId::kArtGcGcCount: {
403 std::string output = std::to_string(heap->GetGcCount());
404 return env->NewStringUTF(output.c_str());
405 }
406 case VMDebugRuntimeStatId::kArtGcGcTime: {
407 std::string output = std::to_string(NsToMs(heap->GetGcTime()));
408 return env->NewStringUTF(output.c_str());
409 }
410 case VMDebugRuntimeStatId::kArtGcBytesAllocated: {
411 std::string output = std::to_string(heap->GetBytesAllocatedEver());
412 return env->NewStringUTF(output.c_str());
413 }
414 case VMDebugRuntimeStatId::kArtGcBytesFreed: {
415 std::string output = std::to_string(heap->GetBytesFreedEver());
416 return env->NewStringUTF(output.c_str());
417 }
418 case VMDebugRuntimeStatId::kArtGcBlockingGcCount: {
419 std::string output = std::to_string(heap->GetBlockingGcCount());
420 return env->NewStringUTF(output.c_str());
421 }
422 case VMDebugRuntimeStatId::kArtGcBlockingGcTime: {
423 std::string output = std::to_string(NsToMs(heap->GetBlockingGcTime()));
424 return env->NewStringUTF(output.c_str());
425 }
426 case VMDebugRuntimeStatId::kArtGcGcCountRateHistogram: {
427 std::ostringstream output;
428 heap->DumpGcCountRateHistogram(output);
429 return env->NewStringUTF(output.str().c_str());
430 }
431 case VMDebugRuntimeStatId::kArtGcBlockingGcCountRateHistogram: {
432 std::ostringstream output;
433 heap->DumpBlockingGcCountRateHistogram(output);
434 return env->NewStringUTF(output.str().c_str());
435 }
436 default:
437 return nullptr;
438 }
439}
440
Andreas Gampeca620d72016-11-08 08:09:33 -0800441static bool SetRuntimeStatValue(JNIEnv* env,
442 jobjectArray result,
443 VMDebugRuntimeStatId id,
444 const std::string& value) {
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -0700445 ScopedLocalRef<jstring> jvalue(env, env->NewStringUTF(value.c_str()));
446 if (jvalue.get() == nullptr) {
447 return false;
448 }
449 env->SetObjectArrayElement(result, static_cast<jint>(id), jvalue.get());
450 return true;
451}
452
453static jobjectArray VMDebug_getRuntimeStatsInternal(JNIEnv* env, jclass) {
454 jobjectArray result = env->NewObjectArray(
455 static_cast<jint>(VMDebugRuntimeStatId::kNumRuntimeStats),
456 WellKnownClasses::java_lang_String,
457 nullptr);
458 if (result == nullptr) {
459 return nullptr;
460 }
461 gc::Heap* heap = Runtime::Current()->GetHeap();
462 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcCount,
463 std::to_string(heap->GetGcCount()))) {
464 return nullptr;
465 }
466 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcTime,
467 std::to_string(NsToMs(heap->GetGcTime())))) {
468 return nullptr;
469 }
470 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBytesAllocated,
471 std::to_string(heap->GetBytesAllocatedEver()))) {
472 return nullptr;
473 }
474 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBytesFreed,
475 std::to_string(heap->GetBytesFreedEver()))) {
476 return nullptr;
477 }
478 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcCount,
479 std::to_string(heap->GetBlockingGcCount()))) {
480 return nullptr;
481 }
482 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcTime,
483 std::to_string(NsToMs(heap->GetBlockingGcTime())))) {
484 return nullptr;
485 }
486 {
487 std::ostringstream output;
488 heap->DumpGcCountRateHistogram(output);
489 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcCountRateHistogram,
490 output.str())) {
491 return nullptr;
492 }
493 }
494 {
495 std::ostringstream output;
496 heap->DumpBlockingGcCountRateHistogram(output);
497 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcCountRateHistogram,
498 output.str())) {
499 return nullptr;
500 }
501 }
502 return result;
503}
504
Leonard Mosescueb842212016-10-06 17:26:36 -0700505static void VMDebug_attachAgent(JNIEnv* env, jclass, jstring agent) {
506 if (agent == nullptr) {
507 ScopedObjectAccess soa(env);
508 ThrowNullPointerException("agent is null");
509 return;
510 }
511
512 if (!Dbg::IsJdwpAllowed()) {
513 ScopedObjectAccess soa(env);
514 ThrowSecurityException("Can't attach agent, process is not debuggable.");
515 return;
516 }
517
518 std::string filename;
519 {
520 ScopedUtfChars chars(env, agent);
521 if (env->ExceptionCheck()) {
522 return;
523 }
524 filename = chars.c_str();
525 }
526
527 Runtime::Current()->AttachAgent(filename);
528}
529
Elliott Hughes0512f022012-03-15 22:10:52 -0700530static JNINativeMethod gMethods[] = {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700531 NATIVE_METHOD(VMDebug, countInstancesOfClass, "(Ljava/lang/Class;Z)J"),
Mathieu Chartierf1820852015-07-10 13:19:51 -0700532 NATIVE_METHOD(VMDebug, countInstancesOfClasses, "([Ljava/lang/Class;Z)[J"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700533 NATIVE_METHOD(VMDebug, crash, "()V"),
534 NATIVE_METHOD(VMDebug, dumpHprofData, "(Ljava/lang/String;Ljava/io/FileDescriptor;)V"),
535 NATIVE_METHOD(VMDebug, dumpHprofDataDdms, "()V"),
536 NATIVE_METHOD(VMDebug, dumpReferenceTables, "()V"),
537 NATIVE_METHOD(VMDebug, getAllocCount, "(I)I"),
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700538 NATIVE_METHOD(VMDebug, getHeapSpaceStats, "([J)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700539 NATIVE_METHOD(VMDebug, getInstructionCount, "([I)V"),
Igor Murashkin3b6f4402017-02-16 16:13:17 -0800540 FAST_NATIVE_METHOD(VMDebug, getLoadedClassCount, "()I"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700541 NATIVE_METHOD(VMDebug, getVmFeatureList, "()[Ljava/lang/String;"),
542 NATIVE_METHOD(VMDebug, infopoint, "(I)V"),
Igor Murashkin3b6f4402017-02-16 16:13:17 -0800543 FAST_NATIVE_METHOD(VMDebug, isDebuggerConnected, "()Z"),
544 FAST_NATIVE_METHOD(VMDebug, isDebuggingEnabled, "()Z"),
Jeff Hao64caa7d2013-08-29 11:18:01 -0700545 NATIVE_METHOD(VMDebug, getMethodTracingMode, "()I"),
Igor Murashkin3b6f4402017-02-16 16:13:17 -0800546 FAST_NATIVE_METHOD(VMDebug, lastDebuggerActivity, "()J"),
547 FAST_NATIVE_METHOD(VMDebug, printLoadedClasses, "(I)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700548 NATIVE_METHOD(VMDebug, resetAllocCount, "(I)V"),
549 NATIVE_METHOD(VMDebug, resetInstructionCount, "()V"),
550 NATIVE_METHOD(VMDebug, startAllocCounting, "()V"),
551 NATIVE_METHOD(VMDebug, startEmulatorTracing, "()V"),
552 NATIVE_METHOD(VMDebug, startInstructionCounting, "()V"),
Jeff Hao23009dc2013-08-22 15:36:42 -0700553 NATIVE_METHOD(VMDebug, startMethodTracingDdmsImpl, "(IIZI)V"),
Shukang Zhouca6e14e2017-01-25 00:35:21 -0800554 NATIVE_METHOD(VMDebug, startMethodTracingFd, "(Ljava/lang/String;Ljava/io/FileDescriptor;IIZIZ)V"),
Jeff Hao4044bda2014-01-06 15:50:45 -0800555 NATIVE_METHOD(VMDebug, startMethodTracingFilename, "(Ljava/lang/String;IIZI)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700556 NATIVE_METHOD(VMDebug, stopAllocCounting, "()V"),
557 NATIVE_METHOD(VMDebug, stopEmulatorTracing, "()V"),
558 NATIVE_METHOD(VMDebug, stopInstructionCounting, "()V"),
559 NATIVE_METHOD(VMDebug, stopMethodTracing, "()V"),
Igor Murashkin3b6f4402017-02-16 16:13:17 -0800560 FAST_NATIVE_METHOD(VMDebug, threadCpuTimeNanos, "()J"),
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -0700561 NATIVE_METHOD(VMDebug, getRuntimeStatInternal, "(I)Ljava/lang/String;"),
Leonard Mosescueb842212016-10-06 17:26:36 -0700562 NATIVE_METHOD(VMDebug, getRuntimeStatsInternal, "()[Ljava/lang/String;"),
563 NATIVE_METHOD(VMDebug, attachAgent, "(Ljava/lang/String;)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700564};
565
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700566void register_dalvik_system_VMDebug(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700567 REGISTER_NATIVE_METHODS("dalvik/system/VMDebug");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700568}
569
570} // namespace art