Skip to content

Commit

Permalink
Merge pull request #3990 from edumunoz/edumunoz/stop-all-debug
Browse files Browse the repository at this point in the history
Fixes #3650: Support stop-all-threads mode debugging for multi-thread…
  • Loading branch information
isidorn committed Mar 17, 2016
2 parents 8f2f9cc + 9af920e commit 5d620ea
Show file tree
Hide file tree
Showing 9 changed files with 509 additions and 36 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -134,18 +134,18 @@ export class DebugEditorModelManager implements IWorkbenchContribution {
const result: editorcommon.IModelDeltaDecoration[] = [];
const focusedStackFrame = this.debugService.getViewModel().getFocusedStackFrame();
const allThreads = this.debugService.getModel().getThreads();
if (!focusedStackFrame || !allThreads[focusedStackFrame.threadId] || !allThreads[focusedStackFrame.threadId].callStack) {
if (!focusedStackFrame || !allThreads[focusedStackFrame.threadId] || !allThreads[focusedStackFrame.threadId].getCachedCallStack()) {
return result;
}

// only show decorations for the currently focussed thread.
const thread = allThreads[focusedStackFrame.threadId];
thread.callStack.filter(sf => sf.source.uri.toString() === modelUrlStr).forEach(sf => {
thread.getCachedCallStack().filter(sf => sf.source.uri.toString() === modelUrlStr).forEach(sf => {
const wholeLineRange = createRange(sf.lineNumber, sf.column, sf.lineNumber, Number.MAX_VALUE);

// compute how to decorate the editor. Different decorations are used if this is a top stack frame, focussed stack frame,
// an exception or a stack frame that did not change the line number (we only decorate the columns, not the whole line).
if (sf === thread.callStack[0]) {
if (sf === thread.getCachedCallStack()[0]) {
result.push({
options: DebugEditorModelManager.TOP_STACK_FRAME_MARGIN,
range: createRange(sf.lineNumber, sf.column, sf.lineNumber, sf.column + 1)
Expand Down
8 changes: 6 additions & 2 deletions src/vs/workbench/parts/debug/browser/debugViewer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,10 @@ export class BaseDebugController extends treedefaults.DefaultController {

export class CallStackDataSource implements tree.IDataSource {

constructor(@debug.IDebugService private debugService: debug.IDebugService) {
// noop
}

public getId(tree: tree.ITree, element: any): string {
return element.getId();
}
Expand All @@ -199,7 +203,7 @@ export class CallStackDataSource implements tree.IDataSource {

public getChildren(tree: tree.ITree, element: any): TPromise<any> {
if (element instanceof model.Thread) {
return TPromise.as((<model.Thread> element).callStack);
return (<model.Thread> element).getCallStack(this.debugService);
}

const threads = (<model.Model> element).getThreads();
Expand All @@ -209,7 +213,7 @@ export class CallStackDataSource implements tree.IDataSource {
});

if (threadsArray.length === 1) {
return TPromise.as(threadsArray[0].callStack);
return threadsArray[0].getCallStack(this.debugService);
} else {
return TPromise.as(threadsArray);
}
Expand Down
12 changes: 8 additions & 4 deletions src/vs/workbench/parts/debug/browser/debugViewlet.ts
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ class VariablesView extends viewlet.CollapsibleViewletView {
this.toDispose.push(this.debugService.addListener2(debug.ServiceEvents.STATE_CHANGED, () => {
collapseAction.enabled = this.debugService.getState() === debug.State.Running || this.debugService.getState() === debug.State.Stopped;
}));

this.toDispose.push(this.tree.addListener2(events.EventType.FOCUS, (e: tree.IFocusEvent) => {
const isMouseClick = (e.payload && e.payload.origin === 'mouse');
const isVariableType = (e.focus instanceof model.Variable);
Expand Down Expand Up @@ -226,7 +226,7 @@ class CallStackView extends viewlet.CollapsibleViewletView {
this.treeContainer = renderViewTree(container);

this.tree = new treeimpl.Tree(this.treeContainer, {
dataSource: new viewer.CallStackDataSource(),
dataSource: this.instantiationService.createInstance(viewer.CallStackDataSource),
renderer: this.instantiationService.createInstance(viewer.CallStackRenderer),
accessibilityProvider: this.instantiationService.createInstance(viewer.CallstackAccessibilityProvider)
}, debugTreeOptions(nls.localize('callStackAriaLabel', "Debug Call Stack")));
Expand Down Expand Up @@ -275,7 +275,7 @@ class CallStackView extends viewlet.CollapsibleViewletView {

this.toDispose.push(this.debugService.getViewModel().addListener2(debug.ViewModelEvents.FOCUSED_STACK_FRAME_UPDATED, () => {
const focussedThread = this.debugService.getModel().getThreads()[this.debugService.getViewModel().getFocusedThreadId()];
if (focussedThread && focussedThread.stoppedDetails && focussedThread.stoppedDetails.reason !== 'step') {
if (focussedThread && focussedThread.stoppedDetails && focussedThread.stoppedDetails.reason && focussedThread.stoppedDetails.reason !== 'step') {
this.pauseMessageLabel.text(nls.localize('debugStopped', "Paused on {0}", focussedThread.stoppedDetails.reason));
if (focussedThread.stoppedDetails.text) {
this.pauseMessageLabel.title(focussedThread.stoppedDetails.text);
Expand All @@ -292,7 +292,11 @@ class CallStackView extends viewlet.CollapsibleViewletView {
if (focused) {
const threads = this.debugService.getModel().getThreads();
for (let ref in threads) {
if (threads[ref].callStack.some(sf => sf === focused)) {
// Only query for threads whose callstacks are already available
// so that we don't perform unnecessary queries to the
// debug adapter. If it's a thread we need to expand, its
// callstack would have already been populated already
if (threads[ref].getCachedCallStack() && threads[ref].getCachedCallStack().some(sf => sf === focused)) {
this.tree.expand(threads[ref]);
}
}
Expand Down
27 changes: 26 additions & 1 deletion src/vs/workbench/parts/debug/common/debug.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ export interface IRawModelUpdate {
thread?: DebugProtocol.Thread;
callStack?: DebugProtocol.StackFrame[];
stoppedDetails?: IRawStoppedDetails;
allThreadsStopped?: boolean;
}

export interface IRawStoppedDetails {
Expand Down Expand Up @@ -53,8 +54,31 @@ export interface IExpression extends ITreeElement, IExpressionContainer {
export interface IThread extends ITreeElement {
threadId: number;
name: string;
callStack: IStackFrame[];
stoppedDetails: IRawStoppedDetails;

/**
* Queries the debug adapter for the callstack and returns a promise with
* the stack frames of the callstack.
* If the thread is not stopped, it returns a promise to an empty array.
*/
getCallStack(debugService: IDebugService): TPromise<IStackFrame[]>;

/**
* Gets the callstack if it has already been received from the debug
* adapter, otherwise it returns undefined.
*/
getCachedCallStack(): IStackFrame[];

/**
* Invalidates the callstack cache
*/
clearCallStack(): void;

/**
* Indicates whether this thread is stopped. The callstack for stopped
* threads can be retrieved from the debug adapter.
*/
stopped: boolean;
}

export interface IScope extends IExpressionContainer {
Expand Down Expand Up @@ -229,6 +253,7 @@ export interface IRawDebugSession extends ee.EventEmitter {
continue(args: DebugProtocol.ContinueArguments): TPromise<DebugProtocol.ContinueResponse>;
pause(args: DebugProtocol.PauseArguments): TPromise<DebugProtocol.PauseResponse>;

stackTrace(args: DebugProtocol.StackTraceArguments): TPromise<DebugProtocol.StackTraceResponse>;
scopes(args: DebugProtocol.ScopesArguments): TPromise<DebugProtocol.ScopesResponse>;
variables(args: DebugProtocol.VariablesArguments): TPromise<DebugProtocol.VariablesResponse>;
evaluate(args: DebugProtocol.EvaluateArguments): TPromise<DebugProtocol.EvaluateResponse>;
Expand Down
97 changes: 79 additions & 18 deletions src/vs/workbench/parts/debug/common/debugModel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import severity from 'vs/base/common/severity';
import types = require('vs/base/common/types');
import arrays = require('vs/base/common/arrays');
import debug = require('vs/workbench/parts/debug/common/debug');
import errors = require('vs/base/common/errors');
import { Source } from 'vs/workbench/parts/debug/common/debugSource';

const MAX_REPL_LENGTH = 10000;
Expand Down Expand Up @@ -92,16 +93,57 @@ export function getFullExpressionName(expression: debug.IExpression, sessionType
}

export class Thread implements debug.IThread {

private promisedCallStack: TPromise<debug.IStackFrame[]>;
private cachedCallStack: debug.IStackFrame[];
public stoppedDetails: debug.IRawStoppedDetails;
public stopped: boolean;

constructor(public name: string, public threadId, public callStack: debug.IStackFrame[]) {
constructor(public name: string, public threadId) {
this.promisedCallStack = undefined;
this.stoppedDetails = undefined;
this.cachedCallStack = undefined;
this.stopped = false;
}

public getId(): string {
return `thread:${ this.name }:${ this.threadId }`;
}

public clearCallStack(): void {
this.promisedCallStack = undefined;
this.cachedCallStack = undefined;
}

public getCachedCallStack(): debug.IStackFrame[] {
return this.cachedCallStack;
}

public getCallStack(debugService: debug.IDebugService): TPromise<debug.IStackFrame[]> {
if (!this.stopped) {
return TPromise.as([]);
}
if (!this.promisedCallStack) {
this.promisedCallStack = this.getCallStackImpl(debugService);
this.promisedCallStack.then(result => {
this.cachedCallStack = result;
}, errors.onUnexpectedError);
}

return this.promisedCallStack;
}

private getCallStackImpl(debugService: debug.IDebugService): TPromise<debug.IStackFrame[]> {
let session = debugService.getActiveSession();
return session.stackTrace({ threadId: this.threadId, levels: 20 }).then(response => {
return response.body.stackFrames.map((rsf, level) => {
if (!rsf) {
return new StackFrame(this.threadId, 0, new Source({ name: 'unknown' }), nls.localize('unknownStack', "Unknown stack location"), undefined, undefined);
}

return new StackFrame(this.threadId, rsf.id, rsf.source ? new Source(rsf.source) : new Source({ name: 'unknown' }), rsf.name, rsf.line, rsf.column);
});
});
}
}

export class OutputElement implements debug.ITreeElement {
Expand Down Expand Up @@ -360,7 +402,7 @@ export class Model extends ee.EventEmitter implements debug.IModel {
if (removeThreads) {
delete this.threads[reference];
} else {
this.threads[reference].callStack = [];
this.threads[reference].clearCallStack();
this.threads[reference].stoppedDetails = undefined;
}
} else {
Expand All @@ -370,7 +412,7 @@ export class Model extends ee.EventEmitter implements debug.IModel {
} else {
for (let ref in this.threads) {
if (this.threads.hasOwnProperty(ref)) {
this.threads[ref].callStack = [];
this.threads[ref].clearCallStack();
this.threads[ref].stoppedDetails = undefined;
}
}
Expand All @@ -380,6 +422,16 @@ export class Model extends ee.EventEmitter implements debug.IModel {
this.emit(debug.ModelEvents.CALLSTACK_UPDATED);
}

public continueThreads(): void {
for (let ref in this.threads) {
if (this.threads.hasOwnProperty(ref)) {
this.threads[ref].stopped = false;
}
}

this.clearThreads(false);
}

public getBreakpoints(): debug.IBreakpoint[] {
return this.breakpoints;
}
Expand Down Expand Up @@ -624,33 +676,42 @@ export class Model extends ee.EventEmitter implements debug.IModel {

public sourceIsUnavailable(source: Source): void {
Object.keys(this.threads).forEach(key => {
this.threads[key].callStack.forEach(stackFrame => {
if (stackFrame.source.uri.toString() === source.uri.toString()) {
stackFrame.source.available = false;
}
});
if (this.threads[key].getCachedCallStack()) {
this.threads[key].getCachedCallStack().forEach(stackFrame => {
if (stackFrame.source.uri.toString() === source.uri.toString()) {
stackFrame.source.available = false;
}
});
}
});

this.emit(debug.ModelEvents.CALLSTACK_UPDATED);
}

public rawUpdate(data: debug.IRawModelUpdate): void {
if (data.thread) {
this.threads[data.threadId] = new Thread(data.thread.name, data.thread.id, []);
this.threads[data.threadId] = new Thread(data.thread.name, data.thread.id);
}

if (data.callStack) {
// convert raw call stack into proper modelled call stack
this.threads[data.threadId].callStack = data.callStack.map(
(rsf, level) => {
if (!rsf) {
return new StackFrame(data.threadId, 0, new Source({ name: 'unknown' }), nls.localize('unknownStack', "Unknown stack location"), undefined, undefined);
if (data.stoppedDetails) {
// Set the availability of the threads' callstacks depending on
// whether the thread is stopped or not
for (let ref in this.threads) {
if (this.threads.hasOwnProperty(ref)) {
if (data.allThreadsStopped) {
// Only update the details if all the threads are stopped
// because we don't want to overwrite the details of other
// threads that have stopped for a different reason
this.threads[ref].stoppedDetails = data.stoppedDetails;
}

return new StackFrame(data.threadId, rsf.id, rsf.source ? new Source(rsf.source) : new Source({ name: 'unknown' }), rsf.name, rsf.line, rsf.column);
});
this.threads[ref].stopped = data.allThreadsStopped;
this.threads[ref].clearCallStack();
}
}

this.threads[data.threadId].stoppedDetails = data.stoppedDetails;
this.threads[data.threadId].stopped = true;
}

this.emit(debug.ModelEvents.CALLSTACK_UPDATED);
Expand Down
4 changes: 2 additions & 2 deletions src/vs/workbench/parts/debug/common/debugSource.ts
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@ export class Source {
// first try to find the raw source amongst the stack frames - since that represenation has more data (source reference),
const threads = model.getThreads();
for (let threadId in threads) {
if (threads.hasOwnProperty(threadId) && threads[threadId].callStack) {
const found = threads[threadId].callStack.filter(sf => sf.source.uri.toString() === uri.toString()).pop();
if (threads.hasOwnProperty(threadId) && threads[threadId].getCachedCallStack()) {
const found = threads[threadId].getCachedCallStack().filter(sf => sf.source.uri.toString() === uri.toString()).pop();
if (found) {
return found.source.raw;
}
Expand Down
16 changes: 10 additions & 6 deletions src/vs/workbench/parts/debug/electron-browser/debugService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -240,13 +240,17 @@ export class DebugService extends ee.EventEmitter implements debug.IDebugService
this.setStateAndEmit(debug.State.Stopped);
const threadId = event.body.threadId;

this.getThreadData(threadId).then(() => {
this.session.stackTrace({ threadId: threadId, levels: 20 }).done((result) => {
this.getThreadData(threadId).done(() => {
let thread = this.model.getThreads()[threadId];

this.model.rawUpdate({ threadId: threadId, callStack: result.body.stackFrames, stoppedDetails: event.body });
this.windowService.getWindow().focus();
const callStack = this.model.getThreads()[threadId].callStack;
this.model.rawUpdate({
threadId: threadId,
stoppedDetails: event.body,
allThreadsStopped: event.body.allThreadsStopped
});

thread.getCallStack(this).then(callStack => {
this.windowService.getWindow().focus();
if (callStack.length > 0) {
// focus first stack frame from top that has source location
const stackFrameToFocus = arrays.first(callStack, sf => !!sf.source, callStack[0]);
Expand All @@ -263,7 +267,7 @@ export class DebugService extends ee.EventEmitter implements debug.IDebugService

this.toDisposeOnSessionEnd.push(this.session.addListener2(debug.SessionEvents.CONTINUED, () => {
aria.status(nls.localize('debuggingContinued', "Debugging continued."));
this.model.clearThreads(false);
this.model.continueThreads();
this.setFocusedStackFrameAndEvaluate(null);
this.setStateAndEmit(this.configurationManager.getConfiguration().noDebug ? debug.State.RunningNoDebug : debug.State.Running);
}));
Expand Down
Loading

0 comments on commit 5d620ea

Please sign in to comment.