/******************************************************************************* * Copyright (c) 2011 Wind River Systems, Inc. and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Wind River Systems - initial API and implementation *******************************************************************************/ package org.eclipse.tm.internal.tcf.debug.ui.commands; import java.util.Map; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Status; import org.eclipse.debug.core.commands.IDebugCommandRequest; import org.eclipse.debug.core.commands.IDropToFrameHandler; import org.eclipse.debug.core.commands.IEnabledStateRequest; import org.eclipse.tm.internal.tcf.debug.actions.TCFActionStepOut; import org.eclipse.tm.internal.tcf.debug.model.TCFContextState; import org.eclipse.tm.internal.tcf.debug.ui.Activator; import org.eclipse.tm.internal.tcf.debug.ui.model.TCFModel; import org.eclipse.tm.internal.tcf.debug.ui.model.TCFNode; import org.eclipse.tm.internal.tcf.debug.ui.model.TCFNodeExecContext; import org.eclipse.tm.internal.tcf.debug.ui.model.TCFNodeStackFrame; import org.eclipse.tm.internal.tcf.debug.ui.model.TCFRunnable; import org.eclipse.tm.tcf.protocol.IChannel; import org.eclipse.tm.tcf.services.IBreakpoints; import org.eclipse.tm.tcf.services.IRunControl; import org.eclipse.tm.tcf.services.IRunControl.RunControlContext; import org.eclipse.tm.tcf.services.IStackTrace.StackTraceContext; import org.eclipse.tm.tcf.util.TCFDataCache; /** * Drop-to-frame command handler for TCF. */ public class DropToFrameCommand implements IDropToFrameHandler { private static class StepStateMachine extends TCFActionStepOut { private final IDebugCommandRequest monitor; private final Runnable done; private final TCFNodeExecContext node; private final TCFNodeStackFrame frame; StepStateMachine(TCFModel model, IDebugCommandRequest monitor, IRunControl.RunControlContext ctx, TCFNodeStackFrame frame, Runnable done) { super(model.getLaunch(), ctx, false); this.monitor = monitor; this.done = done; this.frame = frame; this.node = (TCFNodeExecContext)frame.getParent(); } @Override protected TCFDataCache<TCFContextState> getContextState() { return node.getState(); } @Override protected TCFDataCache<StackTraceContext> getStackFrame() { return frame.getStackTraceContext(); } @Override protected int getStackFrameIndex() { return frame.getFrameNo(); } @Override protected TCFDataCache<?> getStackTrace() { return node.getStackTrace(); } protected void exit(Throwable error) { exit(error, "Drop To Frame"); } @Override protected void exit(Throwable error, String reason) { if (exited) return; super.exit(error, reason); if (error != null && node.getChannel().getState() == IChannel.STATE_OPEN) { monitor.setStatus(new Status(IStatus.ERROR, Activator.PLUGIN_ID, 0, "Cannot drop to frame: " + error.getLocalizedMessage(), error)); } if (aborted) { monitor.setStatus(Status.CANCEL_STATUS); } done.run(); } } private final TCFModel model; public DropToFrameCommand(TCFModel tcfModel) { model = tcfModel; } public void canExecute(final IEnabledStateRequest request) { final Object[] elements = request.getElements(); if (elements.length != 1 || !(elements[0] instanceof TCFNodeStackFrame)) { request.setEnabled(false); request.done(); return; } final TCFNodeStackFrame frameNode = (TCFNodeStackFrame) elements[0]; new TCFRunnable(request) { public void run() { if (frameNode.getFrameNo() < 1) { request.setEnabled(false); done(); return; } TCFNodeExecContext exeNode = (TCFNodeExecContext) frameNode.getParent(); TCFDataCache<IRunControl.RunControlContext> ctx_cache = exeNode.getRunContext(); if (!ctx_cache.validate(this)) { return; } IRunControl.RunControlContext ctx = ctx_cache.getData(); if (!canStepOut(ctx)) { request.setEnabled(false); done(); return; } int action_cnt = model.getLaunch().getContextActionsCount(ctx.getID()); if (action_cnt > 0 || !canStepOut(ctx)) { request.setEnabled(false); done(); return; } TCFDataCache<TCFContextState> state_cache = exeNode.getState(); if (!state_cache.validate(this)) { return; } TCFContextState state_data = state_cache.getData(); request.setEnabled(state_data != null && state_data.is_suspended); done(); } private boolean canStepOut(RunControlContext ctx) { if (ctx == null) return false; if (ctx.canResume(IRunControl.RM_STEP_OUT)) return true; if (!ctx.hasState()) return false; if (ctx.canResume(IRunControl.RM_RESUME) && model.getLaunch().getService(IBreakpoints.class) != null) return true; return false; } }; } public boolean execute(final IDebugCommandRequest request) { final Object[] elements = request.getElements(); if (elements.length != 1 || !(elements[0] instanceof TCFNodeStackFrame)) { request.setStatus(Status.CANCEL_STATUS); request.done(); return false; } final TCFNodeStackFrame frameNode = (TCFNodeStackFrame) elements[0]; new TCFRunnable(request) { public void run() { int frameNo = frameNode.getFrameNo(); if (frameNo < 1) { request.setStatus(Status.CANCEL_STATUS); done(); return; } TCFNodeExecContext exeNode = (TCFNodeExecContext) frameNode.getParent(); TCFDataCache<IRunControl.RunControlContext> ctx_cache = exeNode.getRunContext(); if (!ctx_cache.validate(this)) return; TCFDataCache<TCFContextState> state_cache = exeNode.getState(); if (!state_cache.validate(this)) return; TCFContextState state_data = state_cache.getData(); if (state_data == null || !state_data.is_suspended) { request.setStatus(Status.CANCEL_STATUS); done(); } if (!exeNode.getStackTrace().validate(this)) return; Map<String, TCFNode> stack = exeNode.getStackTrace().getData(); for (TCFNode node : stack.values()) { TCFNodeStackFrame frame_to_step_out = (TCFNodeStackFrame) node; if (frame_to_step_out.getFrameNo() == frameNo - 1) { new StepStateMachine(model, request, ctx_cache.getData(), frame_to_step_out, new Runnable() { public void run() { request.done(); } }); return; } } request.setStatus(Status.CANCEL_STATUS); done(); } }; return false; } }