breakpoints support

This commit is contained in:
Vadim Lopatin 2015-12-14 12:22:35 +03:00
parent 61e281fcf8
commit 44f6c97428
7 changed files with 147 additions and 66 deletions

View File

@ -14,7 +14,7 @@
"copyFiles-windows": ["lib/win32/dcd-server.exe", "lib/win32/dcd-client.exe"],
"dependencies": {
"dlangui": "~>0.7.12",
"dlangui": "~>0.7.15",
"libdparse": "==0.2.0"
},

View File

@ -12,19 +12,21 @@ enum DebuggingState {
stopped
}
interface DebuggerCallback : ProgramExecutionStatusListener {
/// debugger message line
void onDebuggerMessage(string msg);
/// debugger is started and loaded program, you can set breakpoints at this time
void onProgramLoaded(bool successful, bool debugInfoLoaded);
/// state changed: running / paused / stopped
void onDebugState(DebuggingState state, string msg, int param);
void onResponse(ResponseCode code, string msg);
class Breakpoint {
int id;
string file;
string fullFilePath;
string projectFilePath;
int line;
bool enabled;
string projectName;
this() {
id = nextBreakpointId++;
}
}
private static __gshared nextBreakpointId = 1;
interface Debugger : ProgramExecution {
void setDebuggerCallback(DebuggerCallback callback);
void setDebuggerExecutable(string debuggerExecutable);
@ -43,6 +45,22 @@ interface Debugger : ProgramExecution {
void execStepIn();
/// step out
void execStepOut();
/// update list of breakpoints
void setBreakpoints(Breakpoint[] bp);
}
interface DebuggerCallback : ProgramExecutionStatusListener {
/// debugger message line
void onDebuggerMessage(string msg);
/// debugger is started and loaded program, you can set breakpoints at this time
void onProgramLoaded(bool successful, bool debugInfoLoaded);
/// state changed: running / paused / stopped
void onDebugState(DebuggingState state, string msg, int param);
void onResponse(ResponseCode code, string msg);
}
enum ResponseCode : int {
@ -173,6 +191,10 @@ class DebuggerProxy : Debugger, DebuggerCallback {
void execStepOut() {
_debugger.postRequest(delegate() { _debugger.execStepOut(); });
}
/// update list of breakpoints
void setBreakpoints(Breakpoint[] bp) {
_debugger.postRequest(delegate() { _debugger.setBreakpoints(bp); });
}
}
abstract class DebuggerBase : Thread, Debugger {

View File

@ -280,6 +280,12 @@ class GDBInterface : ConsoleDebuggerInterface {
_stepOutRequestId = sendCommand("-exec-finish");
}
/// update list of breakpoints
void setBreakpoints(Breakpoint[] bp) {
// TODO
}
// ~message
void handleStreamLineCLI(string s) {
Log.d("GDB CLI: ", s);

View File

@ -7,7 +7,6 @@ import dlangui.core.collections;
import dlangui.core.settings;
import std.path;
import std.file;
import std.json;
import std.utf;
import std.algorithm;
import std.process;
@ -589,9 +588,6 @@ class Project : WorkspaceItem {
_configurations = ProjectConfiguration.load(_projectFile);
Log.i("Project configurations: ", _configurations);
} catch (JSONException e) {
Log.e("Cannot parse json", e);
return false;
} catch (Exception e) {
Log.e("Cannot read project file", e);
return false;

View File

@ -8,6 +8,7 @@ import dlangide.workspace.idesettings;
const AVAILABLE_TOOLCHAINS = ["default", "dmd", "ldc", "gdc"];
const AVAILABLE_ARCH = ["default", "x86", "x86_64"];
/// local settings for project (not supposed to put under source control)
class ProjectSettings : SettingsFile {
this(string filename) {

View File

@ -1,16 +1,19 @@
module dlangide.workspace.workspace;
import dlangide.workspace.project;
import dlangide.workspace.workspacesettings;
import dlangide.ui.frame;
import dlangui.core.logger;
import dlangui.core.settings;
import std.conv;
import std.path;
import std.file;
import std.json;
import std.range;
import std.utf;
import std.algorithm;
import ddebug.common.debugger;
enum BuildOperation {
Build,
Clean,
@ -38,6 +41,7 @@ class WorkspaceException : Exception
}
immutable string WORKSPACE_EXTENSION = ".dlangidews";
immutable string WORKSPACE_SETTINGS_EXTENSION = ".wssettings";
/// return true if filename matches rules for workspace file names
bool isWorkspaceFile(string filename) {
@ -47,6 +51,8 @@ bool isWorkspaceFile(string filename) {
/// DlangIDE workspace
class Workspace : WorkspaceItem {
protected Project[] _projects;
protected SettingsFile _workspaceFile;
protected WorkspaceSettings _settings;
protected IDEFrame _frame;
protected BuildConfiguration _buildConfiguration;
@ -54,6 +60,8 @@ class Workspace : WorkspaceItem {
this(IDEFrame frame, string fname = null) {
super(fname);
_workspaceFile = new SettingsFile(fname);
_settings = new WorkspaceSettings(fname ? fname ~ WORKSPACE_SETTINGS_EXTENSION : null);
_frame = frame;
}
@ -73,6 +81,7 @@ class Workspace : WorkspaceItem {
@property void startupProject(Project project) {
_startupProject = project;
_frame.setProjectConfigurations(project.configurations.keys.map!(k => k.to!dstring).array);
_settings.startupProjectName = toUTF8(project.name);
}
/// setups currrent project configuration by name
@ -84,8 +93,19 @@ class Workspace : WorkspaceItem {
}
protected void fillStartupProject() {
if (!_startupProject && _projects.length)
startupProject = _projects[0];
string s = _settings.startupProjectName;
if ((!_startupProject || !_startupProject.name.toUTF8.equal(s)) && _projects.length) {
if (!s.empty) {
foreach(p; _projects) {
if (p.name.toUTF8.equal(s)) {
_startupProject = p;
}
}
}
if (!_startupProject) {
startupProject = _projects[0];
}
}
}
/// tries to find source file in one of projects, returns found project source file item, or null if not found
@ -128,32 +148,25 @@ class Workspace : WorkspaceItem {
return toForwardSlashSeparator(relativePath(path, _dir));
}
override bool save(string fname = null) {
if (fname.length > 0)
filename = fname;
try {
JSONValue content;
JSONValue[string] json;
json["name"] = JSONValue(toUTF8(_name));
json["description"] = JSONValue(toUTF8(_description));
JSONValue[string] projects;
foreach (Project p; _projects) {
if (p.isDependency)
continue; // don't save dependency
string pname = toUTF8(p.name);
string ppath = absoluteToRelativePath(p.filename);
projects[pname] = JSONValue(ppath);
}
json["projects"] = projects;
content = json;
string js = content.toPrettyString;
write(_filename, js);
} catch (JSONException e) {
Log.e("Cannot parse json", e);
if (!filename) // no file name specified
return false;
} catch (Exception e) {
Log.e("Cannot read workspace file", e);
_settings.save(filename ~ WORKSPACE_SETTINGS_EXTENSION);
_workspaceFile.setString("name", toUTF8(_name));
_workspaceFile.setString("description", toUTF8(_description));
Setting projects = _workspaceFile.objectByPath("projects", true);
projects.clear(SettingType.OBJECT);
foreach (Project p; _projects) {
if (p.isDependency)
continue; // don't save dependency
string pname = toUTF8(p.name);
string ppath = absoluteToRelativePath(p.filename);
projects[pname] = ppath;
}
if (!_workspaceFile.save(_filename, true)) {
Log.e("Cannot save workspace file");
return false;
}
return true;
@ -166,34 +179,29 @@ class Workspace : WorkspaceItem {
return false;
}
Log.d("Reading workspace from file ", _filename);
try {
string jsonSource = readText!string(_filename);
JSONValue json = parseJSON(jsonSource);
_name = toUTF32(json["name"].str);
_description = toUTF32(json["description"].str);
Log.d("workspace name: ", _name);
Log.d("workspace description: ", _description);
JSONValue projects = json["projects"];
foreach(string key, ref JSONValue value; projects) {
string path = value.str;
Log.d("project: ", key, " path:", path);
if (!isAbsolute(path))
path = buildNormalizedPath(_dir, path); //, "dub.json"
Project project = new Project(this, path);
_projects ~= project;
project.load();
}
string js = json.toPrettyString;
write(_filename, js);
} catch (JSONException e) {
Log.e("Cannot parse json", e);
if (!_workspaceFile.load(_filename)) {
Log.e("Cannot read workspace file");
return false;
} catch (Exception e) {
Log.e("Cannot read workspace file", e);
}
_settings.load(filename ~ WORKSPACE_SETTINGS_EXTENSION);
_name = toUTF32(_workspaceFile["name"].str);
_description = toUTF32(_workspaceFile["description"].str);
Log.d("workspace name: ", _name);
Log.d("workspace description: ", _description);
if (_name.empty()) {
Log.e("empty workspace name");
return false;
}
Setting projects = _workspaceFile.objectByPath("projects", true);
foreach(string key, Setting value; projects) {
string path = value.str;
Log.d("project: ", key, " path:", path);
if (!isAbsolute(path))
path = buildNormalizedPath(_dir, path); //, "dub.json"
Project project = new Project(this, path);
_projects ~= project;
project.load();
}
fillStartupProject();
return true;
}

View File

@ -2,15 +2,63 @@ module dlangide.workspace.workspacesettings;
import dlangui.core.settings;
import dlangui.core.i18n;
import ddebug.common.debugger;
/// local settings for workspace (not supposed to put under source control)
class WorkspaceSettings : SettingsFile {
this(string filename) {
super(filename);
}
private Breakpoint[] _breakpoints;
private string _startupProjectName;
@property string startupProjectName() {
return _startupProjectName;
}
@property void startupProjectName(string s) {
if (s.equal(_startupProjectName)) {
_startupProjectName = s;
save();
}
}
void setBreakpoints(Breakpoint[] bps) {
Setting obj = _setting.settingByPath("breakpoints", SettingType.ARRAY);
obj.clear(SettingType.ARRAY);
int index = 0;
foreach(bp; bps) {
Setting bpObj = new Setting();
bpObj.setInteger("id", bp.id);
bpObj.setString("file", bp.file);
bpObj.setInteger("line", bp.line);
bpObj.setBoolean("enabled", bp.enabled);
bpObj.setString("projectName", bp.projectName);
bpObj.setString("projectFilePath", bp.projectFilePath);
obj[index++] = bpObj;
}
_breakpoints = bps;
save();
}
Breakpoint[] getBreakpoints() {
return _breakpoints;
}
/// override to do something after loading - e.g. set defaults
override void afterLoad() {
_startupProjectName = _setting.getString("startupProject");
Setting obj = _setting.settingByPath("breakpoints", SettingType.ARRAY);
_breakpoints = null;
for (int i = 0; i < obj.length; i++) {
Breakpoint bp = new Breakpoint();
bp.id = cast(int)obj.getInteger("id");
bp.file = obj.getString("file");
bp.line = cast(int)obj.getInteger("line");
bp.enabled = obj.getBoolean("enabled");
_breakpoints ~= bp;
}
}
override void updateDefaults() {