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
|
||||
dparse.lexer, dparse.ast, dparse.parser, dparse.rollback_allocator;
|
||||
import
|
||||
iz.memory;
|
||||
iz.memory, iz.containers;
|
||||
|
||||
export extern(C) int d_rt_init(){ return rt_init(); }
|
||||
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!(b => b != "")
|
||||
.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:
|
||||
* 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;
|
||||
|
||||
|
@ -53,8 +53,7 @@ export extern(C) const(char)* listSymbols(const(char)* src, bool deep)
|
|||
}
|
||||
|
||||
sl.visit(mod);
|
||||
const(char)* result = sl.serialize();
|
||||
return result;
|
||||
return sl.serialize();
|
||||
}
|
||||
|
||||
private:
|
||||
|
@ -137,7 +136,7 @@ static assert (!MustAddGcRange!(SymbolListBuilder!(ListFmt.Pas)));
|
|||
destruct(fmtVisitor);
|
||||
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)
|
||||
{
|
||||
pasStream.put(">\rend\0");
|
||||
return cast(typeof(return))pasStream[].dup.ptr;
|
||||
pasStream.put(">\rend");
|
||||
import std.stdio;
|
||||
writeln(pasStream.length);
|
||||
return (FpcArray!char).fromArray(pasStream);
|
||||
}
|
||||
else
|
||||
{
|
||||
|
|
|
@ -8,11 +8,11 @@ import
|
|||
import
|
||||
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 StringCache cache = StringCache(4096);
|
||||
scope StringCache cache = StringCache(4096);
|
||||
stream.reserve(32);
|
||||
stream.put("object TTodoItems\ritems=<");
|
||||
foreach (fname; joinedFilesToFiles(joinedFiles))
|
||||
|
@ -24,10 +24,10 @@ export extern(C) const(char)* todoItems(const(char)* joinedFiles)
|
|||
.each!(t => analyze(t, fname, stream));
|
||||
}
|
||||
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 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.
|
||||
function halsteadMetrics(const src: PChar): PChar; cdecl; external libdexedd_name;
|
||||
// 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)
|
||||
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
|
||||
|
|
|
@ -751,7 +751,7 @@ end;
|
|||
|
||||
procedure TSymbolListWidget.threadedParsing;
|
||||
begin
|
||||
fTreeDataFromThread := listSymbols(PChar(fSourcecodeForThread), fDeep);
|
||||
fTreeDataFromThread := string(listSymbols(PChar(fSourcecodeForThread), fDeep));
|
||||
end;
|
||||
|
||||
procedure TSymbolListWidget.threadedParsingFinished(sender: TObject);
|
||||
|
|
|
@ -472,7 +472,7 @@ end;
|
|||
|
||||
procedure TTodoListWidget.threadedScanning;
|
||||
begin
|
||||
fSerializedTodoItemFromThread := todoItems(PChar(fFileListForThread));
|
||||
fSerializedTodoItemFromThread := string(todoItems(PChar(fFileListForThread)));
|
||||
end;
|
||||
|
||||
procedure TTodoListWidget.threadedScanningFinished(Sender : TObject);
|
||||
|
|
Loading…
Reference in New Issue