diff --git a/dlangide-monod-linux.dproj b/dlangide-monod-linux.dproj
index 23a6d04..66e9e98 100644
--- a/dlangide-monod-linux.dproj
+++ b/dlangide-monod-linux.dproj
@@ -33,6 +33,7 @@
USE_OPENGL
USE_SDL
USE_FREETYPE
+ USE_GDB_DEBUG
EmbedStandardResources
@@ -192,5 +193,9 @@
+
+
+
+
\ No newline at end of file
diff --git a/src/ddebug/common/debugger.d b/src/ddebug/common/debugger.d
index 9c99a16..3176917 100644
--- a/src/ddebug/common/debugger.d
+++ b/src/ddebug/common/debugger.d
@@ -2,26 +2,106 @@ module ddebug.common.debugger;
import core.thread;
import dlangui.core.logger;
+import ddebug.common.queue;
+
+enum ResponseCode : int {
+ /// Operation finished successfully
+ Ok = 0,
+
+ // more success codes here
+
+ /// General purpose failure code
+ Fail = 1000,
+ /// method is not implemented
+ NotImplemented,
+ /// error running debugger
+ CannotRunDebugger,
+
+ // more error codes here
+}
+
+alias Runnable = void delegate();
+alias DebuggerResponse = void delegate(ResponseCode code, string msg);
interface Debugger {
/// start debugging
- void startDebugging(string executable, string[] args, string workingDir);
+ void startDebugging(string debuggerExecutable, string executable, string[] args, string workingDir, DebuggerResponse response);
}
-interface DebuggerCallback {
+
+
+/// proxy for debugger interface implementing async calls
+class DebuggerProxy : Debugger {
+ private DebuggerBase _debugger;
+ private void delegate(Runnable runnable) _callbackDelegate;
+
+ this(DebuggerBase debugger, void delegate(Runnable runnable) callbackDelegate) {
+ _debugger = debugger;
+ _callbackDelegate = callbackDelegate;
+ }
+
+ void startDebugging(string debuggerExecutable, string executable, string[] args, string workingDir, DebuggerResponse response) {
+ _debugger.postRequest(delegate() {
+ _debugger.startDebugging(debuggerExecutable, executable, args, workingDir,
+ delegate(ResponseCode code, string msg) {
+ _callbackDelegate( delegate() { response(code, msg); } );
+ }
+ );
+ });
+ }
+
}
-class DebuggerBase : Thread {
+class DebuggerBase : Thread, Debugger {
private bool _stopRequested;
private bool _finished;
+ protected string _debuggerExecutable;
+ protected BlockingQueue!Runnable _queue;
+
+ void postRequest(Runnable request) {
+ _queue.put(request);
+ }
this() {
super(&run);
+ _queue = new BlockingQueue!Runnable();
}
+ ~this() {
+ stop();
+ destroy(_queue);
+ _queue = null;
+ }
+
+ void stop() {
+ Log.i("Debugger.stop()");
+ _stopRequested = true;
+ _queue.close();
+ }
+
+ protected void onDebuggerThreadStarted() {
+ }
+
+ protected void onDebuggerThreadFinished() {
+ }
+
+ /// thread func: execute all tasks from queue
private void run() {
+ onDebuggerThreadStarted();
Log.i("Debugger thread started");
+ while (!_stopRequested) {
+ Runnable task;
+ if (_queue.get(task, 0)) {
+ task();
+ }
+ }
Log.i("Debugger thread finished");
_finished = true;
+ onDebuggerThreadFinished();
}
+
+ void startDebugging(string debuggerExecutable, string executable, string[] args, string workingDir, DebuggerResponse response) {
+ response(ResponseCode.NotImplemented, "Not Implemented");
+ }
+
}
diff --git a/src/ddebug/common/queue.d b/src/ddebug/common/queue.d
index 2e2a643..d4669b9 100644
--- a/src/ddebug/common/queue.d
+++ b/src/ddebug/common/queue.d
@@ -82,7 +82,6 @@ class BlockingQueue(T) {
foreach(ref item; items) {
append(item);
}
- append(item);
_condition.notifyAll();
}
}
@@ -99,7 +98,7 @@ class BlockingQueue(T) {
}
if (timeoutMillis <= 0)
_condition.wait(); // no timeout
- else if (!_condition.wait(dur!msecs(timeoutMillis)))
+ else if (!_condition.wait(dur!"msecs"(timeoutMillis)))
return false; // timeout
if (_readPos < _writePos) {
value = _buffer[_readPos++];
@@ -122,7 +121,7 @@ class BlockingQueue(T) {
return true;
if (timeoutMillis <= 0)
_condition.wait(); // no timeout
- else if (!_condition.wait(dur!msecs(timeoutMillis)))
+ else if (!_condition.wait(dur!"msecs"(timeoutMillis)))
return false; // timeout
while (_readPos < _writePos)
values ~= _buffer[_readPos++];
diff --git a/src/ddebug/gdb/gdbinterface.d b/src/ddebug/gdb/gdbinterface.d
new file mode 100644
index 0000000..76ea636
--- /dev/null
+++ b/src/ddebug/gdb/gdbinterface.d
@@ -0,0 +1,94 @@
+module ddebug.gdb.gdbinterface;
+
+public import ddebug.common.debugger;
+import dlangui.core.logger;
+import ddebug.common.queue;
+import dlangide.builders.extprocess;
+import std.utf;
+
+class ConsoleDebuggerInterface : DebuggerBase, TextWriter {
+ protected ExternalProcess _debuggerProcess;
+
+ protected ExternalProcessState runDebuggerProcess(string executable, string[]args, string dir) {
+ _debuggerProcess = new ExternalProcess();
+ ExternalProcessState state = _debuggerProcess.run(executable, args, dir, this);
+ return state;
+ }
+
+ private string[] _stdoutLines;
+ private char[] _stdoutBuf;
+ /// return true to clear lines list
+ protected bool onDebuggerStdoutLines(string[] lines) {
+ return true;
+ }
+ private void onStdoutText(string text) {
+ _stdoutBuf ~= text;
+ // pass full lines
+ int startPos = 0;
+ bool fullLinesFound = false;
+ for (int i = 0; i < _stdoutBuf.length; i++) {
+ if (_stdoutBuf[i] == '\n' || _stdoutBuf[i] == '\r') {
+ if (i <= startPos)
+ _stdoutLines ~= "";
+ else
+ _stdoutLines ~= _stdoutBuf[startPos .. i].dup;
+ fullLinesFound = true;
+ if (i + 1 < _stdoutBuf.length) {
+ if ((_stdoutBuf[i] == '\n' && _stdoutBuf[i + 1] == '\r')
+ || (_stdoutBuf[i] == '\r' && _stdoutBuf[i + 1] == '\n'))
+ i++;
+ }
+ startPos = i + 1;
+ }
+ }
+ if (fullLinesFound) {
+ for (int i = 0; i + startPos < _stdoutBuf.length; i++)
+ _stdoutBuf[i] = _stdoutBuf[i + startPos];
+ _stdoutBuf.length = _stdoutBuf.length - startPos;
+ if (onDebuggerStdoutLines(_stdoutLines)) {
+ _stdoutLines.length = 0;
+ }
+ }
+ }
+
+ /// log lines
+ override void writeText(dstring text) {
+ string text8 = toUTF8(text);
+ postRequest(delegate() {
+ onStdoutText(text8);
+ });
+ }
+
+}
+
+class GDBInterface : ConsoleDebuggerInterface {
+
+ override void startDebugging(string debuggerExecutable, string executable, string[] args, string workingDir, DebuggerResponse response) {
+ string[] debuggerArgs;
+ debuggerArgs ~= "--interpreter=mi";
+ debuggerArgs ~= "--silent";
+ debuggerArgs ~= "--args";
+ debuggerArgs ~= executable;
+ foreach(arg; args)
+ debuggerArgs ~= arg;
+ ExternalProcessState state = runDebuggerProcess(debuggerExecutable, debuggerArgs, workingDir);
+ Log.i("Debugger process state:");
+ if (state == ExternalProcessState.Running) {
+ response(ResponseCode.Ok, "Started");
+ } else {
+ response(ResponseCode.CannotRunDebugger, "Error while trying to run debugger process");
+ }
+ }
+
+ override void stop() {
+ if (_debuggerProcess !is null)
+ _debuggerProcess.kill();
+ super.stop();
+ }
+
+ /// return true to clear lines list
+ override protected bool onDebuggerStdoutLines(string[] lines) {
+ Log.d("onDebuggerStdout ", lines);
+ return true;
+ }
+}
diff --git a/src/dlangide.d b/src/dlangide.d
index 799ba3b..643bfb5 100644
--- a/src/dlangide.d
+++ b/src/dlangide.d
@@ -59,14 +59,17 @@ extern (C) int UIAppMain(string[] args) {
version(USE_WIN_DEBUG) {
debuggerTest();
}
+ version(USE_GDB_DEBUG) {
+ debuggerTest();
+ }
// create window
Window window = Platform.instance.createWindow("Dlang IDE", null, WindowFlag.Resizable, 800, 600);
// set window icon
window.windowIcon = drawableCache.getImage("dlangui-logo1");
- Widget w = new Widget();
- pragma(msg, w.click.return_t, "", w.click.params_t);
+ //Widget w = new Widget();
+ //pragma(msg, w.click.return_t, "", w.click.params_t);
IDEFrame frame = new IDEFrame(window);
@@ -93,6 +96,40 @@ version(USE_WIN_DEBUG) {
}
}
+version(USE_GDB_DEBUG) {
+ void debuggerTest() {
+ import ddebug.gdb.gdbinterface;
+ import core.thread;
+ Log.d("Testing GDB debugger");
+ DebuggerBase debugger = new DebuggerBase();
+ debugger.startDebugging("gdb", "test", [], "", delegate(ResponseCode code, string msg) {
+ Log.d("startDebugging result: ", code, " : ", msg);
+ //assert(code == ResponseCode.NotImplemented);
+ });
+ debugger.stop();
+ destroy(debugger);
+
+ // async
+
+ debugger = new GDBInterface();
+ DebuggerProxy proxy = new DebuggerProxy(debugger, delegate(Runnable runnable) {
+ runnable();
+ });
+ Log.d("calling debugger.start()");
+ debugger.start();
+ Log.d("calling proxy.startDebugging()");
+ proxy.startDebugging("gdb", "/home/lve/src/d/dlangide/test/gdbtest", ["param1", "param2"], "/home/lve/src/d/dlangide/test", delegate(ResponseCode code, string msg) {
+ Log.d("startDebugging result: ", code, " : ", msg);
+ //assert(code == ResponseCode.NotImplemented);
+ });
+ Thread.sleep(dur!"msecs"(2000));
+ debugger.stop();
+ Thread.sleep(dur!"msecs"(2000));
+ destroy(debugger);
+ Log.d("Testing of GDB debugger is finished");
+ }
+}
+
unittest {
void jsonTest() {
import dlangui.core.settings;