| /******************************************************************************* |
| * Copyright (c) 2005, 2006 IBM Corporation 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: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.debug.internal.ui.viewers.update; |
| |
| import java.util.HashMap; |
| import java.util.LinkedHashSet; |
| import java.util.Map; |
| import java.util.Set; |
| |
| import org.eclipse.debug.core.DebugEvent; |
| import org.eclipse.debug.core.DebugException; |
| import org.eclipse.debug.core.DebugPlugin; |
| import org.eclipse.debug.core.model.IStackFrame; |
| import org.eclipse.debug.core.model.IThread; |
| import org.eclipse.debug.internal.ui.viewers.provisional.AbstractModelProxy; |
| import org.eclipse.debug.internal.ui.viewers.provisional.IModelDelta; |
| import org.eclipse.debug.internal.ui.viewers.provisional.ModelDelta; |
| |
| /** |
| * @since 3.2 |
| */ |
| public class ThreadEventHandler extends DebugEventHandler { |
| |
| /** |
| * Queue of suspended threads to choose from when needing |
| * to select a thread when another is resumed. Threads |
| * are added in the order they suspend. |
| */ |
| private Set fThreadQueue = new LinkedHashSet(); |
| |
| /** |
| * Map of previous TOS per thread |
| */ |
| private Map fLastTopFrame = new HashMap(); |
| /** |
| * Constructs and event handler for a threads in the given viewer. |
| * |
| * @param viewer |
| */ |
| public ThreadEventHandler(AbstractModelProxy proxy) { |
| super(proxy); |
| } |
| |
| /* (non-Javadoc) |
| * @see org.eclipse.debug.internal.ui.viewers.update.DebugEventHandler#dispose() |
| */ |
| public synchronized void dispose() { |
| fLastTopFrame.clear(); |
| fThreadQueue.clear(); |
| super.dispose(); |
| } |
| |
| protected void handleSuspend(DebugEvent event) { |
| IThread thread = (IThread) event.getSource(); |
| if (event.isEvaluation()) { |
| ModelDelta delta = buildRootDelta(); |
| ModelDelta node = addPathToThread(delta, thread); |
| node = node.addNode(thread, IModelDelta.NO_CHANGE); |
| try { |
| IStackFrame frame = thread.getTopStackFrame(); |
| if (frame != null) { |
| node.addNode(frame, IModelDelta.STATE); |
| fireDelta(delta); |
| } |
| } catch (DebugException e) { |
| } |
| } else { |
| queueSuspendedThread(event); |
| int extras = IModelDelta.STATE; |
| if (event.getDetail() == DebugEvent.BREAKPOINT | event.getDetail() == DebugEvent.CLIENT_REQUEST) { |
| extras = IModelDelta.EXPAND; |
| } |
| fireDeltaUpdatingTopFrame(thread, IModelDelta.NO_CHANGE | extras); |
| } |
| } |
| |
| private boolean isEqual(Object o1, Object o2) { |
| if (o1 == o2) { |
| return true; |
| } |
| if (o1 == null) { |
| return false; |
| } |
| return o1.equals(o2); |
| } |
| |
| protected void handleResume(DebugEvent event) { |
| IThread thread = removeSuspendedThread(event); |
| fireDeltaAndClearTopFrame(thread, IModelDelta.STATE | IModelDelta.CONTENT); |
| thread = getNextSuspendedThread(); |
| if (thread != null) { |
| fireDeltaUpdatingTopFrame(thread, IModelDelta.NO_CHANGE); |
| } |
| } |
| |
| protected void handleCreate(DebugEvent event) { |
| fireDeltaAndClearTopFrame((IThread) event.getSource(), IModelDelta.ADDED | IModelDelta.STATE); |
| } |
| |
| protected void handleTerminate(DebugEvent event) { |
| fireDeltaAndClearTopFrame((IThread) event.getSource(), IModelDelta.REMOVED); |
| } |
| |
| protected void handleChange(DebugEvent event) { |
| if (event.getDetail() == DebugEvent.STATE) { |
| fireDeltaUpdatingThread((IThread) event.getSource(), IModelDelta.STATE); |
| } else { |
| fireDeltaUpdatingThread((IThread) event.getSource(), IModelDelta.CONTENT); |
| } |
| } |
| |
| protected void handleLateSuspend(DebugEvent suspend, DebugEvent resume) { |
| IThread thread = queueSuspendedThread(suspend); |
| if (suspend.isEvaluation() && suspend.getDetail() == DebugEvent.EVALUATION_IMPLICIT) { |
| // late implicit evaluation - update thread and frame |
| ModelDelta delta = buildRootDelta(); |
| ModelDelta node = addPathToThread(delta, thread); |
| node = node.addNode(thread, IModelDelta.STATE); |
| try { |
| IStackFrame frame = thread.getTopStackFrame(); |
| if (frame != null) { |
| node.addNode(frame, IModelDelta.STATE); |
| fireDelta(delta); |
| } |
| } catch (DebugException e) { |
| } |
| } else { |
| fireDeltaUpdatingTopFrame(thread, IModelDelta.CONTENT | IModelDelta.EXPAND); |
| } |
| } |
| |
| protected void handleSuspendTimeout(DebugEvent event) { |
| IThread thread = removeSuspendedThread(event); |
| if (event.isEvaluation() && event.getDetail() == DebugEvent.EVALUATION_IMPLICIT) { |
| // don't collapse thread when waiting for implicit eval to complete |
| fireDeltaUpdatingThread(thread, IModelDelta.STATE); |
| } else { |
| fireDeltaAndClearTopFrame(thread, IModelDelta.CONTENT); |
| } |
| } |
| |
| private ModelDelta buildRootDelta() { |
| return new ModelDelta(DebugPlugin.getDefault().getLaunchManager(), IModelDelta.NO_CHANGE); |
| } |
| |
| protected ModelDelta addPathToThread(ModelDelta delta, IThread thread) { |
| delta = delta.addNode(thread.getLaunch(), IModelDelta.NO_CHANGE); |
| return delta.addNode(thread.getDebugTarget(), IModelDelta.NO_CHANGE); |
| } |
| |
| private void fireDeltaAndClearTopFrame(IThread thread, int flags) { |
| ModelDelta delta = buildRootDelta(); |
| ModelDelta node = addPathToThread(delta, thread); |
| node.addNode(thread, flags); |
| synchronized (this) { |
| fLastTopFrame.remove(thread); |
| } |
| fireDelta(delta); |
| } |
| |
| private void fireDeltaUpdatingTopFrame(IThread thread, int flags) { |
| ModelDelta delta = buildRootDelta(); |
| ModelDelta node = addPathToThread(delta, thread); |
| IStackFrame prev = null; |
| synchronized (this) { |
| prev = (IStackFrame) fLastTopFrame.get(thread); |
| } |
| IStackFrame frame = null; |
| try { |
| frame = thread.getTopStackFrame(); |
| } catch (DebugException e) { |
| } |
| if (isEqual(frame, prev)) { |
| if (frame == null) { |
| if (thread.isSuspended()) { |
| // no frames, but suspended - update & select |
| node = node.addNode(thread, flags | IModelDelta.STATE | IModelDelta.SELECT); |
| } |
| } else { |
| node = node.addNode(thread, flags); |
| } |
| } else { |
| node = node.addNode(thread, flags | IModelDelta.CONTENT); |
| } |
| if (frame != null) { |
| node.addNode(frame, IModelDelta.STATE | IModelDelta.SELECT); |
| } |
| synchronized (this) { |
| if (!isDisposed()) { |
| fLastTopFrame.put(thread, frame); |
| } |
| } |
| fireDelta(delta); |
| } |
| |
| private void fireDeltaUpdatingThread(IThread thread, int flags) { |
| ModelDelta delta = buildRootDelta(); |
| ModelDelta node = addPathToThread(delta, thread); |
| node = node.addNode(thread, flags); |
| fireDelta(delta); |
| } |
| |
| protected boolean handlesEvent(DebugEvent event) { |
| return event.getSource() instanceof IThread; |
| } |
| |
| protected synchronized IThread queueSuspendedThread(DebugEvent event) { |
| IThread thread = (IThread) event.getSource(); |
| if (!isDisposed()) { |
| fThreadQueue.add(thread); |
| } |
| return thread; |
| } |
| |
| protected synchronized IThread removeSuspendedThread(DebugEvent event) { |
| IThread thread = (IThread)event.getSource(); |
| fThreadQueue.remove(thread); |
| return thread; |
| } |
| |
| protected synchronized IThread getNextSuspendedThread() { |
| if (!fThreadQueue.isEmpty()) { |
| return (IThread) fThreadQueue.iterator().next(); |
| } |
| return null; |
| } |
| |
| } |