2020-05-11 13:06:57 +00:00
|
|
|
import * as os from "os";
|
|
|
|
import * as vscode from 'vscode';
|
2020-05-11 15:49:45 +00:00
|
|
|
import * as path from 'path';
|
2020-05-25 12:56:26 +00:00
|
|
|
import * as ra from './lsp_ext';
|
2020-05-11 13:06:57 +00:00
|
|
|
|
|
|
|
import { Cargo } from './cargo';
|
|
|
|
import { Ctx } from "./ctx";
|
|
|
|
|
|
|
|
const debugOutput = vscode.window.createOutputChannel("Debug");
|
|
|
|
type DebugConfigProvider = (config: ra.Runnable, executable: string, sourceFileMap?: Record<string, string>) => vscode.DebugConfiguration;
|
|
|
|
|
|
|
|
function getLldbDebugConfig(config: ra.Runnable, executable: string, sourceFileMap?: Record<string, string>): vscode.DebugConfiguration {
|
|
|
|
return {
|
|
|
|
type: "lldb",
|
|
|
|
request: "launch",
|
|
|
|
name: config.label,
|
|
|
|
program: executable,
|
|
|
|
args: config.extraArgs,
|
|
|
|
cwd: config.cwd,
|
|
|
|
sourceMap: sourceFileMap,
|
|
|
|
sourceLanguages: ["rust"]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function getCppvsDebugConfig(config: ra.Runnable, executable: string, sourceFileMap?: Record<string, string>): vscode.DebugConfiguration {
|
|
|
|
return {
|
|
|
|
type: (os.platform() === "win32") ? "cppvsdbg" : "cppdbg",
|
|
|
|
request: "launch",
|
|
|
|
name: config.label,
|
|
|
|
program: executable,
|
|
|
|
args: config.extraArgs,
|
|
|
|
cwd: config.cwd,
|
|
|
|
sourceFileMap: sourceFileMap,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getDebugExecutable(config: ra.Runnable): Promise<string> {
|
|
|
|
const cargo = new Cargo(config.cwd || '.', debugOutput);
|
|
|
|
const executable = await cargo.executableFromArgs(config.args);
|
|
|
|
|
|
|
|
// if we are here, there were no compilation errors.
|
|
|
|
return executable;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getDebugConfiguration(ctx: Ctx, config: ra.Runnable): Promise<vscode.DebugConfiguration | undefined> {
|
|
|
|
const editor = ctx.activeRustEditor;
|
|
|
|
if (!editor) return;
|
|
|
|
|
|
|
|
const knownEngines: Record<string, DebugConfigProvider> = {
|
|
|
|
"vadimcn.vscode-lldb": getLldbDebugConfig,
|
|
|
|
"ms-vscode.cpptools": getCppvsDebugConfig
|
|
|
|
};
|
|
|
|
const debugOptions = ctx.config.debug;
|
|
|
|
|
|
|
|
let debugEngine = null;
|
|
|
|
if (debugOptions.engine === "auto") {
|
|
|
|
for (var engineId in knownEngines) {
|
|
|
|
debugEngine = vscode.extensions.getExtension(engineId);
|
|
|
|
if (debugEngine) break;
|
|
|
|
}
|
2020-05-11 15:00:15 +00:00
|
|
|
} else {
|
2020-05-11 13:06:57 +00:00
|
|
|
debugEngine = vscode.extensions.getExtension(debugOptions.engine);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!debugEngine) {
|
|
|
|
vscode.window.showErrorMessage(`Install [CodeLLDB](https://marketplace.visualstudio.com/items?itemName=vadimcn.vscode-lldb)`
|
|
|
|
+ ` or [MS C++ tools](https://marketplace.visualstudio.com/items?itemName=ms-vscode.cpptools) extension for debugging.`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
debugOutput.clear();
|
|
|
|
if (ctx.config.debug.openUpDebugPane) {
|
|
|
|
debugOutput.show(true);
|
|
|
|
}
|
|
|
|
|
2020-05-11 15:49:45 +00:00
|
|
|
const wsFolder = path.normalize(vscode.workspace.workspaceFolders![0].uri.fsPath); // folder exists or RA is not active.
|
|
|
|
function simplifyPath(p: string): string {
|
|
|
|
return path.normalize(p).replace(wsFolder, '${workspaceRoot}');
|
|
|
|
}
|
|
|
|
|
2020-05-11 13:06:57 +00:00
|
|
|
const executable = await getDebugExecutable(config);
|
2020-05-11 15:49:45 +00:00
|
|
|
const debugConfig = knownEngines[debugEngine.id](config, simplifyPath(executable), debugOptions.sourceFileMap);
|
2020-05-11 13:06:57 +00:00
|
|
|
if (debugConfig.type in debugOptions.engineSettings) {
|
|
|
|
const settingsMap = (debugOptions.engineSettings as any)[debugConfig.type];
|
|
|
|
for (var key in settingsMap) {
|
|
|
|
debugConfig[key] = settingsMap[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-14 10:48:02 +00:00
|
|
|
if (debugConfig.name === "run binary") {
|
2020-05-14 13:02:01 +00:00
|
|
|
// The LSP side: crates\rust-analyzer\src\main_loop\handlers.rs,
|
|
|
|
// fn to_lsp_runnable(...) with RunnableKind::Bin
|
2020-05-14 14:32:24 +00:00
|
|
|
debugConfig.name = `run ${path.basename(executable)}`;
|
2020-05-14 10:48:02 +00:00
|
|
|
}
|
|
|
|
|
2020-05-11 15:49:45 +00:00
|
|
|
if (debugConfig.cwd) {
|
|
|
|
debugConfig.cwd = simplifyPath(debugConfig.cwd);
|
|
|
|
}
|
|
|
|
|
2020-05-11 13:06:57 +00:00
|
|
|
return debugConfig;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function startDebugSession(ctx: Ctx, config: ra.Runnable): Promise<boolean> {
|
2020-05-13 12:51:15 +00:00
|
|
|
let debugConfig: vscode.DebugConfiguration | undefined = undefined;
|
|
|
|
let message = "";
|
|
|
|
|
2020-05-14 08:12:10 +00:00
|
|
|
const wsLaunchSection = vscode.workspace.getConfiguration("launch");
|
|
|
|
const configurations = wsLaunchSection.get<any[]>("configurations") || [];
|
|
|
|
|
|
|
|
const index = configurations.findIndex(c => c.name === config.label);
|
|
|
|
if (-1 !== index) {
|
|
|
|
debugConfig = configurations[index];
|
|
|
|
message = " (from launch.json)";
|
|
|
|
debugOutput.clear();
|
|
|
|
} else {
|
2020-05-13 12:51:15 +00:00
|
|
|
debugConfig = await getDebugConfiguration(ctx, config);
|
|
|
|
}
|
|
|
|
|
2020-05-11 13:06:57 +00:00
|
|
|
if (!debugConfig) return false;
|
|
|
|
|
2020-05-13 12:51:15 +00:00
|
|
|
debugOutput.appendLine(`Launching debug configuration${message}:`);
|
2020-05-11 13:06:57 +00:00
|
|
|
debugOutput.appendLine(JSON.stringify(debugConfig, null, 2));
|
|
|
|
return vscode.debug.startDebugging(undefined, debugConfig);
|
|
|
|
}
|