1*e82f7db8SAndroid Build Coastguard Worker /*
2*e82f7db8SAndroid Build Coastguard Worker * Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
3*e82f7db8SAndroid Build Coastguard Worker * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*e82f7db8SAndroid Build Coastguard Worker *
5*e82f7db8SAndroid Build Coastguard Worker * This code is free software; you can redistribute it and/or modify it
6*e82f7db8SAndroid Build Coastguard Worker * under the terms of the GNU General Public License version 2 only, as
7*e82f7db8SAndroid Build Coastguard Worker * published by the Free Software Foundation. Oracle designates this
8*e82f7db8SAndroid Build Coastguard Worker * particular file as subject to the "Classpath" exception as provided
9*e82f7db8SAndroid Build Coastguard Worker * by Oracle in the LICENSE file that accompanied this code.
10*e82f7db8SAndroid Build Coastguard Worker *
11*e82f7db8SAndroid Build Coastguard Worker * This code is distributed in the hope that it will be useful, but WITHOUT
12*e82f7db8SAndroid Build Coastguard Worker * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13*e82f7db8SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14*e82f7db8SAndroid Build Coastguard Worker * version 2 for more details (a copy is included in the LICENSE file that
15*e82f7db8SAndroid Build Coastguard Worker * accompanied this code).
16*e82f7db8SAndroid Build Coastguard Worker *
17*e82f7db8SAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License version
18*e82f7db8SAndroid Build Coastguard Worker * 2 along with this work; if not, write to the Free Software Foundation,
19*e82f7db8SAndroid Build Coastguard Worker * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20*e82f7db8SAndroid Build Coastguard Worker *
21*e82f7db8SAndroid Build Coastguard Worker * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22*e82f7db8SAndroid Build Coastguard Worker * or visit www.oracle.com if you need additional information or have any
23*e82f7db8SAndroid Build Coastguard Worker * questions.
24*e82f7db8SAndroid Build Coastguard Worker */
25*e82f7db8SAndroid Build Coastguard Worker
26*e82f7db8SAndroid Build Coastguard Worker #include <stdatomic.h>
27*e82f7db8SAndroid Build Coastguard Worker
28*e82f7db8SAndroid Build Coastguard Worker #include "vmDebug.h"
29*e82f7db8SAndroid Build Coastguard Worker
30*e82f7db8SAndroid Build Coastguard Worker #include "JDWP.h"
31*e82f7db8SAndroid Build Coastguard Worker #include "debugLoop.h"
32*e82f7db8SAndroid Build Coastguard Worker #include "transport.h"
33*e82f7db8SAndroid Build Coastguard Worker #include "util.h"
34*e82f7db8SAndroid Build Coastguard Worker #include "eventHelper.h"
35*e82f7db8SAndroid Build Coastguard Worker #include "threadControl.h"
36*e82f7db8SAndroid Build Coastguard Worker
37*e82f7db8SAndroid Build Coastguard Worker static _Atomic(jlong) lastDebuggerActivity = ATOMIC_VAR_INIT(0LL);
38*e82f7db8SAndroid Build Coastguard Worker static _Atomic(jboolean) hasSeenDebuggerActivity = ATOMIC_VAR_INIT(JNI_FALSE);
39*e82f7db8SAndroid Build Coastguard Worker
40*e82f7db8SAndroid Build Coastguard Worker // Reset the tracking variables.
vmDebug_onDisconnect()41*e82f7db8SAndroid Build Coastguard Worker void vmDebug_onDisconnect()
42*e82f7db8SAndroid Build Coastguard Worker {
43*e82f7db8SAndroid Build Coastguard Worker atomic_store(&lastDebuggerActivity, 0LL);
44*e82f7db8SAndroid Build Coastguard Worker atomic_store(&hasSeenDebuggerActivity, JNI_FALSE);
45*e82f7db8SAndroid Build Coastguard Worker }
46*e82f7db8SAndroid Build Coastguard Worker
47*e82f7db8SAndroid Build Coastguard Worker // Mark us as having seen actual debugger activity (so isDebuggerConnected can return true) and that
48*e82f7db8SAndroid Build Coastguard Worker // we are currently doing something as the debugger.
vmDebug_notifyDebuggerActivityStart()49*e82f7db8SAndroid Build Coastguard Worker void vmDebug_notifyDebuggerActivityStart()
50*e82f7db8SAndroid Build Coastguard Worker {
51*e82f7db8SAndroid Build Coastguard Worker atomic_store(&lastDebuggerActivity, 0LL);
52*e82f7db8SAndroid Build Coastguard Worker atomic_store(&hasSeenDebuggerActivity, JNI_TRUE);
53*e82f7db8SAndroid Build Coastguard Worker }
54*e82f7db8SAndroid Build Coastguard Worker
55*e82f7db8SAndroid Build Coastguard Worker // Update the timestamp for the last debugger activity.
vmDebug_notifyDebuggerActivityEnd()56*e82f7db8SAndroid Build Coastguard Worker void vmDebug_notifyDebuggerActivityEnd()
57*e82f7db8SAndroid Build Coastguard Worker {
58*e82f7db8SAndroid Build Coastguard Worker atomic_store(&lastDebuggerActivity, milliTime());
59*e82f7db8SAndroid Build Coastguard Worker }
60*e82f7db8SAndroid Build Coastguard Worker
61*e82f7db8SAndroid Build Coastguard Worker // For backwards compatibility we are only considered 'connected' as far as VMDebug is concerned if
62*e82f7db8SAndroid Build Coastguard Worker // we have gotten at least one non-ddms JDWP packet.
63*e82f7db8SAndroid Build Coastguard Worker static jboolean
isDebuggerConnected()64*e82f7db8SAndroid Build Coastguard Worker isDebuggerConnected()
65*e82f7db8SAndroid Build Coastguard Worker {
66*e82f7db8SAndroid Build Coastguard Worker return transport_is_open() && atomic_load(&hasSeenDebuggerActivity);
67*e82f7db8SAndroid Build Coastguard Worker }
68*e82f7db8SAndroid Build Coastguard Worker
69*e82f7db8SAndroid Build Coastguard Worker static jboolean JNICALL
VMDebug_isDebuggerConnected(JNIEnv * env,jclass klass)70*e82f7db8SAndroid Build Coastguard Worker VMDebug_isDebuggerConnected(JNIEnv* env, jclass klass)
71*e82f7db8SAndroid Build Coastguard Worker {
72*e82f7db8SAndroid Build Coastguard Worker return isDebuggerConnected();
73*e82f7db8SAndroid Build Coastguard Worker }
74*e82f7db8SAndroid Build Coastguard Worker
75*e82f7db8SAndroid Build Coastguard Worker static void JNICALL
VMDebug_suspendAllAndSendVmStart(JNIEnv * env,jclass klass)76*e82f7db8SAndroid Build Coastguard Worker VMDebug_suspendAllAndSendVmStart(JNIEnv* env, jclass klass)
77*e82f7db8SAndroid Build Coastguard Worker {
78*e82f7db8SAndroid Build Coastguard Worker jthread currentThread;
79*e82f7db8SAndroid Build Coastguard Worker JVMTI_FUNC_PTR(gdata->jvmti, GetCurrentThread)(gdata->jvmti, ¤tThread);
80*e82f7db8SAndroid Build Coastguard Worker eventHelper_reportVMInit(getEnv(), 0, currentThread, JDWP_SUSPEND_POLICY(ALL));
81*e82f7db8SAndroid Build Coastguard Worker }
82*e82f7db8SAndroid Build Coastguard Worker
83*e82f7db8SAndroid Build Coastguard Worker static jboolean JNICALL
VMDebug_isDebuggingEnabled(JNIEnv * env,jclass klass)84*e82f7db8SAndroid Build Coastguard Worker VMDebug_isDebuggingEnabled(JNIEnv* env, jclass klass)
85*e82f7db8SAndroid Build Coastguard Worker {
86*e82f7db8SAndroid Build Coastguard Worker // We are running the debugger so debugging is definitely enabled.
87*e82f7db8SAndroid Build Coastguard Worker return JNI_TRUE;
88*e82f7db8SAndroid Build Coastguard Worker }
89*e82f7db8SAndroid Build Coastguard Worker
90*e82f7db8SAndroid Build Coastguard Worker static jlong JNICALL
VMDebug_lastDebuggerActivity(JNIEnv * env,jclass klass)91*e82f7db8SAndroid Build Coastguard Worker VMDebug_lastDebuggerActivity(JNIEnv* env, jclass klass)
92*e82f7db8SAndroid Build Coastguard Worker {
93*e82f7db8SAndroid Build Coastguard Worker if (!isDebuggerConnected()) {
94*e82f7db8SAndroid Build Coastguard Worker LOG_ERROR(("VMDebug.lastDebuggerActivity called without active debugger"));
95*e82f7db8SAndroid Build Coastguard Worker return -1;
96*e82f7db8SAndroid Build Coastguard Worker }
97*e82f7db8SAndroid Build Coastguard Worker jlong last_time = atomic_load(&lastDebuggerActivity);
98*e82f7db8SAndroid Build Coastguard Worker if (last_time == 0) {
99*e82f7db8SAndroid Build Coastguard Worker LOG_MISC(("debugger is performing an action"));
100*e82f7db8SAndroid Build Coastguard Worker return 0;
101*e82f7db8SAndroid Build Coastguard Worker }
102*e82f7db8SAndroid Build Coastguard Worker
103*e82f7db8SAndroid Build Coastguard Worker jlong cur_time = milliTime();
104*e82f7db8SAndroid Build Coastguard Worker
105*e82f7db8SAndroid Build Coastguard Worker if (cur_time < last_time) {
106*e82f7db8SAndroid Build Coastguard Worker LOG_ERROR(("Time seemed to go backwards: last was %lld, current is %lld",
107*e82f7db8SAndroid Build Coastguard Worker last_time, cur_time));
108*e82f7db8SAndroid Build Coastguard Worker return 0;
109*e82f7db8SAndroid Build Coastguard Worker }
110*e82f7db8SAndroid Build Coastguard Worker jlong res = cur_time - last_time;
111*e82f7db8SAndroid Build Coastguard Worker LOG_MISC(("Debugger interval is %lld", res));
112*e82f7db8SAndroid Build Coastguard Worker return res;
113*e82f7db8SAndroid Build Coastguard Worker }
114*e82f7db8SAndroid Build Coastguard Worker
115*e82f7db8SAndroid Build Coastguard Worker void
vmDebug_initalize(JNIEnv * env)116*e82f7db8SAndroid Build Coastguard Worker vmDebug_initalize(JNIEnv* env)
117*e82f7db8SAndroid Build Coastguard Worker {
118*e82f7db8SAndroid Build Coastguard Worker WITH_LOCAL_REFS(env, 1) {
119*e82f7db8SAndroid Build Coastguard Worker jclass vmdebug_class = JNI_FUNC_PTR(env,FindClass)(env, "dalvik/system/VMDebug");
120*e82f7db8SAndroid Build Coastguard Worker if (vmdebug_class == NULL) {
121*e82f7db8SAndroid Build Coastguard Worker // The VMDebug class isn't available. We don't need to do anything.
122*e82f7db8SAndroid Build Coastguard Worker LOG_MISC(("dalvik.system.VMDebug does not seem to be available on this runtime."));
123*e82f7db8SAndroid Build Coastguard Worker // Get rid of the ClassNotFoundException.
124*e82f7db8SAndroid Build Coastguard Worker JNI_FUNC_PTR(env,ExceptionClear)(env);
125*e82f7db8SAndroid Build Coastguard Worker goto finish;
126*e82f7db8SAndroid Build Coastguard Worker }
127*e82f7db8SAndroid Build Coastguard Worker
128*e82f7db8SAndroid Build Coastguard Worker JNINativeMethod methods[4];
129*e82f7db8SAndroid Build Coastguard Worker
130*e82f7db8SAndroid Build Coastguard Worker // Take over the implementation of these functions.
131*e82f7db8SAndroid Build Coastguard Worker methods[0].name = "lastDebuggerActivity";
132*e82f7db8SAndroid Build Coastguard Worker methods[0].signature = "()J";
133*e82f7db8SAndroid Build Coastguard Worker methods[0].fnPtr = (void*)VMDebug_lastDebuggerActivity;
134*e82f7db8SAndroid Build Coastguard Worker
135*e82f7db8SAndroid Build Coastguard Worker methods[1].name = "isDebuggingEnabled";
136*e82f7db8SAndroid Build Coastguard Worker methods[1].signature = "()Z";
137*e82f7db8SAndroid Build Coastguard Worker methods[1].fnPtr = (void*)VMDebug_isDebuggingEnabled;
138*e82f7db8SAndroid Build Coastguard Worker
139*e82f7db8SAndroid Build Coastguard Worker methods[2].name = "isDebuggerConnected";
140*e82f7db8SAndroid Build Coastguard Worker methods[2].signature = "()Z";
141*e82f7db8SAndroid Build Coastguard Worker methods[2].fnPtr = (void*)VMDebug_isDebuggerConnected;
142*e82f7db8SAndroid Build Coastguard Worker
143*e82f7db8SAndroid Build Coastguard Worker methods[3].name = "suspendAllAndSendVmStart";
144*e82f7db8SAndroid Build Coastguard Worker methods[3].signature = "()V";
145*e82f7db8SAndroid Build Coastguard Worker methods[3].fnPtr = (void*)VMDebug_suspendAllAndSendVmStart;
146*e82f7db8SAndroid Build Coastguard Worker
147*e82f7db8SAndroid Build Coastguard Worker jint res = JNI_FUNC_PTR(env,RegisterNatives)(env,
148*e82f7db8SAndroid Build Coastguard Worker vmdebug_class,
149*e82f7db8SAndroid Build Coastguard Worker methods,
150*e82f7db8SAndroid Build Coastguard Worker sizeof(methods) / sizeof(JNINativeMethod));
151*e82f7db8SAndroid Build Coastguard Worker if (res != JNI_OK) {
152*e82f7db8SAndroid Build Coastguard Worker EXIT_ERROR(JVMTI_ERROR_INTERNAL,
153*e82f7db8SAndroid Build Coastguard Worker "RegisterNatives returned failure for VMDebug class");
154*e82f7db8SAndroid Build Coastguard Worker }
155*e82f7db8SAndroid Build Coastguard Worker
156*e82f7db8SAndroid Build Coastguard Worker finish: ;
157*e82f7db8SAndroid Build Coastguard Worker } END_WITH_LOCAL_REFS(env);
158*e82f7db8SAndroid Build Coastguard Worker }
159*e82f7db8SAndroid Build Coastguard Worker
160