libdexed-d, return Pascal compatible strings, freed by ref counting

This commit is contained in:
Basile Burg 2020-06-12 19:32:51 +02:00
parent 7932fb6382
commit 25074446f2
6 changed files with 82 additions and 17 deletions

View File

@ -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);
}

View File

@ -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
{ {

View File

@ -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;

View File

@ -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

View File

@ -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);

View File

@ -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);