@theia/debug 1.53.0-next.55 → 1.53.0-next.64
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +62 -62
- package/lib/browser/debug-configuration-manager.js +6 -6
- package/lib/common/inline-debug-adapter.d.ts +0 -1
- package/lib/common/inline-debug-adapter.d.ts.map +1 -1
- package/package.json +15 -15
- package/src/browser/breakpoint/breakpoint-manager.ts +369 -369
- package/src/browser/breakpoint/breakpoint-marker.ts +104 -104
- package/src/browser/console/debug-console-contribution.tsx +240 -240
- package/src/browser/console/debug-console-items.tsx +384 -384
- package/src/browser/console/debug-console-session.ts +205 -205
- package/src/browser/debug-call-stack-item-type-key.ts +20 -20
- package/src/browser/debug-configuration-manager.ts +591 -591
- package/src/browser/debug-configuration-model.ts +100 -100
- package/src/browser/debug-contribution.ts +43 -43
- package/src/browser/debug-frontend-application-contribution.ts +1551 -1551
- package/src/browser/debug-frontend-module.ts +133 -133
- package/src/browser/debug-package.spec.ts +20 -20
- package/src/browser/debug-preferences.ts +98 -98
- package/src/browser/debug-prefix-configuration.ts +195 -195
- package/src/browser/debug-resource.ts +59 -59
- package/src/browser/debug-schema-updater.ts +149 -149
- package/src/browser/debug-session-connection.ts +357 -357
- package/src/browser/debug-session-contribution.ts +157 -157
- package/src/browser/debug-session-manager.ts +683 -683
- package/src/browser/debug-session-options.ts +120 -120
- package/src/browser/debug-session.tsx +974 -974
- package/src/browser/debug-tab-bar-decorator.ts +57 -57
- package/src/browser/debug-watch-manager.ts +93 -93
- package/src/browser/disassembly-view/disassembly-view-accessibility-provider.ts +43 -43
- package/src/browser/disassembly-view/disassembly-view-breakpoint-renderer.ts +119 -119
- package/src/browser/disassembly-view/disassembly-view-contribution.ts +109 -109
- package/src/browser/disassembly-view/disassembly-view-instruction-renderer.ts +245 -245
- package/src/browser/disassembly-view/disassembly-view-table-delegate.ts +39 -39
- package/src/browser/disassembly-view/disassembly-view-utilities.ts +55 -55
- package/src/browser/disassembly-view/disassembly-view-widget.ts +463 -463
- package/src/browser/editor/debug-breakpoint-widget.tsx +293 -293
- package/src/browser/editor/debug-editor-model.ts +529 -529
- package/src/browser/editor/debug-editor-service.ts +192 -192
- package/src/browser/editor/debug-editor.ts +20 -20
- package/src/browser/editor/debug-exception-widget.tsx +122 -122
- package/src/browser/editor/debug-expression-provider.ts +78 -78
- package/src/browser/editor/debug-hover-source.tsx +105 -105
- package/src/browser/editor/debug-hover-widget.ts +298 -298
- package/src/browser/editor/debug-inline-value-decorator.ts +373 -373
- package/src/browser/model/debug-breakpoint.tsx +151 -151
- package/src/browser/model/debug-function-breakpoint.tsx +101 -101
- package/src/browser/model/debug-instruction-breakpoint.tsx +68 -68
- package/src/browser/model/debug-source-breakpoint.tsx +237 -237
- package/src/browser/model/debug-source.ts +93 -93
- package/src/browser/model/debug-stack-frame.tsx +177 -177
- package/src/browser/model/debug-thread.tsx +292 -292
- package/src/browser/preferences/launch-preferences.ts +38 -38
- package/src/browser/style/index.css +453 -453
- package/src/browser/view/debug-action.tsx +57 -57
- package/src/browser/view/debug-breakpoints-source.tsx +53 -53
- package/src/browser/view/debug-breakpoints-widget.ts +71 -71
- package/src/browser/view/debug-configuration-select.tsx +269 -269
- package/src/browser/view/debug-configuration-widget.tsx +121 -121
- package/src/browser/view/debug-exception-breakpoint.tsx +68 -68
- package/src/browser/view/debug-session-widget.ts +124 -124
- package/src/browser/view/debug-stack-frames-source.tsx +75 -75
- package/src/browser/view/debug-stack-frames-widget.ts +135 -135
- package/src/browser/view/debug-threads-source.tsx +48 -48
- package/src/browser/view/debug-threads-widget.ts +126 -126
- package/src/browser/view/debug-toolbar-widget.tsx +145 -145
- package/src/browser/view/debug-variables-source.ts +43 -43
- package/src/browser/view/debug-variables-widget.ts +61 -61
- package/src/browser/view/debug-view-model.ts +230 -230
- package/src/browser/view/debug-watch-expression.tsx +88 -88
- package/src/browser/view/debug-watch-source.ts +41 -41
- package/src/browser/view/debug-watch-widget.ts +61 -61
- package/src/browser/view/debug-widget.ts +97 -97
- package/src/common/debug-adapter-contribution-registry.ts +206 -206
- package/src/common/debug-adapter-session.ts +102 -102
- package/src/common/debug-common.ts +19 -19
- package/src/common/debug-compound.ts +33 -33
- package/src/common/debug-configuration.ts +112 -112
- package/src/common/debug-model.ts +200 -200
- package/src/common/debug-service.ts +184 -184
- package/src/common/debug-uri-utils.ts +24 -24
- package/src/common/inline-debug-adapter.ts +47 -47
- package/src/node/debug-adapter-factory.ts +107 -107
- package/src/node/debug-adapter-session-manager.ts +106 -106
- package/src/node/debug-backend-module.ts +57 -57
- package/src/node/debug-service-impl.ts +119 -119
- package/src/node/stream-debug-adapter.ts +126 -126
|
@@ -1,245 +1,245 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2022 Ericsson and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { open, OpenerService } from '@theia/core/lib/browser';
|
|
18
|
-
import { URI as TheiaURI } from '@theia/core/lib/common/uri';
|
|
19
|
-
import { EditorOpenerOptions } from '@theia/editor/lib/browser';
|
|
20
|
-
import { IDisposable, Uri as URI } from '@theia/monaco-editor-core';
|
|
21
|
-
import { $, addStandardDisposableListener, append } from '@theia/monaco-editor-core/esm/vs/base/browser/dom';
|
|
22
|
-
import { ITableRenderer } from '@theia/monaco-editor-core/esm/vs/base/browser/ui/table/table';
|
|
23
|
-
import { Color } from '@theia/monaco-editor-core/esm/vs/base/common/color';
|
|
24
|
-
import { Disposable, dispose } from '@theia/monaco-editor-core/esm/vs/base/common/lifecycle';
|
|
25
|
-
import { isAbsolute } from '@theia/monaco-editor-core/esm/vs/base/common/path';
|
|
26
|
-
import { Constants } from '@theia/monaco-editor-core/esm/vs/base/common/uint';
|
|
27
|
-
import { applyFontInfo } from '@theia/monaco-editor-core/esm/vs/editor/browser/config/domFontInfo';
|
|
28
|
-
import { StringBuilder } from '@theia/monaco-editor-core/esm/vs/editor/common/core/stringBuilder';
|
|
29
|
-
import { ITextModel } from '@theia/monaco-editor-core/esm/vs/editor/common/model';
|
|
30
|
-
import { ITextModelService } from '@theia/monaco-editor-core/esm/vs/editor/common/services/resolverService';
|
|
31
|
-
import { IThemeService } from '@theia/monaco-editor-core/esm/vs/platform/theme/common/themeService';
|
|
32
|
-
import { DebugProtocol } from '@vscode/debugprotocol';
|
|
33
|
-
import { DebugSource } from '../model/debug-source';
|
|
34
|
-
import { DisassembledInstructionEntry, DisassemblyViewRendererReference, InstructionColumnTemplateData } from './disassembly-view-utilities';
|
|
35
|
-
|
|
36
|
-
// This file is adapted from https://github.com/microsoft/vscode/blob/c061ce5c24fc480342fbc5f23244289d633c56eb/src/vs/workbench/contrib/debug/browser/disassemblyView.ts
|
|
37
|
-
|
|
38
|
-
const topStackFrameColor = 'editor.stackFrameHighlightBackground';
|
|
39
|
-
const focusedStackFrameColor = 'editor.focusedStackFrameHighlightBackground';
|
|
40
|
-
|
|
41
|
-
export class InstructionRenderer extends Disposable implements ITableRenderer<DisassembledInstructionEntry, InstructionColumnTemplateData> {
|
|
42
|
-
|
|
43
|
-
static readonly TEMPLATE_ID = 'instruction';
|
|
44
|
-
|
|
45
|
-
protected static readonly INSTRUCTION_ADDR_MIN_LENGTH = 25;
|
|
46
|
-
protected static readonly INSTRUCTION_BYTES_MIN_LENGTH = 30;
|
|
47
|
-
|
|
48
|
-
templateId: string = InstructionRenderer.TEMPLATE_ID;
|
|
49
|
-
|
|
50
|
-
protected _topStackFrameColor: Color | undefined;
|
|
51
|
-
protected _focusedStackFrameColor: Color | undefined;
|
|
52
|
-
|
|
53
|
-
constructor(
|
|
54
|
-
protected readonly _disassemblyView: DisassemblyViewRendererReference,
|
|
55
|
-
protected readonly openerService: OpenerService,
|
|
56
|
-
protected readonly uriService: { asCanonicalUri(uri: URI): URI },
|
|
57
|
-
@IThemeService themeService: IThemeService,
|
|
58
|
-
@ITextModelService protected readonly textModelService: ITextModelService,
|
|
59
|
-
) {
|
|
60
|
-
super();
|
|
61
|
-
|
|
62
|
-
this._topStackFrameColor = themeService.getColorTheme().getColor(topStackFrameColor);
|
|
63
|
-
this._focusedStackFrameColor = themeService.getColorTheme().getColor(focusedStackFrameColor);
|
|
64
|
-
|
|
65
|
-
this._register(themeService.onDidColorThemeChange(e => {
|
|
66
|
-
this._topStackFrameColor = e.getColor(topStackFrameColor);
|
|
67
|
-
this._focusedStackFrameColor = e.getColor(focusedStackFrameColor);
|
|
68
|
-
}));
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
renderTemplate(container: HTMLElement): InstructionColumnTemplateData {
|
|
72
|
-
const sourcecode = append(container, $('.sourcecode'));
|
|
73
|
-
const instruction = append(container, $('.instruction'));
|
|
74
|
-
this.applyFontInfo(sourcecode);
|
|
75
|
-
this.applyFontInfo(instruction);
|
|
76
|
-
const currentElement: { element?: DisassembledInstructionEntry } = { element: undefined };
|
|
77
|
-
const cellDisposable: IDisposable[] = [];
|
|
78
|
-
|
|
79
|
-
const disposables = [
|
|
80
|
-
this._disassemblyView.onDidChangeStackFrame(() => this.rerenderBackground(instruction, sourcecode, currentElement.element)),
|
|
81
|
-
addStandardDisposableListener(sourcecode, 'dblclick', () => this.openSourceCode(currentElement.element?.instruction!)),
|
|
82
|
-
];
|
|
83
|
-
|
|
84
|
-
return { currentElement, instruction, sourcecode, cellDisposable, disposables };
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
renderElement(element: DisassembledInstructionEntry, index: number, templateData: InstructionColumnTemplateData, height: number | undefined): void {
|
|
88
|
-
this.renderElementInner(element, index, templateData, height);
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
protected async renderElementInner(element: DisassembledInstructionEntry, index: number, column: InstructionColumnTemplateData, height: number | undefined): Promise<void> {
|
|
92
|
-
column.currentElement.element = element;
|
|
93
|
-
const instruction = element.instruction;
|
|
94
|
-
column.sourcecode.innerText = '';
|
|
95
|
-
const sb = new StringBuilder(1000);
|
|
96
|
-
|
|
97
|
-
if (this._disassemblyView.isSourceCodeRender && instruction.location?.path && instruction.line) {
|
|
98
|
-
const sourceURI = this.getUriFromSource(instruction);
|
|
99
|
-
|
|
100
|
-
if (sourceURI) {
|
|
101
|
-
let textModel: ITextModel | undefined = undefined;
|
|
102
|
-
const sourceSB = new StringBuilder(10000);
|
|
103
|
-
const ref = await this.textModelService.createModelReference(sourceURI);
|
|
104
|
-
textModel = ref.object.textEditorModel;
|
|
105
|
-
column.cellDisposable.push(ref);
|
|
106
|
-
|
|
107
|
-
// templateData could have moved on during async. Double check if it is still the same source.
|
|
108
|
-
if (textModel && column.currentElement.element === element) {
|
|
109
|
-
let lineNumber = instruction.line;
|
|
110
|
-
|
|
111
|
-
while (lineNumber && lineNumber >= 1 && lineNumber <= textModel.getLineCount()) {
|
|
112
|
-
const lineContent = textModel.getLineContent(lineNumber);
|
|
113
|
-
sourceSB.appendString(` ${lineNumber}: `);
|
|
114
|
-
sourceSB.appendString(lineContent + '\n');
|
|
115
|
-
|
|
116
|
-
if (instruction.endLine && lineNumber < instruction.endLine) {
|
|
117
|
-
lineNumber++;
|
|
118
|
-
continue;
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
break;
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
column.sourcecode.innerText = sourceSB.build();
|
|
125
|
-
}
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
let spacesToAppend = 10;
|
|
130
|
-
|
|
131
|
-
if (instruction.address !== '-1') {
|
|
132
|
-
sb.appendString(instruction.address);
|
|
133
|
-
if (instruction.address.length < InstructionRenderer.INSTRUCTION_ADDR_MIN_LENGTH) {
|
|
134
|
-
spacesToAppend = InstructionRenderer.INSTRUCTION_ADDR_MIN_LENGTH - instruction.address.length;
|
|
135
|
-
}
|
|
136
|
-
for (let i = 0; i < spacesToAppend; i++) {
|
|
137
|
-
sb.appendString(' ');
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
if (instruction.instructionBytes) {
|
|
142
|
-
sb.appendString(instruction.instructionBytes);
|
|
143
|
-
spacesToAppend = 10;
|
|
144
|
-
if (instruction.instructionBytes.length < InstructionRenderer.INSTRUCTION_BYTES_MIN_LENGTH) {
|
|
145
|
-
spacesToAppend = InstructionRenderer.INSTRUCTION_BYTES_MIN_LENGTH - instruction.instructionBytes.length;
|
|
146
|
-
}
|
|
147
|
-
for (let i = 0; i < spacesToAppend; i++) {
|
|
148
|
-
sb.appendString(' ');
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
sb.appendString(instruction.instruction);
|
|
153
|
-
column.instruction.innerText = sb.build();
|
|
154
|
-
|
|
155
|
-
this.rerenderBackground(column.instruction, column.sourcecode, element);
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
disposeElement(element: DisassembledInstructionEntry, index: number, templateData: InstructionColumnTemplateData, height: number | undefined): void {
|
|
159
|
-
dispose(templateData.cellDisposable);
|
|
160
|
-
templateData.cellDisposable = [];
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
disposeTemplate(templateData: InstructionColumnTemplateData): void {
|
|
164
|
-
dispose(templateData.disposables);
|
|
165
|
-
templateData.disposables = [];
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
protected rerenderBackground(instruction: HTMLElement, sourceCode: HTMLElement, element?: DisassembledInstructionEntry): void {
|
|
169
|
-
if (element && this._disassemblyView.currentInstructionAddresses.includes(element.instruction.address)) {
|
|
170
|
-
instruction.style.background = this._topStackFrameColor?.toString() || 'transparent';
|
|
171
|
-
} else if (element?.instruction.address === this._disassemblyView.focusedInstructionAddress) {
|
|
172
|
-
instruction.style.background = this._focusedStackFrameColor?.toString() || 'transparent';
|
|
173
|
-
} else {
|
|
174
|
-
instruction.style.background = 'transparent';
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
protected openSourceCode(instruction: DebugProtocol.DisassembledInstruction | undefined): void {
|
|
179
|
-
if (instruction) {
|
|
180
|
-
const sourceURI = this.getUriFromSource(instruction);
|
|
181
|
-
const selection: EditorOpenerOptions['selection'] = instruction.endLine ? {
|
|
182
|
-
start: { line: instruction.line!, character: instruction.column ?? 0 },
|
|
183
|
-
end: { line: instruction.endLine, character: instruction.endColumn ?? Constants.MAX_SAFE_SMALL_INTEGER }
|
|
184
|
-
} : {
|
|
185
|
-
start: { line: instruction.line!, character: instruction.column ?? 0 },
|
|
186
|
-
end: { line: instruction.line, character: instruction.endColumn ?? Constants.MAX_SAFE_SMALL_INTEGER }
|
|
187
|
-
};
|
|
188
|
-
|
|
189
|
-
const openerOptions: EditorOpenerOptions = {
|
|
190
|
-
selection,
|
|
191
|
-
mode: 'activate',
|
|
192
|
-
widgetOptions: { area: 'main' }
|
|
193
|
-
};
|
|
194
|
-
open(this.openerService, new TheiaURI(sourceURI.toString()), openerOptions);
|
|
195
|
-
}
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
protected getUriFromSource(instruction: DebugProtocol.DisassembledInstruction): URI {
|
|
199
|
-
// Try to resolve path before consulting the debugSession.
|
|
200
|
-
const path = instruction.location!.path;
|
|
201
|
-
if (path && isUri(path)) { // path looks like a uri
|
|
202
|
-
return this.uriService.asCanonicalUri(URI.parse(path));
|
|
203
|
-
}
|
|
204
|
-
// assume a filesystem path
|
|
205
|
-
if (path && isAbsolute(path)) {
|
|
206
|
-
return this.uriService.asCanonicalUri(URI.file(path));
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
return getUriFromSource(instruction.location!, instruction.location!.path, this._disassemblyView.debugSession!.id, this.uriService);
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
protected applyFontInfo(element: HTMLElement): void {
|
|
213
|
-
applyFontInfo(element, this._disassemblyView.fontInfo);
|
|
214
|
-
element.style.whiteSpace = 'pre';
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
export function getUriFromSource(raw: DebugProtocol.Source, path: string | undefined, sessionId: string, uriIdentityService: { asCanonicalUri(uri: URI): URI }): URI {
|
|
219
|
-
if (typeof raw.sourceReference === 'number' && raw.sourceReference > 0) {
|
|
220
|
-
return URI.from({
|
|
221
|
-
scheme: DebugSource.SCHEME,
|
|
222
|
-
path,
|
|
223
|
-
query: `session=${sessionId}&ref=${raw.sourceReference}`
|
|
224
|
-
});
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
if (path && isUri(path)) { // path looks like a uri
|
|
228
|
-
return uriIdentityService.asCanonicalUri(URI.parse(path));
|
|
229
|
-
}
|
|
230
|
-
// assume a filesystem path
|
|
231
|
-
if (path && isAbsolute(path)) {
|
|
232
|
-
return uriIdentityService.asCanonicalUri(URI.file(path));
|
|
233
|
-
}
|
|
234
|
-
// path is relative: since VS Code cannot deal with this by itself
|
|
235
|
-
// create a debug url that will result in a DAP 'source' request when the url is resolved.
|
|
236
|
-
return uriIdentityService.asCanonicalUri(URI.from({
|
|
237
|
-
scheme: DebugSource.SCHEME,
|
|
238
|
-
path,
|
|
239
|
-
query: `session=${sessionId}`
|
|
240
|
-
}));
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
function isUri(candidate: string | undefined): boolean {
|
|
244
|
-
return Boolean(candidate && candidate.match(DebugSource.SCHEME_PATTERN));
|
|
245
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2022 Ericsson and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { open, OpenerService } from '@theia/core/lib/browser';
|
|
18
|
+
import { URI as TheiaURI } from '@theia/core/lib/common/uri';
|
|
19
|
+
import { EditorOpenerOptions } from '@theia/editor/lib/browser';
|
|
20
|
+
import { IDisposable, Uri as URI } from '@theia/monaco-editor-core';
|
|
21
|
+
import { $, addStandardDisposableListener, append } from '@theia/monaco-editor-core/esm/vs/base/browser/dom';
|
|
22
|
+
import { ITableRenderer } from '@theia/monaco-editor-core/esm/vs/base/browser/ui/table/table';
|
|
23
|
+
import { Color } from '@theia/monaco-editor-core/esm/vs/base/common/color';
|
|
24
|
+
import { Disposable, dispose } from '@theia/monaco-editor-core/esm/vs/base/common/lifecycle';
|
|
25
|
+
import { isAbsolute } from '@theia/monaco-editor-core/esm/vs/base/common/path';
|
|
26
|
+
import { Constants } from '@theia/monaco-editor-core/esm/vs/base/common/uint';
|
|
27
|
+
import { applyFontInfo } from '@theia/monaco-editor-core/esm/vs/editor/browser/config/domFontInfo';
|
|
28
|
+
import { StringBuilder } from '@theia/monaco-editor-core/esm/vs/editor/common/core/stringBuilder';
|
|
29
|
+
import { ITextModel } from '@theia/monaco-editor-core/esm/vs/editor/common/model';
|
|
30
|
+
import { ITextModelService } from '@theia/monaco-editor-core/esm/vs/editor/common/services/resolverService';
|
|
31
|
+
import { IThemeService } from '@theia/monaco-editor-core/esm/vs/platform/theme/common/themeService';
|
|
32
|
+
import { DebugProtocol } from '@vscode/debugprotocol';
|
|
33
|
+
import { DebugSource } from '../model/debug-source';
|
|
34
|
+
import { DisassembledInstructionEntry, DisassemblyViewRendererReference, InstructionColumnTemplateData } from './disassembly-view-utilities';
|
|
35
|
+
|
|
36
|
+
// This file is adapted from https://github.com/microsoft/vscode/blob/c061ce5c24fc480342fbc5f23244289d633c56eb/src/vs/workbench/contrib/debug/browser/disassemblyView.ts
|
|
37
|
+
|
|
38
|
+
const topStackFrameColor = 'editor.stackFrameHighlightBackground';
|
|
39
|
+
const focusedStackFrameColor = 'editor.focusedStackFrameHighlightBackground';
|
|
40
|
+
|
|
41
|
+
export class InstructionRenderer extends Disposable implements ITableRenderer<DisassembledInstructionEntry, InstructionColumnTemplateData> {
|
|
42
|
+
|
|
43
|
+
static readonly TEMPLATE_ID = 'instruction';
|
|
44
|
+
|
|
45
|
+
protected static readonly INSTRUCTION_ADDR_MIN_LENGTH = 25;
|
|
46
|
+
protected static readonly INSTRUCTION_BYTES_MIN_LENGTH = 30;
|
|
47
|
+
|
|
48
|
+
templateId: string = InstructionRenderer.TEMPLATE_ID;
|
|
49
|
+
|
|
50
|
+
protected _topStackFrameColor: Color | undefined;
|
|
51
|
+
protected _focusedStackFrameColor: Color | undefined;
|
|
52
|
+
|
|
53
|
+
constructor(
|
|
54
|
+
protected readonly _disassemblyView: DisassemblyViewRendererReference,
|
|
55
|
+
protected readonly openerService: OpenerService,
|
|
56
|
+
protected readonly uriService: { asCanonicalUri(uri: URI): URI },
|
|
57
|
+
@IThemeService themeService: IThemeService,
|
|
58
|
+
@ITextModelService protected readonly textModelService: ITextModelService,
|
|
59
|
+
) {
|
|
60
|
+
super();
|
|
61
|
+
|
|
62
|
+
this._topStackFrameColor = themeService.getColorTheme().getColor(topStackFrameColor);
|
|
63
|
+
this._focusedStackFrameColor = themeService.getColorTheme().getColor(focusedStackFrameColor);
|
|
64
|
+
|
|
65
|
+
this._register(themeService.onDidColorThemeChange(e => {
|
|
66
|
+
this._topStackFrameColor = e.getColor(topStackFrameColor);
|
|
67
|
+
this._focusedStackFrameColor = e.getColor(focusedStackFrameColor);
|
|
68
|
+
}));
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
renderTemplate(container: HTMLElement): InstructionColumnTemplateData {
|
|
72
|
+
const sourcecode = append(container, $('.sourcecode'));
|
|
73
|
+
const instruction = append(container, $('.instruction'));
|
|
74
|
+
this.applyFontInfo(sourcecode);
|
|
75
|
+
this.applyFontInfo(instruction);
|
|
76
|
+
const currentElement: { element?: DisassembledInstructionEntry } = { element: undefined };
|
|
77
|
+
const cellDisposable: IDisposable[] = [];
|
|
78
|
+
|
|
79
|
+
const disposables = [
|
|
80
|
+
this._disassemblyView.onDidChangeStackFrame(() => this.rerenderBackground(instruction, sourcecode, currentElement.element)),
|
|
81
|
+
addStandardDisposableListener(sourcecode, 'dblclick', () => this.openSourceCode(currentElement.element?.instruction!)),
|
|
82
|
+
];
|
|
83
|
+
|
|
84
|
+
return { currentElement, instruction, sourcecode, cellDisposable, disposables };
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
renderElement(element: DisassembledInstructionEntry, index: number, templateData: InstructionColumnTemplateData, height: number | undefined): void {
|
|
88
|
+
this.renderElementInner(element, index, templateData, height);
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
protected async renderElementInner(element: DisassembledInstructionEntry, index: number, column: InstructionColumnTemplateData, height: number | undefined): Promise<void> {
|
|
92
|
+
column.currentElement.element = element;
|
|
93
|
+
const instruction = element.instruction;
|
|
94
|
+
column.sourcecode.innerText = '';
|
|
95
|
+
const sb = new StringBuilder(1000);
|
|
96
|
+
|
|
97
|
+
if (this._disassemblyView.isSourceCodeRender && instruction.location?.path && instruction.line) {
|
|
98
|
+
const sourceURI = this.getUriFromSource(instruction);
|
|
99
|
+
|
|
100
|
+
if (sourceURI) {
|
|
101
|
+
let textModel: ITextModel | undefined = undefined;
|
|
102
|
+
const sourceSB = new StringBuilder(10000);
|
|
103
|
+
const ref = await this.textModelService.createModelReference(sourceURI);
|
|
104
|
+
textModel = ref.object.textEditorModel;
|
|
105
|
+
column.cellDisposable.push(ref);
|
|
106
|
+
|
|
107
|
+
// templateData could have moved on during async. Double check if it is still the same source.
|
|
108
|
+
if (textModel && column.currentElement.element === element) {
|
|
109
|
+
let lineNumber = instruction.line;
|
|
110
|
+
|
|
111
|
+
while (lineNumber && lineNumber >= 1 && lineNumber <= textModel.getLineCount()) {
|
|
112
|
+
const lineContent = textModel.getLineContent(lineNumber);
|
|
113
|
+
sourceSB.appendString(` ${lineNumber}: `);
|
|
114
|
+
sourceSB.appendString(lineContent + '\n');
|
|
115
|
+
|
|
116
|
+
if (instruction.endLine && lineNumber < instruction.endLine) {
|
|
117
|
+
lineNumber++;
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
break;
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
column.sourcecode.innerText = sourceSB.build();
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
let spacesToAppend = 10;
|
|
130
|
+
|
|
131
|
+
if (instruction.address !== '-1') {
|
|
132
|
+
sb.appendString(instruction.address);
|
|
133
|
+
if (instruction.address.length < InstructionRenderer.INSTRUCTION_ADDR_MIN_LENGTH) {
|
|
134
|
+
spacesToAppend = InstructionRenderer.INSTRUCTION_ADDR_MIN_LENGTH - instruction.address.length;
|
|
135
|
+
}
|
|
136
|
+
for (let i = 0; i < spacesToAppend; i++) {
|
|
137
|
+
sb.appendString(' ');
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
if (instruction.instructionBytes) {
|
|
142
|
+
sb.appendString(instruction.instructionBytes);
|
|
143
|
+
spacesToAppend = 10;
|
|
144
|
+
if (instruction.instructionBytes.length < InstructionRenderer.INSTRUCTION_BYTES_MIN_LENGTH) {
|
|
145
|
+
spacesToAppend = InstructionRenderer.INSTRUCTION_BYTES_MIN_LENGTH - instruction.instructionBytes.length;
|
|
146
|
+
}
|
|
147
|
+
for (let i = 0; i < spacesToAppend; i++) {
|
|
148
|
+
sb.appendString(' ');
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
sb.appendString(instruction.instruction);
|
|
153
|
+
column.instruction.innerText = sb.build();
|
|
154
|
+
|
|
155
|
+
this.rerenderBackground(column.instruction, column.sourcecode, element);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
disposeElement(element: DisassembledInstructionEntry, index: number, templateData: InstructionColumnTemplateData, height: number | undefined): void {
|
|
159
|
+
dispose(templateData.cellDisposable);
|
|
160
|
+
templateData.cellDisposable = [];
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
disposeTemplate(templateData: InstructionColumnTemplateData): void {
|
|
164
|
+
dispose(templateData.disposables);
|
|
165
|
+
templateData.disposables = [];
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
protected rerenderBackground(instruction: HTMLElement, sourceCode: HTMLElement, element?: DisassembledInstructionEntry): void {
|
|
169
|
+
if (element && this._disassemblyView.currentInstructionAddresses.includes(element.instruction.address)) {
|
|
170
|
+
instruction.style.background = this._topStackFrameColor?.toString() || 'transparent';
|
|
171
|
+
} else if (element?.instruction.address === this._disassemblyView.focusedInstructionAddress) {
|
|
172
|
+
instruction.style.background = this._focusedStackFrameColor?.toString() || 'transparent';
|
|
173
|
+
} else {
|
|
174
|
+
instruction.style.background = 'transparent';
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
protected openSourceCode(instruction: DebugProtocol.DisassembledInstruction | undefined): void {
|
|
179
|
+
if (instruction) {
|
|
180
|
+
const sourceURI = this.getUriFromSource(instruction);
|
|
181
|
+
const selection: EditorOpenerOptions['selection'] = instruction.endLine ? {
|
|
182
|
+
start: { line: instruction.line!, character: instruction.column ?? 0 },
|
|
183
|
+
end: { line: instruction.endLine, character: instruction.endColumn ?? Constants.MAX_SAFE_SMALL_INTEGER }
|
|
184
|
+
} : {
|
|
185
|
+
start: { line: instruction.line!, character: instruction.column ?? 0 },
|
|
186
|
+
end: { line: instruction.line, character: instruction.endColumn ?? Constants.MAX_SAFE_SMALL_INTEGER }
|
|
187
|
+
};
|
|
188
|
+
|
|
189
|
+
const openerOptions: EditorOpenerOptions = {
|
|
190
|
+
selection,
|
|
191
|
+
mode: 'activate',
|
|
192
|
+
widgetOptions: { area: 'main' }
|
|
193
|
+
};
|
|
194
|
+
open(this.openerService, new TheiaURI(sourceURI.toString()), openerOptions);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
protected getUriFromSource(instruction: DebugProtocol.DisassembledInstruction): URI {
|
|
199
|
+
// Try to resolve path before consulting the debugSession.
|
|
200
|
+
const path = instruction.location!.path;
|
|
201
|
+
if (path && isUri(path)) { // path looks like a uri
|
|
202
|
+
return this.uriService.asCanonicalUri(URI.parse(path));
|
|
203
|
+
}
|
|
204
|
+
// assume a filesystem path
|
|
205
|
+
if (path && isAbsolute(path)) {
|
|
206
|
+
return this.uriService.asCanonicalUri(URI.file(path));
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
return getUriFromSource(instruction.location!, instruction.location!.path, this._disassemblyView.debugSession!.id, this.uriService);
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
protected applyFontInfo(element: HTMLElement): void {
|
|
213
|
+
applyFontInfo(element, this._disassemblyView.fontInfo);
|
|
214
|
+
element.style.whiteSpace = 'pre';
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
export function getUriFromSource(raw: DebugProtocol.Source, path: string | undefined, sessionId: string, uriIdentityService: { asCanonicalUri(uri: URI): URI }): URI {
|
|
219
|
+
if (typeof raw.sourceReference === 'number' && raw.sourceReference > 0) {
|
|
220
|
+
return URI.from({
|
|
221
|
+
scheme: DebugSource.SCHEME,
|
|
222
|
+
path,
|
|
223
|
+
query: `session=${sessionId}&ref=${raw.sourceReference}`
|
|
224
|
+
});
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
if (path && isUri(path)) { // path looks like a uri
|
|
228
|
+
return uriIdentityService.asCanonicalUri(URI.parse(path));
|
|
229
|
+
}
|
|
230
|
+
// assume a filesystem path
|
|
231
|
+
if (path && isAbsolute(path)) {
|
|
232
|
+
return uriIdentityService.asCanonicalUri(URI.file(path));
|
|
233
|
+
}
|
|
234
|
+
// path is relative: since VS Code cannot deal with this by itself
|
|
235
|
+
// create a debug url that will result in a DAP 'source' request when the url is resolved.
|
|
236
|
+
return uriIdentityService.asCanonicalUri(URI.from({
|
|
237
|
+
scheme: DebugSource.SCHEME,
|
|
238
|
+
path,
|
|
239
|
+
query: `session=${sessionId}`
|
|
240
|
+
}));
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
function isUri(candidate: string | undefined): boolean {
|
|
244
|
+
return Boolean(candidate && candidate.match(DebugSource.SCHEME_PATTERN));
|
|
245
|
+
}
|
|
@@ -1,39 +1,39 @@
|
|
|
1
|
-
// *****************************************************************************
|
|
2
|
-
// Copyright (C) 2022 Ericsson and others.
|
|
3
|
-
//
|
|
4
|
-
// This program and the accompanying materials are made available under the
|
|
5
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
-
//
|
|
8
|
-
// This Source Code may also be made available under the following Secondary
|
|
9
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
-
// with the GNU Classpath Exception which is available at
|
|
12
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
-
//
|
|
14
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
-
// *****************************************************************************
|
|
16
|
-
|
|
17
|
-
import { ITableVirtualDelegate } from '@theia/monaco-editor-core/esm/vs/base/browser/ui/table/table';
|
|
18
|
-
import { BareFontInfo } from '@theia/monaco-editor-core/esm/vs/editor/common/config/fontInfo';
|
|
19
|
-
import { DisassembledInstructionEntry } from './disassembly-view-utilities';
|
|
20
|
-
|
|
21
|
-
// This file is adapted from https://github.com/microsoft/vscode/blob/c061ce5c24fc480342fbc5f23244289d633c56eb/src/vs/workbench/contrib/debug/browser/disassemblyView.ts
|
|
22
|
-
|
|
23
|
-
export class DisassemblyViewTableDelegate implements ITableVirtualDelegate<DisassembledInstructionEntry> {
|
|
24
|
-
constructor(protected readonly fontInfoProvider: { fontInfo: BareFontInfo, isSourceCodeRender: boolean }) { }
|
|
25
|
-
|
|
26
|
-
headerRowHeight = 0;
|
|
27
|
-
|
|
28
|
-
getHeight(row: DisassembledInstructionEntry): number {
|
|
29
|
-
if (this.fontInfoProvider.isSourceCodeRender && row.instruction.location?.path && row.instruction.line !== undefined) {
|
|
30
|
-
if (row.instruction.endLine !== undefined) {
|
|
31
|
-
return this.fontInfoProvider.fontInfo.lineHeight + (row.instruction.endLine - row.instruction.line + 2);
|
|
32
|
-
} else {
|
|
33
|
-
return this.fontInfoProvider.fontInfo.lineHeight * 2;
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
return this.fontInfoProvider.fontInfo.lineHeight;
|
|
38
|
-
}
|
|
39
|
-
}
|
|
1
|
+
// *****************************************************************************
|
|
2
|
+
// Copyright (C) 2022 Ericsson and others.
|
|
3
|
+
//
|
|
4
|
+
// This program and the accompanying materials are made available under the
|
|
5
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
6
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
7
|
+
//
|
|
8
|
+
// This Source Code may also be made available under the following Secondary
|
|
9
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
10
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
11
|
+
// with the GNU Classpath Exception which is available at
|
|
12
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
13
|
+
//
|
|
14
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
15
|
+
// *****************************************************************************
|
|
16
|
+
|
|
17
|
+
import { ITableVirtualDelegate } from '@theia/monaco-editor-core/esm/vs/base/browser/ui/table/table';
|
|
18
|
+
import { BareFontInfo } from '@theia/monaco-editor-core/esm/vs/editor/common/config/fontInfo';
|
|
19
|
+
import { DisassembledInstructionEntry } from './disassembly-view-utilities';
|
|
20
|
+
|
|
21
|
+
// This file is adapted from https://github.com/microsoft/vscode/blob/c061ce5c24fc480342fbc5f23244289d633c56eb/src/vs/workbench/contrib/debug/browser/disassemblyView.ts
|
|
22
|
+
|
|
23
|
+
export class DisassemblyViewTableDelegate implements ITableVirtualDelegate<DisassembledInstructionEntry> {
|
|
24
|
+
constructor(protected readonly fontInfoProvider: { fontInfo: BareFontInfo, isSourceCodeRender: boolean }) { }
|
|
25
|
+
|
|
26
|
+
headerRowHeight = 0;
|
|
27
|
+
|
|
28
|
+
getHeight(row: DisassembledInstructionEntry): number {
|
|
29
|
+
if (this.fontInfoProvider.isSourceCodeRender && row.instruction.location?.path && row.instruction.line !== undefined) {
|
|
30
|
+
if (row.instruction.endLine !== undefined) {
|
|
31
|
+
return this.fontInfoProvider.fontInfo.lineHeight + (row.instruction.endLine - row.instruction.line + 2);
|
|
32
|
+
} else {
|
|
33
|
+
return this.fontInfoProvider.fontInfo.lineHeight * 2;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
return this.fontInfoProvider.fontInfo.lineHeight;
|
|
38
|
+
}
|
|
39
|
+
}
|