mirror of https://gitlab.com/basile.b/dexed.git
libdexed-d, return Pascal compatible strings, freed by ref counting
This commit is contained in:
parent
7932fb6382
commit
25074446f2
|
@ -7,7 +7,7 @@ import
|
||||||
import
|
import
|
||||||
dparse.lexer, dparse.ast, dparse.parser, dparse.rollback_allocator;
|
dparse.lexer, dparse.ast, dparse.parser, dparse.rollback_allocator;
|
||||||
import
|
import
|
||||||
iz.memory;
|
iz.memory, iz.containers;
|
||||||
|
|
||||||
export extern(C) int d_rt_init(){ return rt_init(); }
|
export extern(C) int d_rt_init(){ return rt_init(); }
|
||||||
export extern(C) int d_rt_term(){ return rt_term(); }
|
export extern(C) int d_rt_term(){ return rt_term(); }
|
||||||
|
@ -437,3 +437,67 @@ alias joinedFilesToFiles = (const char* a) => a[0 .. a.strlen]
|
||||||
.filter!exists
|
.filter!exists
|
||||||
.filter!(b => b != "")
|
.filter!(b => b != "")
|
||||||
.array;
|
.array;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Translation of the FPC ref counted array type.
|
||||||
|
*/
|
||||||
|
struct FpcArray(T)
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
|
||||||
|
size_t _refCount;
|
||||||
|
size_t _length;
|
||||||
|
T _data;
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts either and iz.containers.array or a D dynamic array to
|
||||||
|
* an array compatible with Freepascal built in arrays.
|
||||||
|
*
|
||||||
|
* The memory is allocated in the C heap and can be freed from
|
||||||
|
* the Pascal side, e.g on ref count decrement.
|
||||||
|
*/
|
||||||
|
static FpcArray!T* fromArray(A)(auto ref A array) nothrow @nogc
|
||||||
|
if (is(Unqual!A == T[]) || is(Unqual!A == iz.containers.Array!T))
|
||||||
|
{
|
||||||
|
alias RT = FpcArray!T;
|
||||||
|
enum isChar = is(Unqual!T == char);
|
||||||
|
size_t len = size_t.sizeof * 2 + T.sizeof * array.length + ubyte(isChar);
|
||||||
|
void* mem = getMem(len);
|
||||||
|
// init refcount to -1 and length to izArray length
|
||||||
|
*cast(size_t*) (mem + 0 ) = -1;
|
||||||
|
*cast(size_t*) (mem + size_t.sizeof) = array.length;
|
||||||
|
// copy izArray data
|
||||||
|
mem[size_t.sizeof * 2 .. len - ubyte(isChar)] = array.ptr[0.. array.length * T.sizeof];
|
||||||
|
// FreePascal specifies that strings are guaranteed to be zero terminated.
|
||||||
|
static if (isChar)
|
||||||
|
*cast(char*) (mem + len - 1) = '\0';
|
||||||
|
auto result = cast(RT*) (mem + size_t.sizeof * 2);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t length() const pure nothrow @nogc
|
||||||
|
{
|
||||||
|
return _length;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t refCount() const pure nothrow @nogc
|
||||||
|
{
|
||||||
|
return _refCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
inout(T)* data() inout pure nothrow @nogc
|
||||||
|
{
|
||||||
|
return &_data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest
|
||||||
|
{
|
||||||
|
enum test = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_è-(')àç_è";
|
||||||
|
Array!char s = test;
|
||||||
|
auto fpcString = (FpcArray!char).fromArray(s);
|
||||||
|
assert(fpcString.length == test.length);
|
||||||
|
assert(fpcString.data[0..fpcString.length] == test);
|
||||||
|
}
|
||||||
|
|
|
@ -25,7 +25,7 @@ import
|
||||||
* Returns:
|
* Returns:
|
||||||
* The serialized symbols, as a C string.
|
* The serialized symbols, as a C string.
|
||||||
*/
|
*/
|
||||||
export extern(C) const(char)* listSymbols(const(char)* src, bool deep)
|
export extern(C) FpcArray!char* listSymbols(const(char)* src, bool deep)
|
||||||
{
|
{
|
||||||
Appender!(AstErrors) errors;
|
Appender!(AstErrors) errors;
|
||||||
|
|
||||||
|
@ -53,8 +53,7 @@ export extern(C) const(char)* listSymbols(const(char)* src, bool deep)
|
||||||
}
|
}
|
||||||
|
|
||||||
sl.visit(mod);
|
sl.visit(mod);
|
||||||
const(char)* result = sl.serialize();
|
return sl.serialize();
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -137,7 +136,7 @@ static assert (!MustAddGcRange!(SymbolListBuilder!(ListFmt.Pas)));
|
||||||
destruct(fmtVisitor);
|
destruct(fmtVisitor);
|
||||||
static if (Fmt == ListFmt.Pas)
|
static if (Fmt == ListFmt.Pas)
|
||||||
{
|
{
|
||||||
destruct(pasStream);
|
//destruct(pasStream);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -171,12 +170,14 @@ static assert (!MustAddGcRange!(SymbolListBuilder!(ListFmt.Pas)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const(char)* serialize()
|
FpcArray!char* serialize()
|
||||||
{
|
{
|
||||||
static if (Fmt == ListFmt.Pas)
|
static if (Fmt == ListFmt.Pas)
|
||||||
{
|
{
|
||||||
pasStream.put(">\rend\0");
|
pasStream.put(">\rend");
|
||||||
return cast(typeof(return))pasStream[].dup.ptr;
|
import std.stdio;
|
||||||
|
writeln(pasStream.length);
|
||||||
|
return (FpcArray!char).fromArray(pasStream);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
|
@ -8,11 +8,11 @@ import
|
||||||
import
|
import
|
||||||
common;
|
common;
|
||||||
|
|
||||||
export extern(C) const(char)* todoItems(const(char)* joinedFiles)
|
export extern(C) FpcArray!(char)* todoItems(const(char)* joinedFiles)
|
||||||
{
|
{
|
||||||
scope Appender!string stream;
|
scope Appender!(char[]) stream;
|
||||||
scope LexerConfig config = LexerConfig("", StringBehavior.source);
|
scope LexerConfig config = LexerConfig("", StringBehavior.source);
|
||||||
scope StringCache cache = StringCache(4096);
|
scope StringCache cache = StringCache(4096);
|
||||||
stream.reserve(32);
|
stream.reserve(32);
|
||||||
stream.put("object TTodoItems\ritems=<");
|
stream.put("object TTodoItems\ritems=<");
|
||||||
foreach (fname; joinedFilesToFiles(joinedFiles))
|
foreach (fname; joinedFilesToFiles(joinedFiles))
|
||||||
|
@ -24,10 +24,10 @@ export extern(C) const(char)* todoItems(const(char)* joinedFiles)
|
||||||
.each!(t => analyze(t, fname, stream));
|
.each!(t => analyze(t, fname, stream));
|
||||||
}
|
}
|
||||||
stream.put(">end");
|
stream.put(">end");
|
||||||
return stream.data.toStringz();
|
return (FpcArray!char).fromArray(stream.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void analyze(const(Token) token, const(char)[] fname, ref Appender!string stream)
|
private void analyze(const(Token) token, const(char)[] fname, ref Appender!(char[]) stream)
|
||||||
{
|
{
|
||||||
string text = token.text.strip.patchPascalString;
|
string text = token.text.strip.patchPascalString;
|
||||||
string identifier;
|
string identifier;
|
||||||
|
|
|
@ -74,9 +74,9 @@ function listFilesImports(const files: PChar): PDStrings; cdecl; external libdex
|
||||||
// Get the variables necessary to compute the Halstead metrics of the functions within a module.
|
// Get the variables necessary to compute the Halstead metrics of the functions within a module.
|
||||||
function halsteadMetrics(const src: PChar): PChar; cdecl; external libdexedd_name;
|
function halsteadMetrics(const src: PChar): PChar; cdecl; external libdexedd_name;
|
||||||
// Get the list of declarations within a module.
|
// Get the list of declarations within a module.
|
||||||
function listSymbols(const src: PChar; deep: Boolean): PChar; cdecl; external libdexedd_name;
|
function listSymbols(const src: PChar; deep: Boolean): pointer; cdecl; external libdexedd_name;
|
||||||
// Get the TODO items located in `files` (list of files joined with pathseparaotr and null terminated)
|
// Get the TODO items located in `files` (list of files joined with pathseparaotr and null terminated)
|
||||||
function todoItems(joinedFiles: PChar): PChar; cdecl; external libdexedd_name;
|
function todoItems(joinedFiles: PChar): pointer; cdecl; external libdexedd_name;
|
||||||
|
|
||||||
(**
|
(**
|
||||||
* Gets the module name and the imports of the source code located in
|
* Gets the module name and the imports of the source code located in
|
||||||
|
|
|
@ -751,7 +751,7 @@ end;
|
||||||
|
|
||||||
procedure TSymbolListWidget.threadedParsing;
|
procedure TSymbolListWidget.threadedParsing;
|
||||||
begin
|
begin
|
||||||
fTreeDataFromThread := listSymbols(PChar(fSourcecodeForThread), fDeep);
|
fTreeDataFromThread := string(listSymbols(PChar(fSourcecodeForThread), fDeep));
|
||||||
end;
|
end;
|
||||||
|
|
||||||
procedure TSymbolListWidget.threadedParsingFinished(sender: TObject);
|
procedure TSymbolListWidget.threadedParsingFinished(sender: TObject);
|
||||||
|
|
|
@ -472,7 +472,7 @@ end;
|
||||||
|
|
||||||
procedure TTodoListWidget.threadedScanning;
|
procedure TTodoListWidget.threadedScanning;
|
||||||
begin
|
begin
|
||||||
fSerializedTodoItemFromThread := todoItems(PChar(fFileListForThread));
|
fSerializedTodoItemFromThread := string(todoItems(PChar(fFileListForThread)));
|
||||||
end;
|
end;
|
||||||
|
|
||||||
procedure TTodoListWidget.threadedScanningFinished(Sender : TObject);
|
procedure TTodoListWidget.threadedScanningFinished(Sender : TObject);
|
||||||
|
|
Loading…
Reference in New Issue