mirror of https://github.com/buggins/dlangide.git
MI parser
This commit is contained in:
parent
c0574c8219
commit
d760fe671d
|
@ -10,6 +10,7 @@ import std.utf;
|
||||||
import std.conv : to;
|
import std.conv : to;
|
||||||
import std.array : empty;
|
import std.array : empty;
|
||||||
import std.algorithm : startsWith, equal;
|
import std.algorithm : startsWith, equal;
|
||||||
|
import core.thread;
|
||||||
|
|
||||||
abstract class ConsoleDebuggerInterface : DebuggerBase, TextWriter {
|
abstract class ConsoleDebuggerInterface : DebuggerBase, TextWriter {
|
||||||
protected ExternalProcess _debuggerProcess;
|
protected ExternalProcess _debuggerProcess;
|
||||||
|
@ -202,6 +203,7 @@ class GDBInterface : ConsoleDebuggerInterface {
|
||||||
ExternalProcessState state = runDebuggerProcess(_debuggerExecutable, debuggerArgs, _executableWorkingDir);
|
ExternalProcessState state = runDebuggerProcess(_debuggerExecutable, debuggerArgs, _executableWorkingDir);
|
||||||
Log.i("Debugger process state:");
|
Log.i("Debugger process state:");
|
||||||
if (state == ExternalProcessState.Running) {
|
if (state == ExternalProcessState.Running) {
|
||||||
|
Thread.sleep(dur!"seconds"(1));
|
||||||
_callback.onProgramLoaded(true, true);
|
_callback.onProgramLoaded(true, true);
|
||||||
//sendCommand("-break-insert main");
|
//sendCommand("-break-insert main");
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -6,6 +6,53 @@ import std.conv : to;
|
||||||
import std.array : empty;
|
import std.array : empty;
|
||||||
import std.algorithm : startsWith, equal;
|
import std.algorithm : startsWith, equal;
|
||||||
|
|
||||||
|
/// result class
|
||||||
|
enum ResultClass {
|
||||||
|
done,
|
||||||
|
running,
|
||||||
|
connected,
|
||||||
|
error,
|
||||||
|
exit,
|
||||||
|
other
|
||||||
|
}
|
||||||
|
|
||||||
|
/// async message class
|
||||||
|
enum AsyncClass {
|
||||||
|
running,
|
||||||
|
stopped,
|
||||||
|
library_loaded,
|
||||||
|
library_unloaded,
|
||||||
|
thread_group_added,
|
||||||
|
thread_group_started,
|
||||||
|
thread_group_exited,
|
||||||
|
thread_created,
|
||||||
|
thread_exited,
|
||||||
|
other
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse GDB MI output string
|
||||||
|
MIValue parseMI(string s) {
|
||||||
|
string src = s;
|
||||||
|
try {
|
||||||
|
bool err = false;
|
||||||
|
Log.e("Tokenizing MI output: " ~ src);
|
||||||
|
MIToken[] tokens = tokenizeMI(s, err);
|
||||||
|
if (err) {
|
||||||
|
// tokenizer error
|
||||||
|
Log.e("Cannot tokenize MI output `" ~ src ~ "`");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Log.v("Parsing tokens " ~ tokens.dumpTokens);
|
||||||
|
MIValue[] items = parseMIList(tokens);
|
||||||
|
MIList res = new MIList(items);
|
||||||
|
Log.d("Parse result:\n" ~ res.toString);
|
||||||
|
return res;
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e("Cannot parse MI output `" ~ src ~ "`", e.msg);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
string parseIdent(ref string s) {
|
string parseIdent(ref string s) {
|
||||||
string res = null;
|
string res = null;
|
||||||
int len = 0;
|
int len = 0;
|
||||||
|
@ -44,15 +91,6 @@ ResultClass resultByName(string s) {
|
||||||
return ResultClass.other;
|
return ResultClass.other;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum ResultClass {
|
|
||||||
done,
|
|
||||||
running,
|
|
||||||
connected,
|
|
||||||
error,
|
|
||||||
exit,
|
|
||||||
other
|
|
||||||
}
|
|
||||||
|
|
||||||
AsyncClass asyncByName(string s) {
|
AsyncClass asyncByName(string s) {
|
||||||
if (s.equal("stopped")) return AsyncClass.stopped;
|
if (s.equal("stopped")) return AsyncClass.stopped;
|
||||||
if (s.equal("running")) return AsyncClass.running;
|
if (s.equal("running")) return AsyncClass.running;
|
||||||
|
@ -66,19 +104,6 @@ AsyncClass asyncByName(string s) {
|
||||||
return AsyncClass.other;
|
return AsyncClass.other;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum AsyncClass {
|
|
||||||
running,
|
|
||||||
stopped,
|
|
||||||
library_loaded,
|
|
||||||
library_unloaded,
|
|
||||||
thread_group_added,
|
|
||||||
thread_group_started,
|
|
||||||
thread_group_exited,
|
|
||||||
thread_created,
|
|
||||||
thread_exited,
|
|
||||||
other
|
|
||||||
}
|
|
||||||
|
|
||||||
enum MITokenType {
|
enum MITokenType {
|
||||||
/// end of line
|
/// end of line
|
||||||
eol,
|
eol,
|
||||||
|
@ -109,6 +134,10 @@ struct MIToken {
|
||||||
this.type = type;
|
this.type = type;
|
||||||
this.str = str;
|
this.str = str;
|
||||||
}
|
}
|
||||||
|
string toString() {
|
||||||
|
//return type == MITokenType.str ? to!string(type) ~ ":\"" ~ str ~ "\"": to!string(type) ~ ":" ~ str;
|
||||||
|
return (type == MITokenType.str) ? "\"" ~ str ~ "\"" : "`" ~ str ~ "`";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MIToken parseMIToken(ref string s) {
|
MIToken parseMIToken(ref string s) {
|
||||||
|
@ -173,33 +202,11 @@ MIToken[] tokenizeMI(string s, out bool error) {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parse GDB MI output string
|
string dumpTokens(MIToken[] tokens, int maxTokens = 40, int maxChars = 80) {
|
||||||
MIValue parseMI(string s) {
|
|
||||||
string src = s;
|
|
||||||
try {
|
|
||||||
bool err = false;
|
|
||||||
MIToken[] tokens = tokenizeMI(s, err);
|
|
||||||
if (err) {
|
|
||||||
// tokenizer error
|
|
||||||
Log.e("Cannot tokenize MI output `" ~ src ~ "`");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
MIValue[] items = parseMIList(tokens);
|
|
||||||
return new MIList(items);
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e("Cannot parse MI output `" ~ src ~ "`", e.msg);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
string dumpTokens(MIToken[] tokens) {
|
|
||||||
char[] buf;
|
char[] buf;
|
||||||
for (int i = 0; i < 10 && i < tokens.length; i++) {
|
for (int i = 0; i < maxTokens && i < tokens.length && buf.length < maxChars; i++) {
|
||||||
if (tokens[i].type == MITokenType.str)
|
buf ~= tokens[i].toString;
|
||||||
buf ~= '\"';
|
buf ~= ' ';
|
||||||
buf ~= tokens[i].str;
|
|
||||||
if (tokens[i].type == MITokenType.str)
|
|
||||||
buf ~= '\"';
|
|
||||||
}
|
}
|
||||||
return buf.dup;
|
return buf.dup;
|
||||||
}
|
}
|
||||||
|
@ -207,7 +214,7 @@ string dumpTokens(MIToken[] tokens) {
|
||||||
MIValue parseMIValue(ref MIToken[] tokens) {
|
MIValue parseMIValue(ref MIToken[] tokens) {
|
||||||
MIToken[] srctokens;
|
MIToken[] srctokens;
|
||||||
if (tokens.length == 0)
|
if (tokens.length == 0)
|
||||||
return null;
|
throw new Exception("parseMIValue: Unexpected end of line when value is expected");
|
||||||
MITokenType tokenType = tokens.length > 0 ? tokens[0].type : MITokenType.eol;
|
MITokenType tokenType = tokens.length > 0 ? tokens[0].type : MITokenType.eol;
|
||||||
MITokenType nextTokenType = tokens.length > 1 ? tokens[1].type : MITokenType.eol;
|
MITokenType nextTokenType = tokens.length > 1 ? tokens[1].type : MITokenType.eol;
|
||||||
if (tokenType == MITokenType.ident) {
|
if (tokenType == MITokenType.ident) {
|
||||||
|
@ -220,15 +227,15 @@ MIValue parseMIValue(ref MIToken[] tokens) {
|
||||||
tokens = tokens[1..$]; // skip ident
|
tokens = tokens[1..$]; // skip ident
|
||||||
tokens = tokens[1..$]; // skip =
|
tokens = tokens[1..$]; // skip =
|
||||||
MIValue value = parseMIValue(tokens);
|
MIValue value = parseMIValue(tokens);
|
||||||
tokens = tokens[1..$]; // skip value
|
//tokens = tokens[1..$]; // skip value
|
||||||
MIValue res = new MIKeyValue(ident, value);
|
MIValue res = new MIKeyValue(ident, value);
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
throw new Exception("parseMIValue: Unexpected token " ~ to!string(tokenType) ~ " near " ~ srctokens.dumpTokens);
|
throw new Exception("parseMIValue: Unexpected token " ~ tokens[0].toString ~ " near " ~ srctokens.dumpTokens);
|
||||||
} else if (tokenType == MITokenType.str) {
|
} else if (tokenType == MITokenType.str) {
|
||||||
string str = tokens[0].str;
|
string str = tokens[0].str;
|
||||||
tokens = tokens[1..$];
|
tokens = tokens[1..$];
|
||||||
MIValue res = new MIString(str);
|
return new MIString(str);
|
||||||
} else if (tokenType == MITokenType.curlyOpen) {
|
} else if (tokenType == MITokenType.curlyOpen) {
|
||||||
tokens = tokens[1..$];
|
tokens = tokens[1..$];
|
||||||
MIValue[] list = parseMIList(tokens, MITokenType.curlyClose);
|
MIValue[] list = parseMIList(tokens, MITokenType.curlyClose);
|
||||||
|
@ -238,27 +245,28 @@ MIValue parseMIValue(ref MIToken[] tokens) {
|
||||||
MIValue[] list = parseMIList(tokens, MITokenType.squareClose);
|
MIValue[] list = parseMIList(tokens, MITokenType.squareClose);
|
||||||
return new MIList(list);
|
return new MIList(list);
|
||||||
}
|
}
|
||||||
throw new Exception("parseMIValue: Invalid token at end of list: " ~ tokenType.to!string ~ " near " ~ srctokens.dumpTokens);
|
throw new Exception("parseMIValue: unexpected token " ~ tokens[0].toString ~ " near " ~ srctokens.dumpTokens);
|
||||||
}
|
}
|
||||||
|
|
||||||
MIValue[] parseMIList(ref MIToken[] tokens, MITokenType closingToken = MITokenType.eol) {
|
MIValue[] parseMIList(ref MIToken[] tokens, MITokenType closingToken = MITokenType.eol) {
|
||||||
|
Log.v("parseMIList: " ~ tokens.dumpTokens);
|
||||||
MIValue[] res;
|
MIValue[] res;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
MITokenType tokenType = tokens.length > 0 ? tokens[0].type : MITokenType.eol;
|
MITokenType tokenType = tokens.length > 0 ? tokens[0].type : MITokenType.eol;
|
||||||
if (tokenType == MITokenType.eol)
|
|
||||||
return res;
|
|
||||||
if (tokenType == closingToken) {
|
if (tokenType == closingToken) {
|
||||||
tokens = tokens[1..$];
|
tokens = tokens[1..$];
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
if (tokenType == MITokenType.eol)
|
||||||
|
throw new Exception("parseMIList: Unexpected eol in list");
|
||||||
|
if (res.length > 0) {
|
||||||
|
// comma required
|
||||||
|
if (tokenType != MITokenType.comma)
|
||||||
|
throw new Exception("parseMIList: comma expected, found " ~ tokens[0].toString ~ " near " ~ tokens.dumpTokens);
|
||||||
|
tokens = tokens[1..$];
|
||||||
|
}
|
||||||
MIValue value = parseMIValue(tokens);
|
MIValue value = parseMIValue(tokens);
|
||||||
res ~= value;
|
res ~= value;
|
||||||
tokenType = tokens.length > 0 ? tokens[0].type : MITokenType.eol;
|
|
||||||
if (tokenType == MITokenType.comma) {
|
|
||||||
tokens = tokens[1..$];
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
throw new Exception("parseMIList: Unexpected token " ~ to!string(tokenType) ~ " in list near " ~ tokens.dumpTokens);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -277,6 +285,11 @@ enum MIValueType {
|
||||||
map,
|
map,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void dumpLevel(ref char[] buf, int level) {
|
||||||
|
for (int i = 0; i < level; i++)
|
||||||
|
buf ~= " ";
|
||||||
|
}
|
||||||
|
|
||||||
class MIValue {
|
class MIValue {
|
||||||
MIValueType type;
|
MIValueType type;
|
||||||
this(MIValueType type) {
|
this(MIValueType type) {
|
||||||
|
@ -286,6 +299,15 @@ class MIValue {
|
||||||
@property int length() { return 1; }
|
@property int length() { return 1; }
|
||||||
MIValue opIndex(int index) { return null; }
|
MIValue opIndex(int index) { return null; }
|
||||||
MIValue opIndex(string key) { return null; }
|
MIValue opIndex(string key) { return null; }
|
||||||
|
void dump(ref char[] buf, int level) {
|
||||||
|
dumpLevel(buf, level);
|
||||||
|
buf ~= str;
|
||||||
|
}
|
||||||
|
override string toString() {
|
||||||
|
char[] buf;
|
||||||
|
dump(buf, 0);
|
||||||
|
return buf.dup;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MIKeyValue : MIValue {
|
class MIKeyValue : MIValue {
|
||||||
|
@ -299,6 +321,15 @@ class MIKeyValue : MIValue {
|
||||||
@property string key() { return _key; }
|
@property string key() { return _key; }
|
||||||
override @property string str() { return _key; }
|
override @property string str() { return _key; }
|
||||||
@property MIValue value() { return _value; }
|
@property MIValue value() { return _value; }
|
||||||
|
override void dump(ref char[] buf, int level) {
|
||||||
|
dumpLevel(buf, level);
|
||||||
|
buf ~= _key;
|
||||||
|
buf ~= "=";
|
||||||
|
if (!value)
|
||||||
|
buf ~= "null";
|
||||||
|
else
|
||||||
|
_value.dump(buf, level + 1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MIIdent : MIValue {
|
class MIIdent : MIValue {
|
||||||
|
@ -317,6 +348,12 @@ class MIString : MIValue {
|
||||||
_str = str;
|
_str = str;
|
||||||
}
|
}
|
||||||
override @property string str() { return _str; }
|
override @property string str() { return _str; }
|
||||||
|
override void dump(ref char[] buf, int level) {
|
||||||
|
dumpLevel(buf, level);
|
||||||
|
buf ~= '\"';
|
||||||
|
buf ~= str;
|
||||||
|
buf ~= '\"';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MIList : MIValue {
|
class MIList : MIValue {
|
||||||
|
@ -349,6 +386,20 @@ class MIList : MIValue {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
override void dump(ref char[] buf, int level) {
|
||||||
|
buf ~= (type == MIValueType.map) ? "{" : "[";
|
||||||
|
if (length) {
|
||||||
|
buf ~= "\n";
|
||||||
|
for (int i = 0; i < length; i++) {
|
||||||
|
_items[i].dump(buf, level + 1);
|
||||||
|
if (i < length - 1)
|
||||||
|
buf ~= ",";
|
||||||
|
}
|
||||||
|
buf ~= "\n";
|
||||||
|
dumpLevel(buf, level);
|
||||||
|
}
|
||||||
|
buf ~= (type == MIValueType.map) ? "}" : "]";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MIMap : MIList {
|
class MIMap : MIList {
|
||||||
|
@ -372,7 +423,6 @@ string parseCString(ref string s) {
|
||||||
char ch = nextChar(s);
|
char ch = nextChar(s);
|
||||||
if (!ch)
|
if (!ch)
|
||||||
return null;
|
return null;
|
||||||
s = s[1 .. $];
|
|
||||||
if (ch != '\"')
|
if (ch != '\"')
|
||||||
return null;
|
return null;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
|
Loading…
Reference in New Issue