1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
/*******************************************************************************
* Copyright (c) 2007, 2010 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.HashSet;
import java.util.Set;
import org.eclipse.core.runtime.Status;
import org.eclipse.debug.core.commands.IDebugCommandHandler;
import org.eclipse.debug.core.commands.IDebugCommandRequest;
import org.eclipse.debug.core.commands.IEnabledStateRequest;
import org.eclipse.tm.internal.tcf.debug.model.TCFContextState;
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.TCFRunnable;
import org.eclipse.tm.tcf.services.IRunControl;
import org.eclipse.tm.tcf.util.TCFDataCache;
abstract class StepCommand implements IDebugCommandHandler {
private static final int MAX_ACTION_CNT = 4;
protected final TCFModel model;
public StepCommand(TCFModel model) {
this.model = model;
}
protected abstract boolean canExecute(IRunControl.RunControlContext ctx);
protected abstract void execute(IDebugCommandRequest monitor,
IRunControl.RunControlContext ctx, boolean src_step, Runnable done);
private boolean getContextSet(boolean exec, Object[] elements, Set<IRunControl.RunControlContext> set, Runnable done) {
for (int i = 0; i < elements.length; i++) {
TCFNode node = null;
if (elements[i] instanceof TCFNode) node = (TCFNode)elements[i];
else node = model.getRootNode();
while (node != null && !node.isDisposed()) {
IRunControl.RunControlContext ctx = null;
if (node instanceof TCFNodeExecContext) {
TCFDataCache<IRunControl.RunControlContext> cache = ((TCFNodeExecContext)node).getRunContext();
if (!cache.validate(done)) return false;
ctx = cache.getData();
}
if (ctx == null || !canExecute(ctx)) {
node = node.getParent();
}
else {
int action_cnt = model.getLaunch().getContextActionsCount(ctx.getID());
if (exec && action_cnt >= MAX_ACTION_CNT) break;
if (action_cnt == 0 && !ctx.isContainer()) {
TCFDataCache<TCFContextState> state_cache = ((TCFNodeExecContext)node).getState();
if (!state_cache.validate(done)) return false;
TCFContextState state_data = state_cache.getData();
if (state_data != null && state_data.is_suspended) set.add(ctx);
}
else {
set.add(ctx);
}
break;
}
}
}
return true;
}
public final void canExecute(final IEnabledStateRequest monitor) {
new TCFRunnable(monitor) {
public void run() {
if (done) return;
if (!monitor.isCanceled()) {
Set<IRunControl.RunControlContext> set = new HashSet<IRunControl.RunControlContext>();
if (!getContextSet(false, monitor.getElements(), set, this)) return;
monitor.setEnabled(set.size() > 0);
monitor.setStatus(Status.OK_STATUS);
}
done();
}
};
}
public final boolean execute(final IDebugCommandRequest monitor) {
new TCFRunnable(monitor) {
public void run() {
if (done) return;
Set<IRunControl.RunControlContext> set = new HashSet<IRunControl.RunControlContext>();
if (!getContextSet(true, monitor.getElements(), set, this)) return;
if (set.size() == 0) {
monitor.setStatus(Status.OK_STATUS);
monitor.done();
}
else {
final Set<Runnable> wait_list = new HashSet<Runnable>();
for (IRunControl.RunControlContext ctx : set) {
Runnable done = new Runnable() {
public void run() {
wait_list.remove(this);
if (wait_list.isEmpty()) monitor.done();
}
};
wait_list.add(done);
execute(monitor, ctx, !model.isInstructionSteppingEnabled(), done);
}
}
}
};
return true;
}
}
|