2007-12-01 00:00:00 +00:00
|
|
|
/*
|
2010-05-25 15:58:33 -07:00
|
|
|
* Copyright (c) 1998, 2005, Oracle and/or its affiliates. All rights reserved.
|
2007-12-01 00:00:00 +00:00
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
2010-05-25 15:58:33 -07:00
|
|
|
* published by the Free Software Foundation. Oracle designates this
|
2007-12-01 00:00:00 +00:00
|
|
|
* particular file as subject to the "Classpath" exception as provided
|
2010-05-25 15:58:33 -07:00
|
|
|
* by Oracle in the LICENSE file that accompanied this code.
|
2007-12-01 00:00:00 +00:00
|
|
|
*
|
|
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
2010-05-25 15:58:33 -07:00
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
2007-12-01 00:00:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
#include "stepControl.h"
|
|
|
|
#include "eventHandler.h"
|
|
|
|
#include "eventHelper.h"
|
|
|
|
#include "threadControl.h"
|
|
|
|
#include "SDE.h"
|
|
|
|
|
|
|
|
static jrawMonitorID stepLock;
|
|
|
|
|
|
|
|
static jint
|
|
|
|
getFrameCount(jthread thread)
|
|
|
|
{
|
|
|
|
jint count = 0;
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
error = JVMTI_FUNC_PTR(gdata->jvmti,GetFrameCount)
|
|
|
|
(gdata->jvmti, thread, &count);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
EXIT_ERROR(error, "getting frame count");
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Most enabling/disabling of JVMTI events happens implicitly through
|
|
|
|
* the inserting and freeing of handlers for those events. Stepping is
|
|
|
|
* different because requested steps are usually not identical to JVMTI steps.
|
|
|
|
* They usually require multiple events step, and otherwise, before they
|
|
|
|
* complete. While a step request is pending, we may need to temporarily
|
|
|
|
* disable and re-enable stepping, but we can't just remove the handlers
|
|
|
|
* because that would break the application's ability to remove the
|
|
|
|
* events. So, for step events only, we directly enable and disable stepping.
|
|
|
|
* This is safe because there can only ever be one pending step request
|
|
|
|
* per thread.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
enableStepping(jthread thread)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
LOG_STEP(("enableStepping: thread=%p", thread));
|
|
|
|
|
|
|
|
error = threadControl_setEventMode(JVMTI_ENABLE, EI_SINGLE_STEP,
|
|
|
|
thread);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
EXIT_ERROR(error, "enabling single step");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
disableStepping(jthread thread)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
LOG_STEP(("disableStepping: thread=%p", thread));
|
|
|
|
|
|
|
|
error = threadControl_setEventMode(JVMTI_DISABLE, EI_SINGLE_STEP,
|
|
|
|
thread);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
EXIT_ERROR(error, "disabling single step");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static jvmtiError
|
|
|
|
getFrameLocation(jthread thread,
|
|
|
|
jclass *pclazz, jmethodID *pmethod, jlocation *plocation)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
*pclazz = NULL;
|
|
|
|
*pmethod = NULL;
|
|
|
|
*plocation = -1;
|
|
|
|
|
|
|
|
error = JVMTI_FUNC_PTR(gdata->jvmti,GetFrameLocation)
|
|
|
|
(gdata->jvmti, thread, 0, pmethod, plocation);
|
|
|
|
if (error == JVMTI_ERROR_NONE && *pmethod!=NULL ) {
|
|
|
|
/* This also serves to verify that the methodID is valid */
|
|
|
|
error = methodClass(*pmethod, pclazz);
|
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
getLineNumberTable(jmethodID method, jint *pcount,
|
|
|
|
jvmtiLineNumberEntry **ptable)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
*pcount = 0;
|
|
|
|
*ptable = NULL;
|
|
|
|
|
|
|
|
/* If the method is native or obsolete, don't even ask for the line table */
|
|
|
|
if ( isMethodObsolete(method) || isMethodNative(method)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
error = JVMTI_FUNC_PTR(gdata->jvmti,GetLineNumberTable)
|
|
|
|
(gdata->jvmti, method, pcount, ptable);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
*pcount = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint
|
|
|
|
findLineNumber(jthread thread, jlocation location,
|
|
|
|
jvmtiLineNumberEntry *lines, jint count)
|
|
|
|
{
|
|
|
|
jint line = -1;
|
|
|
|
|
|
|
|
if (location != -1) {
|
|
|
|
if (count > 0) {
|
|
|
|
jint i;
|
|
|
|
/* any preface before first line is assigned to first line */
|
|
|
|
for (i=1; i<count; i++) {
|
|
|
|
if (location < lines[i].start_location) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
line = lines[i-1].line_number;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jboolean
|
|
|
|
hasLineNumbers(jmethodID method)
|
|
|
|
{
|
|
|
|
jint count;
|
|
|
|
jvmtiLineNumberEntry *table;
|
|
|
|
|
|
|
|
getLineNumberTable(method, &count, &table);
|
|
|
|
if ( count == 0 ) {
|
|
|
|
return JNI_FALSE;
|
|
|
|
} else {
|
|
|
|
jvmtiDeallocate(table);
|
|
|
|
}
|
|
|
|
return JNI_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jvmtiError
|
|
|
|
initState(JNIEnv *env, jthread thread, StepRequest *step)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initial values that may be changed below
|
|
|
|
*/
|
|
|
|
step->fromLine = -1;
|
|
|
|
step->fromNative = JNI_FALSE;
|
|
|
|
step->frameExited = JNI_FALSE;
|
|
|
|
step->fromStackDepth = getFrameCount(thread);
|
|
|
|
|
|
|
|
if (step->fromStackDepth <= 0) {
|
|
|
|
/*
|
|
|
|
* If there are no stack frames, treat the step as though
|
|
|
|
* from a native frame. This is most likely to occur at the
|
|
|
|
* beginning of a debug session, right after the VM_INIT event,
|
|
|
|
* so we need to do something intelligent.
|
|
|
|
*/
|
|
|
|
step->fromNative = JNI_TRUE;
|
|
|
|
return JVMTI_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Try to get a notification on frame pop. If we're in an opaque frame
|
|
|
|
* we won't be able to, but we can use other methods to detect that
|
|
|
|
* a native frame has exited.
|
|
|
|
*
|
|
|
|
* TO DO: explain the need for this notification.
|
|
|
|
*/
|
|
|
|
error = JVMTI_FUNC_PTR(gdata->jvmti,NotifyFramePop)
|
|
|
|
(gdata->jvmti, thread, 0);
|
|
|
|
if (error == JVMTI_ERROR_OPAQUE_FRAME) {
|
|
|
|
step->fromNative = JNI_TRUE;
|
|
|
|
error = JVMTI_ERROR_NONE;
|
|
|
|
/* continue without error */
|
|
|
|
} else if (error == JVMTI_ERROR_DUPLICATE) {
|
|
|
|
error = JVMTI_ERROR_NONE;
|
|
|
|
/* Already being notified, continue without error */
|
|
|
|
} else if (error != JVMTI_ERROR_NONE) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_STEP(("initState(): frame=%d", step->fromStackDepth));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note: we can't undo the frame pop notify, so
|
|
|
|
* we'll just have to let the handler ignore it if
|
|
|
|
* there are any errors below.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (step->granularity == JDWP_STEP_SIZE(LINE) ) {
|
|
|
|
|
|
|
|
LOG_STEP(("initState(): Begin line step"));
|
|
|
|
|
|
|
|
WITH_LOCAL_REFS(env, 1) {
|
|
|
|
|
|
|
|
jclass clazz;
|
|
|
|
jmethodID method;
|
|
|
|
jlocation location;
|
|
|
|
|
|
|
|
error = getFrameLocation(thread, &clazz, &method, &location);
|
|
|
|
if (error == JVMTI_ERROR_NONE) {
|
|
|
|
/* Clear out previous line table only if we changed methods */
|
|
|
|
if ( method != step->method ) {
|
|
|
|
step->lineEntryCount = 0;
|
|
|
|
if (step->lineEntries != NULL) {
|
|
|
|
jvmtiDeallocate(step->lineEntries);
|
|
|
|
step->lineEntries = NULL;
|
|
|
|
}
|
|
|
|
step->method = method;
|
|
|
|
getLineNumberTable(step->method,
|
|
|
|
&step->lineEntryCount, &step->lineEntries);
|
|
|
|
if (step->lineEntryCount > 0) {
|
|
|
|
convertLineNumberTable(env, clazz,
|
|
|
|
&step->lineEntryCount, &step->lineEntries);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
step->fromLine = findLineNumber(thread, location,
|
|
|
|
step->lineEntries, step->lineEntryCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
} END_WITH_LOCAL_REFS(env);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TO DO: The step handlers (handleFrameChange and handleStep can
|
|
|
|
* be broken down and made simpler now that we can install and de-install event
|
|
|
|
* handlers.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
handleFramePopEvent(JNIEnv *env, EventInfo *evinfo,
|
|
|
|
HandlerNode *node,
|
|
|
|
struct bag *eventBag)
|
|
|
|
{
|
|
|
|
StepRequest *step;
|
|
|
|
jthread thread = evinfo->thread;
|
|
|
|
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_THREAD, "getting step request");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (step->pending) {
|
|
|
|
/*
|
|
|
|
* Note: current depth is reported as *before* the pending frame
|
|
|
|
* pop.
|
|
|
|
*/
|
|
|
|
jint currentDepth;
|
|
|
|
jint fromDepth;
|
|
|
|
jint afterPopDepth;
|
|
|
|
|
|
|
|
currentDepth = getFrameCount(thread);
|
|
|
|
fromDepth = step->fromStackDepth;
|
|
|
|
afterPopDepth = currentDepth-1;
|
|
|
|
|
|
|
|
LOG_STEP(("handleFramePopEvent: BEGIN fromDepth=%d, currentDepth=%d",
|
|
|
|
fromDepth, currentDepth));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we are exiting the original stepping frame, record that
|
|
|
|
* fact here. Once the next step event comes in, we can safely
|
|
|
|
* stop stepping there.
|
|
|
|
*/
|
|
|
|
if (fromDepth > afterPopDepth ) {
|
|
|
|
step->frameExited = JNI_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (step->depth == JDWP_STEP_DEPTH(OVER)) {
|
|
|
|
/*
|
|
|
|
* Either
|
|
|
|
* 1) the original stepping frame is about to be popped
|
|
|
|
* [fromDepth == currentDepth]. Re-enable stepping to
|
|
|
|
* reach a point where we can stop.
|
|
|
|
* 2) a method called from the stepping frame has returned
|
|
|
|
* (during which we had stepping disabled)
|
|
|
|
* [fromDepth == currentDepth - 1]. Re-enable stepping
|
|
|
|
* so that we can continue instructions steps in the
|
|
|
|
* original stepping frame.
|
|
|
|
* 3) a method further down the call chain has notified
|
|
|
|
* of a frame pop [fromDepth < currentDepth - 1]. This
|
|
|
|
* *might* represent case (2) above if the stepping frame
|
|
|
|
* was calling a native method which in turn called a
|
|
|
|
* java method. If so, we must enable stepping to
|
|
|
|
* ensure that we get control back after the intervening
|
|
|
|
* native frame is popped (you can't get frame pop
|
|
|
|
* notifications on native frames). If the native caller
|
|
|
|
* calls another Java method before returning,
|
|
|
|
* stepping will be diabled again and another frame pop
|
|
|
|
* will be awaited.
|
|
|
|
*
|
|
|
|
* If it turns out that this is not case (2) with native
|
|
|
|
* methods, then the enabled stepping is benign and
|
|
|
|
* will be disabled again on the next step event.
|
|
|
|
*
|
|
|
|
* Note that the condition not covered above,
|
|
|
|
* [fromDepth > currentDepth] shouldn't happen since it means
|
|
|
|
* that too many frames have been popped. For robustness,
|
|
|
|
* we enable stepping in that case too, so that the errant
|
|
|
|
* step-over can be stopped.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
LOG_STEP(("handleFramePopEvent: starting singlestep, depth==OVER"));
|
|
|
|
enableStepping(thread);
|
|
|
|
} else if (step->depth == JDWP_STEP_DEPTH(OUT) &&
|
|
|
|
fromDepth > afterPopDepth) {
|
|
|
|
/*
|
|
|
|
* The original stepping frame is about to be popped. Step
|
|
|
|
* until we reach the next safe place to stop.
|
|
|
|
*/
|
|
|
|
LOG_STEP(("handleFramePopEvent: starting singlestep, depth==OUT && fromDepth > afterPopDepth (%d>%d)",fromDepth, afterPopDepth));
|
|
|
|
enableStepping(thread);
|
|
|
|
} else if (step->methodEnterHandlerNode != NULL &&
|
|
|
|
fromDepth >= afterPopDepth) {
|
|
|
|
/*
|
|
|
|
* We installed a method entry event handler as part of a
|
|
|
|
* step into operation. We've popped back to the original
|
|
|
|
* stepping frame without finding a place to stop.
|
|
|
|
* Resume stepping in the original frame.
|
|
|
|
*/
|
|
|
|
LOG_STEP(("handleFramePopEvent: starting singlestep, have methodEnter handler && depth==OUT && fromDepth >= afterPopDepth (%d>%d)",fromDepth, afterPopDepth));
|
|
|
|
enableStepping(thread);
|
|
|
|
(void)eventHandler_free(step->methodEnterHandlerNode);
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
LOG_STEP(("handleFramePopEvent: finished"));
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
handleExceptionCatchEvent(JNIEnv *env, EventInfo *evinfo,
|
|
|
|
HandlerNode *node,
|
|
|
|
struct bag *eventBag)
|
|
|
|
{
|
|
|
|
StepRequest *step;
|
|
|
|
jthread thread = evinfo->thread;
|
|
|
|
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_THREAD, "getting step request");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (step->pending) {
|
|
|
|
/*
|
|
|
|
* Determine where we are on the call stack relative to where
|
|
|
|
* we started.
|
|
|
|
*/
|
|
|
|
jint currentDepth = getFrameCount(thread);
|
|
|
|
jint fromDepth = step->fromStackDepth;
|
|
|
|
|
|
|
|
LOG_STEP(("handleExceptionCatchEvent: fromDepth=%d, currentDepth=%d",
|
|
|
|
fromDepth, currentDepth));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we are exiting the original stepping frame, record that
|
|
|
|
* fact here. Once the next step event comes in, we can safely
|
|
|
|
* stop stepping there.
|
|
|
|
*/
|
|
|
|
if (fromDepth > currentDepth) {
|
|
|
|
step->frameExited = JNI_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (step->depth == JDWP_STEP_DEPTH(OVER) &&
|
|
|
|
fromDepth >= currentDepth) {
|
|
|
|
/*
|
|
|
|
* Either the original stepping frame is done,
|
|
|
|
* or a called method has returned (during which we had stepping
|
|
|
|
* disabled). In either case we must resume stepping.
|
|
|
|
*/
|
|
|
|
enableStepping(thread);
|
|
|
|
} else if (step->depth == JDWP_STEP_DEPTH(OUT) &&
|
|
|
|
fromDepth > currentDepth) {
|
|
|
|
/*
|
|
|
|
* The original stepping frame is done. Step
|
|
|
|
* until we reach the next safe place to stop.
|
|
|
|
*/
|
|
|
|
enableStepping(thread);
|
|
|
|
} else if (step->methodEnterHandlerNode != NULL &&
|
|
|
|
fromDepth >= currentDepth) {
|
|
|
|
/*
|
|
|
|
* We installed a method entry event handler as part of a
|
|
|
|
* step into operation. We've popped back to the original
|
|
|
|
* stepping frame or higher without finding a place to stop.
|
|
|
|
* Resume stepping in the original frame.
|
|
|
|
*/
|
|
|
|
enableStepping(thread);
|
|
|
|
(void)eventHandler_free(step->methodEnterHandlerNode);
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
handleMethodEnterEvent(JNIEnv *env, EventInfo *evinfo,
|
|
|
|
HandlerNode *node,
|
|
|
|
struct bag *eventBag)
|
|
|
|
{
|
|
|
|
StepRequest *step;
|
|
|
|
jthread thread;
|
|
|
|
|
|
|
|
thread = evinfo->thread;
|
|
|
|
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_THREAD, "getting step request");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (step->pending) {
|
|
|
|
jclass clazz;
|
|
|
|
jmethodID method;
|
|
|
|
char *classname;
|
|
|
|
|
|
|
|
LOG_STEP(("handleMethodEnterEvent: thread=%p", thread));
|
|
|
|
|
|
|
|
clazz = evinfo->clazz;
|
|
|
|
method = evinfo->method;
|
|
|
|
classname = getClassname(clazz);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This handler is relevant only to step into
|
|
|
|
*/
|
|
|
|
JDI_ASSERT(step->depth == JDWP_STEP_DEPTH(INTO));
|
|
|
|
|
|
|
|
if ( (!eventFilter_predictFiltering(step->stepHandlerNode,
|
|
|
|
clazz, classname))
|
|
|
|
&& ( step->granularity != JDWP_STEP_SIZE(LINE)
|
|
|
|
|| hasLineNumbers(method) ) ) {
|
|
|
|
/*
|
|
|
|
* We've found a suitable method in which to stop. Step
|
|
|
|
* until we reach the next safe location to complete the step->,
|
|
|
|
* and we can get rid of the method entry handler.
|
|
|
|
*/
|
|
|
|
enableStepping(thread);
|
|
|
|
if ( step->methodEnterHandlerNode != NULL ) {
|
|
|
|
(void)eventHandler_free(step->methodEnterHandlerNode);
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
jvmtiDeallocate(classname);
|
|
|
|
classname = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
completeStep(JNIEnv *env, jthread thread, StepRequest *step)
|
|
|
|
{
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We've completed a step; reset state for the next one, if any
|
|
|
|
*/
|
|
|
|
|
|
|
|
LOG_STEP(("completeStep: thread=%p", thread));
|
|
|
|
|
|
|
|
if (step->methodEnterHandlerNode != NULL) {
|
|
|
|
(void)eventHandler_free(step->methodEnterHandlerNode);
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
error = initState(env, thread, step);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
/*
|
|
|
|
* None of the initState errors should happen after one step
|
|
|
|
* has successfully completed.
|
|
|
|
*/
|
|
|
|
EXIT_ERROR(error, "initializing step state");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
jboolean
|
|
|
|
stepControl_handleStep(JNIEnv *env, jthread thread,
|
|
|
|
jclass clazz, jmethodID method)
|
|
|
|
{
|
|
|
|
jboolean completed = JNI_FALSE;
|
|
|
|
StepRequest *step;
|
|
|
|
jint currentDepth;
|
|
|
|
jint fromDepth;
|
|
|
|
jvmtiError error;
|
|
|
|
char *classname;
|
|
|
|
|
|
|
|
classname = NULL;
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_THREAD, "getting step request");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If no step is currently pending, ignore the event
|
|
|
|
*/
|
|
|
|
if (!step->pending) {
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_STEP(("stepControl_handleStep: thread=%p", thread));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We never filter step into instruction. It's always over on the
|
|
|
|
* first step event.
|
|
|
|
*/
|
|
|
|
if (step->depth == JDWP_STEP_DEPTH(INTO) &&
|
|
|
|
step->granularity == JDWP_STEP_SIZE(MIN)) {
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, into min"));
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we have left the method in which
|
|
|
|
* stepping started, the step is always complete.
|
|
|
|
*/
|
|
|
|
if (step->frameExited) {
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, frame exited"));
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Determine where we are on the call stack relative to where
|
|
|
|
* we started.
|
|
|
|
*/
|
|
|
|
currentDepth = getFrameCount(thread);
|
|
|
|
fromDepth = step->fromStackDepth;
|
|
|
|
|
|
|
|
if (fromDepth > currentDepth) {
|
|
|
|
/*
|
|
|
|
* We have returned from the caller. There are cases where
|
|
|
|
* we don't get frame pop notifications
|
|
|
|
* (e.g. stepping from opaque frames), and that's when
|
|
|
|
* this code will be reached. Complete the step->
|
|
|
|
*/
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, fromDepth>currentDepth(%d>%d)", fromDepth, currentDepth));
|
|
|
|
} else if (fromDepth < currentDepth) {
|
|
|
|
/* We have dropped into a called method. */
|
|
|
|
if ( step->depth == JDWP_STEP_DEPTH(INTO)
|
|
|
|
&& (!eventFilter_predictFiltering(step->stepHandlerNode, clazz,
|
|
|
|
(classname = getClassname(clazz))))
|
|
|
|
&& hasLineNumbers(method) ) {
|
|
|
|
|
|
|
|
/* Stepped into a method with lines, so we're done */
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, fromDepth<currentDepth(%d<%d) and into method with lines", fromDepth, currentDepth));
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* We need to continue, but don't want the overhead of step
|
|
|
|
* events from this method. So, we disable stepping and
|
|
|
|
* enable a frame pop. If we're stepping into, we also
|
|
|
|
* enable method enter events because a called frame may be
|
|
|
|
* where we want to stop.
|
|
|
|
*/
|
|
|
|
disableStepping(thread);
|
|
|
|
|
|
|
|
if (step->depth == JDWP_STEP_DEPTH(INTO)) {
|
|
|
|
step->methodEnterHandlerNode =
|
|
|
|
eventHandler_createInternalThreadOnly(
|
|
|
|
EI_METHOD_ENTRY,
|
|
|
|
handleMethodEnterEvent, thread);
|
|
|
|
if (step->methodEnterHandlerNode == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_EVENT_TYPE,
|
|
|
|
"installing event method enter handler");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
error = JVMTI_FUNC_PTR(gdata->jvmti,NotifyFramePop)
|
|
|
|
(gdata->jvmti, thread, 0);
|
|
|
|
if (error == JVMTI_ERROR_DUPLICATE) {
|
|
|
|
error = JVMTI_ERROR_NONE;
|
|
|
|
} else if (error != JVMTI_ERROR_NONE) {
|
|
|
|
EXIT_ERROR(error, "setting up notify frame pop");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
jvmtiDeallocate(classname);
|
|
|
|
classname = NULL;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* We are at the same stack depth where stepping started.
|
|
|
|
* Instruction steps are complete at this point. For line
|
|
|
|
* steps we must check to see whether we've moved to a
|
|
|
|
* different line.
|
|
|
|
*/
|
|
|
|
if (step->granularity == JDWP_STEP_SIZE(MIN)) {
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, fromDepth==currentDepth(%d) and min", fromDepth));
|
|
|
|
} else {
|
|
|
|
if (step->fromLine != -1) {
|
|
|
|
jint line = -1;
|
|
|
|
jlocation location;
|
|
|
|
jmethodID method;
|
|
|
|
WITH_LOCAL_REFS(env, 1) {
|
|
|
|
jclass clazz;
|
|
|
|
error = getFrameLocation(thread,
|
|
|
|
&clazz, &method, &location);
|
|
|
|
if ( isMethodObsolete(method)) {
|
|
|
|
method = NULL;
|
|
|
|
location = -1;
|
|
|
|
}
|
|
|
|
if (error != JVMTI_ERROR_NONE || location == -1) {
|
|
|
|
EXIT_ERROR(error, "getting frame location");
|
|
|
|
}
|
|
|
|
if ( method == step->method ) {
|
|
|
|
LOG_STEP(("stepControl_handleStep: checking line location"));
|
|
|
|
log_debugee_location("stepControl_handleStep: checking line loc",
|
|
|
|
thread, method, location);
|
|
|
|
line = findLineNumber(thread, location,
|
|
|
|
step->lineEntries, step->lineEntryCount);
|
|
|
|
}
|
|
|
|
if (line != step->fromLine) {
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, fromDepth==currentDepth(%d) and different line", fromDepth));
|
|
|
|
}
|
|
|
|
} END_WITH_LOCAL_REFS(env);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* This is a rare case. We have stepped from a location
|
|
|
|
* inside a native method to a location within a Java
|
|
|
|
* method at the same stack depth. This means that
|
|
|
|
* the original native method returned to another
|
|
|
|
* native method which, in turn, invoked a Java method.
|
|
|
|
*
|
|
|
|
* Since the original frame was native, we were unable
|
|
|
|
* to ask for a frame pop event, and, thus, could not
|
|
|
|
* set the step->frameExited flag when the original
|
|
|
|
* method was done. Instead we end up here
|
|
|
|
* and act just as though the frameExited flag was set
|
|
|
|
* and complete the step immediately.
|
|
|
|
*/
|
|
|
|
completed = JNI_TRUE;
|
|
|
|
LOG_STEP(("stepControl_handleStep: completed, fromDepth==currentDepth(%d) and no line", fromDepth));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LOG_STEP(("stepControl_handleStep: finished"));
|
|
|
|
}
|
|
|
|
done:
|
|
|
|
if (completed) {
|
|
|
|
completeStep(env, thread, step);
|
|
|
|
}
|
|
|
|
stepControl_unlock();
|
|
|
|
return completed;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
stepControl_initialize(void)
|
|
|
|
{
|
|
|
|
stepLock = debugMonitorCreate("JDWP Step Handler Lock");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
stepControl_reset(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Reset step control request stack depth and line number.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
stepControl_resetRequest(jthread thread)
|
|
|
|
{
|
|
|
|
|
|
|
|
StepRequest *step;
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
LOG_STEP(("stepControl_resetRequest: thread=%p", thread));
|
|
|
|
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
|
|
|
|
if (step != NULL) {
|
|
|
|
JNIEnv *env;
|
|
|
|
env = getEnv();
|
|
|
|
error = initState(env, thread, step);
|
|
|
|
if (error != JVMTI_ERROR_NONE) {
|
|
|
|
EXIT_ERROR(error, "initializing step state");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_THREAD, "getting step request");
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
initEvents(jthread thread, StepRequest *step)
|
|
|
|
{
|
|
|
|
/* Need to install frame pop handler and exception catch handler when
|
|
|
|
* single-stepping is enabled (i.e. step-into or step-over/step-out
|
|
|
|
* when fromStackDepth > 0).
|
|
|
|
*/
|
|
|
|
if (step->depth == JDWP_STEP_DEPTH(INTO) || step->fromStackDepth > 0) {
|
|
|
|
/*
|
|
|
|
* TO DO: These might be able to applied more selectively to
|
|
|
|
* boost performance.
|
|
|
|
*/
|
|
|
|
step->catchHandlerNode = eventHandler_createInternalThreadOnly(
|
|
|
|
EI_EXCEPTION_CATCH,
|
|
|
|
handleExceptionCatchEvent,
|
|
|
|
thread);
|
|
|
|
step->framePopHandlerNode = eventHandler_createInternalThreadOnly(
|
|
|
|
EI_FRAME_POP,
|
|
|
|
handleFramePopEvent,
|
|
|
|
thread);
|
|
|
|
|
|
|
|
if (step->catchHandlerNode == NULL ||
|
|
|
|
step->framePopHandlerNode == NULL) {
|
|
|
|
EXIT_ERROR(AGENT_ERROR_INVALID_EVENT_TYPE,
|
|
|
|
"installing step event handlers");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Initially enable stepping:
|
|
|
|
* 1) For step into, always
|
|
|
|
* 2) For step over, unless right after the VM_INIT.
|
|
|
|
* Enable stepping for STEP_MIN or STEP_LINE with or without line numbers.
|
|
|
|
* If the class is redefined then non EMCP methods may not have line
|
|
|
|
* number info. So enable line stepping for non line number so that it
|
|
|
|
* behaves like STEP_MIN/STEP_OVER.
|
|
|
|
* 3) For step out, only if stepping from native, except right after VM_INIT
|
|
|
|
*
|
|
|
|
* (right after VM_INIT, a step->over or out is identical to running
|
|
|
|
* forever)
|
|
|
|
*/
|
|
|
|
switch (step->depth) {
|
|
|
|
case JDWP_STEP_DEPTH(INTO):
|
|
|
|
enableStepping(thread);
|
|
|
|
break;
|
|
|
|
case JDWP_STEP_DEPTH(OVER):
|
|
|
|
if (step->fromStackDepth > 0 && !step->fromNative ) {
|
|
|
|
enableStepping(thread);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case JDWP_STEP_DEPTH(OUT):
|
|
|
|
if (step->fromNative &&
|
|
|
|
(step->fromStackDepth > 0)) {
|
|
|
|
enableStepping(thread);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
JDI_ASSERT(JNI_FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
jvmtiError
|
|
|
|
stepControl_beginStep(JNIEnv *env, jthread thread, jint size, jint depth,
|
|
|
|
HandlerNode *node)
|
|
|
|
{
|
|
|
|
StepRequest *step;
|
|
|
|
jvmtiError error;
|
|
|
|
jvmtiError error2;
|
|
|
|
|
|
|
|
LOG_STEP(("stepControl_beginStep: thread=%p,size=%d,depth=%d",
|
|
|
|
thread, size, depth));
|
|
|
|
|
|
|
|
eventHandler_lock(); /* for proper lock order */
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step == NULL) {
|
|
|
|
error = AGENT_ERROR_INVALID_THREAD;
|
|
|
|
/* Normally not getting a StepRequest struct pointer is a fatal error
|
|
|
|
* but on a beginStep, we just return an error code.
|
|
|
|
*/
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* In case the thread isn't already suspended, do it again.
|
|
|
|
*/
|
|
|
|
error = threadControl_suspendThread(thread, JNI_FALSE);
|
|
|
|
if (error == JVMTI_ERROR_NONE) {
|
|
|
|
/*
|
|
|
|
* Overwrite any currently executing step.
|
|
|
|
*/
|
|
|
|
step->granularity = size;
|
|
|
|
step->depth = depth;
|
|
|
|
step->catchHandlerNode = NULL;
|
|
|
|
step->framePopHandlerNode = NULL;
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
step->stepHandlerNode = node;
|
|
|
|
error = initState(env, thread, step);
|
|
|
|
if (error == JVMTI_ERROR_NONE) {
|
|
|
|
initEvents(thread, step);
|
|
|
|
}
|
|
|
|
/* false means it is not okay to unblock the commandLoop thread */
|
|
|
|
error2 = threadControl_resumeThread(thread, JNI_FALSE);
|
|
|
|
if (error2 != JVMTI_ERROR_NONE && error == JVMTI_ERROR_NONE) {
|
|
|
|
error = error2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If everything went ok, indicate a step is pending.
|
|
|
|
*/
|
|
|
|
if (error == JVMTI_ERROR_NONE) {
|
|
|
|
step->pending = JNI_TRUE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
EXIT_ERROR(error, "stepControl_beginStep: cannot suspend thread");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
eventHandler_unlock();
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
clearStep(jthread thread, StepRequest *step)
|
|
|
|
{
|
|
|
|
if (step->pending) {
|
|
|
|
|
|
|
|
disableStepping(thread);
|
|
|
|
if ( step->catchHandlerNode != NULL ) {
|
|
|
|
(void)eventHandler_free(step->catchHandlerNode);
|
|
|
|
step->catchHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
if ( step->framePopHandlerNode!= NULL ) {
|
|
|
|
(void)eventHandler_free(step->framePopHandlerNode);
|
|
|
|
step->framePopHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
if ( step->methodEnterHandlerNode != NULL ) {
|
|
|
|
(void)eventHandler_free(step->methodEnterHandlerNode);
|
|
|
|
step->methodEnterHandlerNode = NULL;
|
|
|
|
}
|
|
|
|
step->pending = JNI_FALSE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Warning: Do not clear step->method, step->lineEntryCount,
|
|
|
|
* or step->lineEntries here, they will likely
|
|
|
|
* be needed on the next step.
|
|
|
|
*/
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
jvmtiError
|
|
|
|
stepControl_endStep(jthread thread)
|
|
|
|
{
|
|
|
|
StepRequest *step;
|
|
|
|
jvmtiError error;
|
|
|
|
|
|
|
|
LOG_STEP(("stepControl_endStep: thread=%p", thread));
|
|
|
|
|
|
|
|
eventHandler_lock(); /* for proper lock order */
|
|
|
|
stepControl_lock();
|
|
|
|
|
|
|
|
step = threadControl_getStepRequest(thread);
|
|
|
|
if (step != NULL) {
|
|
|
|
clearStep(thread, step);
|
|
|
|
error = JVMTI_ERROR_NONE;
|
|
|
|
} else {
|
|
|
|
/* If the stepRequest can't be gotten, then this thread no longer
|
|
|
|
* exists, just return, don't die here, this is normal at
|
|
|
|
* termination time. Return JVMTI_ERROR_NONE so the thread Ref
|
|
|
|
* can be tossed.
|
|
|
|
*/
|
|
|
|
error = JVMTI_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
stepControl_unlock();
|
|
|
|
eventHandler_unlock();
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
stepControl_clearRequest(jthread thread, StepRequest *step)
|
|
|
|
{
|
|
|
|
LOG_STEP(("stepControl_clearRequest: thread=%p", thread));
|
|
|
|
clearStep(thread, step);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
stepControl_lock(void)
|
|
|
|
{
|
|
|
|
debugMonitorEnter(stepLock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
stepControl_unlock(void)
|
|
|
|
{
|
|
|
|
debugMonitorExit(stepLock);
|
|
|
|
}
|