blob: 46881b00fa5bebce49f83efaece308856ee86aa1 [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"
Elliott Hugheseac76672012-05-24 21:56:51 -070025#include "class_linker.h"
Ian Rogers62d6c772013-02-27 08:32:07 -080026#include "common_throws.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070027#include "debugger.h"
Mathieu Chartier7410f292013-11-24 13:17:35 -080028#include "gc/space/bump_pointer_space.h"
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -070029#include "gc/space/dlmalloc_space.h"
30#include "gc/space/large_object_space.h"
31#include "gc/space/space-inl.h"
Mathieu Chartier1f3b5352014-02-03 14:00:42 -080032#include "gc/space/zygote_space.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070033#include "hprof/hprof.h"
34#include "jni_internal.h"
Ian Rogers2dd0e2c2013-01-24 12:42:14 -080035#include "mirror/class.h"
Ian Rogersdd157d72014-05-15 14:47:50 -070036#include "ScopedLocalRef.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070037#include "ScopedUtfChars.h"
Ian Rogers53b8b092014-03-13 23:45:53 -070038#include "scoped_fast_native_object_access.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070039#include "trace.h"
Ian Rogersdd157d72014-05-15 14:47:50 -070040#include "well_known_classes.h"
Elliott Hugheseac76672012-05-24 21:56:51 -070041
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070042namespace art {
43
Elliott Hughes0512f022012-03-15 22:10:52 -070044static jobjectArray VMDebug_getVmFeatureList(JNIEnv* env, jclass) {
Ian Rogersdd157d72014-05-15 14:47:50 -070045 static const char* features[] = {
46 "method-trace-profiling",
47 "method-trace-profiling-streaming",
48 "method-sample-profiling",
49 "hprof-heap-dump",
50 "hprof-heap-dump-streaming",
51 };
52 jobjectArray result = env->NewObjectArray(arraysize(features),
53 WellKnownClasses::java_lang_String,
54 nullptr);
55 if (result != nullptr) {
56 for (size_t i = 0; i < arraysize(features); ++i) {
57 ScopedLocalRef<jstring> jfeature(env, env->NewStringUTF(features[i]));
58 if (jfeature.get() == nullptr) {
59 return nullptr;
60 }
61 env->SetObjectArrayElement(result, i, jfeature.get());
62 }
63 }
64 return result;
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070065}
66
Elliott Hughes0512f022012-03-15 22:10:52 -070067static void VMDebug_startAllocCounting(JNIEnv*, jclass) {
Mathieu Chartier9ef78b52014-09-25 17:03:12 -070068 Runtime::Current()->SetStatsEnabled(true);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070069}
70
Elliott Hughes0512f022012-03-15 22:10:52 -070071static void VMDebug_stopAllocCounting(JNIEnv*, jclass) {
Mathieu Chartier9ef78b52014-09-25 17:03:12 -070072 Runtime::Current()->SetStatsEnabled(false);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070073}
74
Elliott Hughes1bac54f2012-03-16 12:48:31 -070075static jint VMDebug_getAllocCount(JNIEnv*, jclass, jint kind) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070076 return Runtime::Current()->GetStat(kind);
77}
78
Elliott Hughes0512f022012-03-15 22:10:52 -070079static void VMDebug_resetAllocCount(JNIEnv*, jclass, jint kinds) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070080 Runtime::Current()->ResetStats(kinds);
81}
82
Jeff Hao23009dc2013-08-22 15:36:42 -070083static void VMDebug_startMethodTracingDdmsImpl(JNIEnv*, jclass, jint bufferSize, jint flags,
84 jboolean samplingEnabled, jint intervalUs) {
Andreas Gampe7e7e0f42015-03-29 15:26:23 -070085 Trace::Start("[DDMS]", -1, bufferSize, flags, Trace::TraceOutputMode::kDDMS,
86 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
87 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070088}
89
Ian Rogers00f7d0e2012-07-19 15:28:27 -070090static void VMDebug_startMethodTracingFd(JNIEnv* env, jclass, jstring javaTraceFilename,
Jeff Hao4044bda2014-01-06 15:50:45 -080091 jobject javaFd, jint bufferSize, jint flags,
92 jboolean samplingEnabled, jint intervalUs) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -070093 int originalFd = jniGetFDFromFileDescriptor(env, javaFd);
94 if (originalFd < 0) {
95 return;
96 }
97
98 int fd = dup(originalFd);
99 if (fd < 0) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700100 ScopedObjectAccess soa(env);
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +0000101 soa.Self()->ThrowNewExceptionF("Ljava/lang/RuntimeException;",
Ian Rogers62d6c772013-02-27 08:32:07 -0800102 "dup(%d) failed: %s", originalFd, strerror(errno));
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700103 return;
104 }
105
106 ScopedUtfChars traceFilename(env, javaTraceFilename);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700107 if (traceFilename.c_str() == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700108 return;
109 }
Andreas Gampe7e7e0f42015-03-29 15:26:23 -0700110 Trace::Start(traceFilename.c_str(), fd, bufferSize, flags, Trace::TraceOutputMode::kFile,
111 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
112 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700113}
114
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700115static void VMDebug_startMethodTracingFilename(JNIEnv* env, jclass, jstring javaTraceFilename,
Jeff Hao4044bda2014-01-06 15:50:45 -0800116 jint bufferSize, jint flags,
117 jboolean samplingEnabled, jint intervalUs) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700118 ScopedUtfChars traceFilename(env, javaTraceFilename);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700119 if (traceFilename.c_str() == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700120 return;
121 }
Andreas Gampe7e7e0f42015-03-29 15:26:23 -0700122 Trace::Start(traceFilename.c_str(), -1, bufferSize, flags, Trace::TraceOutputMode::kFile,
123 samplingEnabled ? Trace::TraceMode::kSampling : Trace::TraceMode::kMethodTracing,
124 intervalUs);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700125}
126
Jeff Hao64caa7d2013-08-29 11:18:01 -0700127static jint VMDebug_getMethodTracingMode(JNIEnv*, jclass) {
128 return Trace::GetMethodTracingMode();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700129}
130
Elliott Hughes0512f022012-03-15 22:10:52 -0700131static void VMDebug_stopMethodTracing(JNIEnv*, jclass) {
jeffhaoe343b762011-12-05 16:36:44 -0800132 Trace::Stop();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700133}
134
Elliott Hughes0512f022012-03-15 22:10:52 -0700135static void VMDebug_startEmulatorTracing(JNIEnv*, jclass) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700136 UNIMPLEMENTED(WARNING);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700137 // dvmEmulatorTraceStart();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700138}
139
Elliott Hughes0512f022012-03-15 22:10:52 -0700140static void VMDebug_stopEmulatorTracing(JNIEnv*, jclass) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700141 UNIMPLEMENTED(WARNING);
Brian Carlstrom7934ac22013-07-26 10:54:15 -0700142 // dvmEmulatorTraceStop();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700143}
144
Elliott Hughes0512f022012-03-15 22:10:52 -0700145static jboolean VMDebug_isDebuggerConnected(JNIEnv*, jclass) {
Elliott Hughesc0f09332012-03-26 13:27:06 -0700146 return Dbg::IsDebuggerActive();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700147}
148
Elliott Hughes0512f022012-03-15 22:10:52 -0700149static jboolean VMDebug_isDebuggingEnabled(JNIEnv*, jclass) {
Elliott Hughesc0f09332012-03-26 13:27:06 -0700150 return Dbg::IsJdwpConfigured();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700151}
152
Elliott Hughes0512f022012-03-15 22:10:52 -0700153static jlong VMDebug_lastDebuggerActivity(JNIEnv*, jclass) {
Elliott Hughes872d4ec2011-10-21 17:07:15 -0700154 return Dbg::LastDebuggerActivity();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700155}
156
Ian Rogers62d6c772013-02-27 08:32:07 -0800157static void ThrowUnsupportedOperationException(JNIEnv* env) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700158 ScopedObjectAccess soa(env);
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700159 soa.Self()->ThrowNewException("Ljava/lang/UnsupportedOperationException;", nullptr);
Ian Rogers62d6c772013-02-27 08:32:07 -0800160}
161
162static void VMDebug_startInstructionCounting(JNIEnv* env, jclass) {
163 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700164}
165
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700166static void VMDebug_stopInstructionCounting(JNIEnv* env, jclass) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800167 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700168}
169
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700170static void VMDebug_getInstructionCount(JNIEnv* env, jclass, jintArray /*javaCounts*/) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800171 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700172}
173
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700174static void VMDebug_resetInstructionCount(JNIEnv* env, jclass) {
Ian Rogers62d6c772013-02-27 08:32:07 -0800175 ThrowUnsupportedOperationException(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700176}
177
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700178static void VMDebug_printLoadedClasses(JNIEnv* env, jclass, jint flags) {
Ian Rogers53b8b092014-03-13 23:45:53 -0700179 ScopedFastNativeObjectAccess soa(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700180 return Runtime::Current()->GetClassLinker()->DumpAllClasses(flags);
181}
182
Ian Rogers7dfb28c2013-08-22 08:18:36 -0700183static jint VMDebug_getLoadedClassCount(JNIEnv* env, jclass) {
Ian Rogers53b8b092014-03-13 23:45:53 -0700184 ScopedFastNativeObjectAccess soa(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700185 return Runtime::Current()->GetClassLinker()->NumLoadedClasses();
186}
187
188/*
189 * Returns the thread-specific CPU-time clock value for the current thread,
190 * or -1 if the feature isn't supported.
191 */
Elliott Hughes0512f022012-03-15 22:10:52 -0700192static jlong VMDebug_threadCpuTimeNanos(JNIEnv*, jclass) {
193 return ThreadCpuNanoTime();
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700194}
195
196/*
197 * static void dumpHprofData(String fileName, FileDescriptor fd)
198 *
199 * Cause "hprof" data to be dumped. We can throw an IOException if an
200 * error occurs during file handling.
201 */
Elliott Hughes0512f022012-03-15 22:10:52 -0700202static void VMDebug_dumpHprofData(JNIEnv* env, jclass, jstring javaFilename, jobject javaFd) {
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700203 // Only one of these may be null.
204 if (javaFilename == nullptr && javaFd == nullptr) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700205 ScopedObjectAccess soa(env);
Nicolas Geoffray0aa50ce2015-03-10 11:03:29 +0000206 ThrowNullPointerException("fileName == null && fd == null");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700207 return;
208 }
209
210 std::string filename;
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700211 if (javaFilename != nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700212 ScopedUtfChars chars(env, javaFilename);
213 if (env->ExceptionCheck()) {
214 return;
215 }
216 filename = chars.c_str();
217 } else {
218 filename = "[fd]";
219 }
220
221 int fd = -1;
Mathieu Chartier2cebb242015-04-21 16:50:40 -0700222 if (javaFd != nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700223 fd = jniGetFDFromFileDescriptor(env, javaFd);
224 if (fd < 0) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700225 ScopedObjectAccess soa(env);
Ian Rogers62d6c772013-02-27 08:32:07 -0800226 ThrowRuntimeException("Invalid file descriptor");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700227 return;
228 }
229 }
230
Elliott Hughes622a6982012-06-08 17:58:54 -0700231 hprof::DumpHeap(filename.c_str(), fd, false);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700232}
233
Elliott Hugheseac76672012-05-24 21:56:51 -0700234static void VMDebug_dumpHprofDataDdms(JNIEnv*, jclass) {
Elliott Hughes622a6982012-06-08 17:58:54 -0700235 hprof::DumpHeap("[DDMS]", -1, true);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700236}
237
Elliott Hughes0512f022012-03-15 22:10:52 -0700238static void VMDebug_dumpReferenceTables(JNIEnv* env, jclass) {
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700239 ScopedObjectAccess soa(env);
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700240 LOG(INFO) << "--- reference table dump ---";
241
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700242 soa.Env()->DumpReferenceTables(LOG(INFO));
243 soa.Vm()->DumpReferenceTables(LOG(INFO));
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700244
245 LOG(INFO) << "---";
246}
247
Elliott Hughes0512f022012-03-15 22:10:52 -0700248static void VMDebug_crash(JNIEnv*, jclass) {
Elliott Hughes81ff3182012-03-23 20:35:56 -0700249 LOG(FATAL) << "Crashing runtime on request";
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700250}
251
Elliott Hughes0512f022012-03-15 22:10:52 -0700252static void VMDebug_infopoint(JNIEnv*, jclass, jint id) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700253 LOG(INFO) << "VMDebug infopoint " << id << " hit";
254}
255
Ian Rogers00f7d0e2012-07-19 15:28:27 -0700256static jlong VMDebug_countInstancesOfClass(JNIEnv* env, jclass, jclass javaClass,
257 jboolean countAssignable) {
258 ScopedObjectAccess soa(env);
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800259 gc::Heap* heap = Runtime::Current()->GetHeap();
Mathieu Chartier83c8ee02014-01-28 14:50:23 -0800260 // We only want reachable instances, so do a GC. Heap::VisitObjects visits all of the heap
261 // objects in the all spaces and the allocation stack.
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800262 heap->CollectGarbage(false);
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800263 mirror::Class* c = soa.Decode<mirror::Class*>(javaClass);
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800264 if (c == nullptr) {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700265 return 0;
266 }
Ian Rogers2dd0e2c2013-01-24 12:42:14 -0800267 std::vector<mirror::Class*> classes;
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800268 classes.push_back(c);
269 uint64_t count = 0;
Mathieu Chartier412c7fc2014-02-07 12:18:39 -0800270 heap->CountInstances(classes, countAssignable, &count);
Elliott Hughesec0f83d2013-01-15 16:54:08 -0800271 return count;
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700272}
273
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700274// We export the VM internal per-heap-space size/alloc/free metrics
275// for the zygote space, alloc space (application heap), and the large
276// object space for dumpsys meminfo. The other memory region data such
277// as PSS, private/shared dirty/shared data are available via
278// /proc/<pid>/smaps.
279static void VMDebug_getHeapSpaceStats(JNIEnv* env, jclass, jlongArray data) {
Ian Rogers6b99dd12013-07-25 12:11:32 -0700280 jlong* arr = reinterpret_cast<jlong*>(env->GetPrimitiveArrayCritical(data, 0));
Mathieu Chartier7410f292013-11-24 13:17:35 -0800281 if (arr == nullptr || env->GetArrayLength(data) < 9) {
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700282 return;
283 }
284
285 size_t allocSize = 0;
286 size_t allocUsed = 0;
287 size_t zygoteSize = 0;
288 size_t zygoteUsed = 0;
289 size_t largeObjectsSize = 0;
290 size_t largeObjectsUsed = 0;
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700291 gc::Heap* heap = Runtime::Current()->GetHeap();
Mathieu Chartier7410f292013-11-24 13:17:35 -0800292 for (gc::space::ContinuousSpace* space : heap->GetContinuousSpaces()) {
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700293 if (space->IsImageSpace()) {
294 // Currently don't include the image space.
295 } else if (space->IsZygoteSpace()) {
Mathieu Chartier1f3b5352014-02-03 14:00:42 -0800296 gc::space::ZygoteSpace* zygote_space = space->AsZygoteSpace();
297 zygoteSize += zygote_space->Size();
298 zygoteUsed += zygote_space->GetBytesAllocated();
Mathieu Chartier7410f292013-11-24 13:17:35 -0800299 } else if (space->IsMallocSpace()) {
300 // This is a malloc space.
Hiroshi Yamauchicf58d4a2013-09-26 14:21:22 -0700301 gc::space::MallocSpace* malloc_space = space->AsMallocSpace();
302 allocSize += malloc_space->GetFootprint();
303 allocUsed += malloc_space->GetBytesAllocated();
Mathieu Chartier7410f292013-11-24 13:17:35 -0800304 } else if (space->IsBumpPointerSpace()) {
Mathieu Chartier692fafd2013-11-29 17:24:40 -0800305 ScopedObjectAccess soa(env);
Mathieu Chartier7410f292013-11-24 13:17:35 -0800306 gc::space::BumpPointerSpace* bump_pointer_space = space->AsBumpPointerSpace();
307 allocSize += bump_pointer_space->Size();
308 allocUsed += bump_pointer_space->GetBytesAllocated();
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700309 }
310 }
Mathieu Chartier7410f292013-11-24 13:17:35 -0800311 for (gc::space::DiscontinuousSpace* space : heap->GetDiscontinuousSpaces()) {
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700312 if (space->IsLargeObjectSpace()) {
313 largeObjectsSize += space->AsLargeObjectSpace()->GetBytesAllocated();
314 largeObjectsUsed += largeObjectsSize;
315 }
316 }
317
318 size_t allocFree = allocSize - allocUsed;
319 size_t zygoteFree = zygoteSize - zygoteUsed;
320 size_t largeObjectsFree = largeObjectsSize - largeObjectsUsed;
321
322 int j = 0;
323 arr[j++] = allocSize;
324 arr[j++] = allocUsed;
325 arr[j++] = allocFree;
326 arr[j++] = zygoteSize;
327 arr[j++] = zygoteUsed;
328 arr[j++] = zygoteFree;
329 arr[j++] = largeObjectsSize;
330 arr[j++] = largeObjectsUsed;
331 arr[j++] = largeObjectsFree;
332 env->ReleasePrimitiveArrayCritical(data, arr, 0);
333}
334
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -0700335// The runtime stat names for VMDebug.getRuntimeStat().
336enum class VMDebugRuntimeStatId {
337 kArtGcGcCount = 0,
338 kArtGcGcTime,
339 kArtGcBytesAllocated,
340 kArtGcBytesFreed,
341 kArtGcBlockingGcCount,
342 kArtGcBlockingGcTime,
343 kArtGcGcCountRateHistogram,
344 kArtGcBlockingGcCountRateHistogram,
345 kNumRuntimeStats,
346};
347
348static jobject VMDebug_getRuntimeStatInternal(JNIEnv* env, jclass, jint statId) {
349 gc::Heap* heap = Runtime::Current()->GetHeap();
350 switch (static_cast<VMDebugRuntimeStatId>(statId)) {
351 case VMDebugRuntimeStatId::kArtGcGcCount: {
352 std::string output = std::to_string(heap->GetGcCount());
353 return env->NewStringUTF(output.c_str());
354 }
355 case VMDebugRuntimeStatId::kArtGcGcTime: {
356 std::string output = std::to_string(NsToMs(heap->GetGcTime()));
357 return env->NewStringUTF(output.c_str());
358 }
359 case VMDebugRuntimeStatId::kArtGcBytesAllocated: {
360 std::string output = std::to_string(heap->GetBytesAllocatedEver());
361 return env->NewStringUTF(output.c_str());
362 }
363 case VMDebugRuntimeStatId::kArtGcBytesFreed: {
364 std::string output = std::to_string(heap->GetBytesFreedEver());
365 return env->NewStringUTF(output.c_str());
366 }
367 case VMDebugRuntimeStatId::kArtGcBlockingGcCount: {
368 std::string output = std::to_string(heap->GetBlockingGcCount());
369 return env->NewStringUTF(output.c_str());
370 }
371 case VMDebugRuntimeStatId::kArtGcBlockingGcTime: {
372 std::string output = std::to_string(NsToMs(heap->GetBlockingGcTime()));
373 return env->NewStringUTF(output.c_str());
374 }
375 case VMDebugRuntimeStatId::kArtGcGcCountRateHistogram: {
376 std::ostringstream output;
377 heap->DumpGcCountRateHistogram(output);
378 return env->NewStringUTF(output.str().c_str());
379 }
380 case VMDebugRuntimeStatId::kArtGcBlockingGcCountRateHistogram: {
381 std::ostringstream output;
382 heap->DumpBlockingGcCountRateHistogram(output);
383 return env->NewStringUTF(output.str().c_str());
384 }
385 default:
386 return nullptr;
387 }
388}
389
390static bool SetRuntimeStatValue(JNIEnv* env, jobjectArray result, VMDebugRuntimeStatId id,
391 std::string value) {
392 ScopedLocalRef<jstring> jvalue(env, env->NewStringUTF(value.c_str()));
393 if (jvalue.get() == nullptr) {
394 return false;
395 }
396 env->SetObjectArrayElement(result, static_cast<jint>(id), jvalue.get());
397 return true;
398}
399
400static jobjectArray VMDebug_getRuntimeStatsInternal(JNIEnv* env, jclass) {
401 jobjectArray result = env->NewObjectArray(
402 static_cast<jint>(VMDebugRuntimeStatId::kNumRuntimeStats),
403 WellKnownClasses::java_lang_String,
404 nullptr);
405 if (result == nullptr) {
406 return nullptr;
407 }
408 gc::Heap* heap = Runtime::Current()->GetHeap();
409 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcCount,
410 std::to_string(heap->GetGcCount()))) {
411 return nullptr;
412 }
413 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcTime,
414 std::to_string(NsToMs(heap->GetGcTime())))) {
415 return nullptr;
416 }
417 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBytesAllocated,
418 std::to_string(heap->GetBytesAllocatedEver()))) {
419 return nullptr;
420 }
421 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBytesFreed,
422 std::to_string(heap->GetBytesFreedEver()))) {
423 return nullptr;
424 }
425 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcCount,
426 std::to_string(heap->GetBlockingGcCount()))) {
427 return nullptr;
428 }
429 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcTime,
430 std::to_string(NsToMs(heap->GetBlockingGcTime())))) {
431 return nullptr;
432 }
433 {
434 std::ostringstream output;
435 heap->DumpGcCountRateHistogram(output);
436 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcGcCountRateHistogram,
437 output.str())) {
438 return nullptr;
439 }
440 }
441 {
442 std::ostringstream output;
443 heap->DumpBlockingGcCountRateHistogram(output);
444 if (!SetRuntimeStatValue(env, result, VMDebugRuntimeStatId::kArtGcBlockingGcCountRateHistogram,
445 output.str())) {
446 return nullptr;
447 }
448 }
449 return result;
450}
451
Elliott Hughes0512f022012-03-15 22:10:52 -0700452static JNINativeMethod gMethods[] = {
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700453 NATIVE_METHOD(VMDebug, countInstancesOfClass, "(Ljava/lang/Class;Z)J"),
454 NATIVE_METHOD(VMDebug, crash, "()V"),
455 NATIVE_METHOD(VMDebug, dumpHprofData, "(Ljava/lang/String;Ljava/io/FileDescriptor;)V"),
456 NATIVE_METHOD(VMDebug, dumpHprofDataDdms, "()V"),
457 NATIVE_METHOD(VMDebug, dumpReferenceTables, "()V"),
458 NATIVE_METHOD(VMDebug, getAllocCount, "(I)I"),
Hiroshi Yamauchi09b07a92013-07-15 13:17:06 -0700459 NATIVE_METHOD(VMDebug, getHeapSpaceStats, "([J)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700460 NATIVE_METHOD(VMDebug, getInstructionCount, "([I)V"),
Ian Rogers53b8b092014-03-13 23:45:53 -0700461 NATIVE_METHOD(VMDebug, getLoadedClassCount, "!()I"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700462 NATIVE_METHOD(VMDebug, getVmFeatureList, "()[Ljava/lang/String;"),
463 NATIVE_METHOD(VMDebug, infopoint, "(I)V"),
Ian Rogers53b8b092014-03-13 23:45:53 -0700464 NATIVE_METHOD(VMDebug, isDebuggerConnected, "!()Z"),
465 NATIVE_METHOD(VMDebug, isDebuggingEnabled, "!()Z"),
Jeff Hao64caa7d2013-08-29 11:18:01 -0700466 NATIVE_METHOD(VMDebug, getMethodTracingMode, "()I"),
Ian Rogers53b8b092014-03-13 23:45:53 -0700467 NATIVE_METHOD(VMDebug, lastDebuggerActivity, "!()J"),
468 NATIVE_METHOD(VMDebug, printLoadedClasses, "!(I)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700469 NATIVE_METHOD(VMDebug, resetAllocCount, "(I)V"),
470 NATIVE_METHOD(VMDebug, resetInstructionCount, "()V"),
471 NATIVE_METHOD(VMDebug, startAllocCounting, "()V"),
472 NATIVE_METHOD(VMDebug, startEmulatorTracing, "()V"),
473 NATIVE_METHOD(VMDebug, startInstructionCounting, "()V"),
Jeff Hao23009dc2013-08-22 15:36:42 -0700474 NATIVE_METHOD(VMDebug, startMethodTracingDdmsImpl, "(IIZI)V"),
Jeff Hao4044bda2014-01-06 15:50:45 -0800475 NATIVE_METHOD(VMDebug, startMethodTracingFd, "(Ljava/lang/String;Ljava/io/FileDescriptor;IIZI)V"),
476 NATIVE_METHOD(VMDebug, startMethodTracingFilename, "(Ljava/lang/String;IIZI)V"),
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700477 NATIVE_METHOD(VMDebug, stopAllocCounting, "()V"),
478 NATIVE_METHOD(VMDebug, stopEmulatorTracing, "()V"),
479 NATIVE_METHOD(VMDebug, stopInstructionCounting, "()V"),
480 NATIVE_METHOD(VMDebug, stopMethodTracing, "()V"),
Ian Rogers53b8b092014-03-13 23:45:53 -0700481 NATIVE_METHOD(VMDebug, threadCpuTimeNanos, "!()J"),
Hiroshi Yamauchia1c9f012015-04-02 10:18:12 -0700482 NATIVE_METHOD(VMDebug, getRuntimeStatInternal, "(I)Ljava/lang/String;"),
483 NATIVE_METHOD(VMDebug, getRuntimeStatsInternal, "()[Ljava/lang/String;")
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700484};
485
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700486void register_dalvik_system_VMDebug(JNIEnv* env) {
Elliott Hugheseac76672012-05-24 21:56:51 -0700487 REGISTER_NATIVE_METHODS("dalvik/system/VMDebug");
Elliott Hughes9d5ccec2011-09-19 13:19:50 -0700488}
489
490} // namespace art