mirror of https://github.com/adamdruppe/arsd.git
omg so much scary stuff
This commit is contained in:
parent
3316e500a5
commit
046d55e880
|
@ -5,26 +5,27 @@ module arsd.declarativeloader;
|
||||||
|
|
||||||
import std.range;
|
import std.range;
|
||||||
|
|
||||||
// @VariableLength indicates the value is saved in a MIDI like format
|
///
|
||||||
// @BigEndian, @LittleEndian
|
|
||||||
// @NumBytes!Field or @NumElements!Field controls length of embedded arrays
|
|
||||||
// @Tagged!Field indicates a tagged union. Each struct within should have @Tag(X) which is a value of Field
|
|
||||||
// @MustBe() causes it to throw if not the given value
|
|
||||||
|
|
||||||
// @NotSaved indicates a struct member that is not actually saved in the file
|
|
||||||
|
|
||||||
enum BigEndian;
|
enum BigEndian;
|
||||||
|
///
|
||||||
enum LittleEndian;
|
enum LittleEndian;
|
||||||
|
/// @VariableLength indicates the value is saved in a MIDI like format
|
||||||
enum VariableLength;
|
enum VariableLength;
|
||||||
|
/// @NumBytes!Field or @NumElements!Field controls length of embedded arrays
|
||||||
struct NumBytes(alias field) {}
|
struct NumBytes(alias field) {}
|
||||||
|
/// ditto
|
||||||
struct NumElements(alias field) {}
|
struct NumElements(alias field) {}
|
||||||
|
/// @Tagged!Field indicates a tagged union. Each struct within should have @Tag(X) which is a value of Field
|
||||||
struct Tagged(alias field) {}
|
struct Tagged(alias field) {}
|
||||||
struct TagStruct(T) { T t; }
|
/// ditto
|
||||||
auto Tag(T)(T t) {
|
auto Tag(T)(T t) {
|
||||||
return TagStruct!T(t);
|
return TagStruct!T(t);
|
||||||
}
|
}
|
||||||
enum NotSaved;
|
struct TagStruct(T) { T t; }
|
||||||
struct MustBeStruct(T) { T t; }
|
struct MustBeStruct(T) { T t; }
|
||||||
|
/// The marked field is not in the actual file
|
||||||
|
enum NotSaved;
|
||||||
|
/// Insists the field must be a certain value, like for magic numbers
|
||||||
auto MustBe(T)(T t) {
|
auto MustBe(T)(T t) {
|
||||||
return MustBeStruct!T(t);
|
return MustBeStruct!T(t);
|
||||||
}
|
}
|
||||||
|
@ -65,7 +66,7 @@ union N(ty) {
|
||||||
ubyte[ty.sizeof] bytes;
|
ubyte[ty.sizeof] bytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
// input range of ubytes...
|
/// input range of ubytes...
|
||||||
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
||||||
int bytesConsumed;
|
int bytesConsumed;
|
||||||
ubyte next() {
|
ubyte next() {
|
||||||
|
@ -160,9 +161,19 @@ int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
while(numElementsRemaining) {
|
while(numElementsRemaining) {
|
||||||
|
//import std.stdio; writeln(memberName);
|
||||||
E piece;
|
E piece;
|
||||||
auto by = loadFrom(piece, r, endianness);
|
auto by = loadFrom(piece, r, endianness);
|
||||||
numElementsRemaining--;
|
numElementsRemaining--;
|
||||||
|
|
||||||
|
// such a filthy hack, needed for Java's mistake though :(
|
||||||
|
static if(__traits(compiles, piece.takesTwoSlots())) {
|
||||||
|
if(piece.takesTwoSlots()) {
|
||||||
|
__traits(getMember, t, memberName) ~= piece;
|
||||||
|
numElementsRemaining--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bytesConsumed += by;
|
bytesConsumed += by;
|
||||||
__traits(getMember, t, memberName) ~= piece;
|
__traits(getMember, t, memberName) ~= piece;
|
||||||
}
|
}
|
||||||
|
|
60
jni.d
60
jni.d
|
@ -293,8 +293,12 @@ inout(char)[] fixupJavaClassName(inout(char)[] s) {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct JavaTranslationConfig {
|
struct JavaTranslationConfig {
|
||||||
|
/// List the Java methods, imported to D.
|
||||||
bool doImports;
|
bool doImports;
|
||||||
|
/// List the native methods, assuming they should be exported from D
|
||||||
bool doExports;
|
bool doExports;
|
||||||
|
/// Put implementations inline. If false, this separates interface from impl for quicker builds with dmd -i.
|
||||||
|
bool inlineImplementations;
|
||||||
}
|
}
|
||||||
|
|
||||||
void rawClassBytesToD()(ubyte[] classBytes, string dPackagePrefix, string outputDirectory, JavaTranslationConfig jtc) {
|
void rawClassBytesToD()(ubyte[] classBytes, string dPackagePrefix, string outputDirectory, JavaTranslationConfig jtc) {
|
||||||
|
@ -345,8 +349,15 @@ void rawClassBytesToD()(ubyte[] classBytes, string dPackagePrefix, string output
|
||||||
thisModule ~= ".";
|
thisModule ~= ".";
|
||||||
thisModule ~= lastClassName;
|
thisModule ~= lastClassName;
|
||||||
|
|
||||||
dco = "module " ~ thisModule ~ ";\n\n";
|
bool isInterface = (cf.access_flags & 0x0200) ? true : false;
|
||||||
dco ~= "import arsd.jni : IJavaObjectImplementation, JavaPackageId, ImportExportImpl, JavaName, IJavaObject;\n\n";
|
|
||||||
|
if(jtc.inlineImplementations) {
|
||||||
|
dco = "module " ~ thisModule ~ ";\n\n";
|
||||||
|
} else {
|
||||||
|
dco ~= "module " ~ thisModule ~ "_d_interface;\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
dco ~= "import arsd.jni : IJavaObjectImplementation, JavaPackageId, JavaName, IJavaObject, ImportExportImpl;\n\n";
|
||||||
|
|
||||||
string[string] javaPackages;
|
string[string] javaPackages;
|
||||||
|
|
||||||
|
@ -354,9 +365,9 @@ void rawClassBytesToD()(ubyte[] classBytes, string dPackagePrefix, string output
|
||||||
if(lastClassName != originalClassName)
|
if(lastClassName != originalClassName)
|
||||||
dc ~= "@JavaName(\""~originalClassName~"\")\n";
|
dc ~= "@JavaName(\""~originalClassName~"\")\n";
|
||||||
|
|
||||||
// FIXME: what if it is an interface?
|
// so overriding Java classes from D is iffy and with separate implementation
|
||||||
|
// non final leads to linker errors anyway...
|
||||||
dc ~= "final class " ~ lastClassName ~ " : IJavaObject {\n";// JavaClass!(\""~javaPackage.replace("/", ".")~"\", "~lastClassName~") {\n";
|
dc ~= (isInterface ? "interface " : "final class ") ~ lastClassName ~ " : IJavaObject {\n";
|
||||||
foreach(method; cf.methodsListing) {
|
foreach(method; cf.methodsListing) {
|
||||||
bool native = (method.flags & 0x0100) ? true : false;
|
bool native = (method.flags & 0x0100) ? true : false;
|
||||||
if(native && !jtc.doExports)
|
if(native && !jtc.doExports)
|
||||||
|
@ -393,30 +404,51 @@ void rawClassBytesToD()(ubyte[] classBytes, string dPackagePrefix, string output
|
||||||
string ret = ctor ? "" : javaSignatureToDTypeString(retJava, javaPackages);
|
string ret = ctor ? "" : javaSignatureToDTypeString(retJava, javaPackages);
|
||||||
string args = javaSignatureToDTypeString(argsJava, javaPackages);
|
string args = javaSignatureToDTypeString(argsJava, javaPackages);
|
||||||
|
|
||||||
|
if(!jtc.inlineImplementations) {
|
||||||
|
if(ctor && args.length == 0)
|
||||||
|
continue; // FIXME skipping default ctor to avoid factory from trying to get to it in separate compilation
|
||||||
|
}
|
||||||
|
|
||||||
dc ~= "\t"~port~" " ~ ret ~ (ret.length ? " " : "") ~ (ctor ? "this" : name) ~ "("~args~")"~(native ? " {}" : ";")~"\n";
|
dc ~= "\t"~port~" " ~ ret ~ (ret.length ? " " : "") ~ (ctor ? "this" : name) ~ "("~args~")"~(native ? " {}" : ";")~"\n";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME what if there is a name conflict? the prefix kinda handles it but i dont like how ugly it is
|
|
||||||
foreach(pkg, prefix; javaPackages) {
|
foreach(pkg, prefix; javaPackages) {
|
||||||
auto m = (dPackagePrefix.length ? (dPackagePrefix ~ ".") : "") ~ pkg;
|
auto m = (dPackagePrefix.length ? (dPackagePrefix ~ ".") : "") ~ pkg;
|
||||||
// keeping thisModule because of the prefix nonsense
|
// keeping thisModule because of the prefix nonsense
|
||||||
//if(m == thisModule)
|
//if(m == thisModule)
|
||||||
//continue;
|
//continue;
|
||||||
dco ~= "import " ~ prefix ~ " = " ~ m ~ ";\n";
|
if(jtc.inlineImplementations)
|
||||||
|
dco ~= "import " ~ prefix ~ " = " ~ m ~ ";\n";
|
||||||
|
else
|
||||||
|
dco ~= "import " ~ prefix ~ " = " ~ m ~ "_d_interface;\n";
|
||||||
}
|
}
|
||||||
if(javaPackages.keys.length)
|
if(javaPackages.keys.length)
|
||||||
dco ~= "\n";
|
dco ~= "\n";
|
||||||
dco ~= dc;
|
dco ~= dc;
|
||||||
|
|
||||||
dco ~= "\tmixin IJavaObjectImplementation!(false);\n";
|
if(!isInterface)
|
||||||
|
dco ~= "\tmixin IJavaObjectImplementation!(false);\n";
|
||||||
dco ~= "\tmixin JavaPackageId!(\""~originalJavaPackage.replace("/", ".")~"\", \""~originalClassName~"\");\n";
|
dco ~= "\tmixin JavaPackageId!(\""~originalJavaPackage.replace("/", ".")~"\", \""~originalClassName~"\");\n";
|
||||||
|
|
||||||
dco ~= "}\n";
|
dco ~= "}\n";
|
||||||
|
|
||||||
dco ~= "\nmixin ImportExportImpl!"~lastClassName~";\n";
|
if(jtc.inlineImplementations) {
|
||||||
|
if(!isInterface)
|
||||||
|
dco ~= "\nmixin ImportExportImpl!"~lastClassName~";\n";
|
||||||
|
std.file.write(filename, dco);
|
||||||
|
} else {
|
||||||
|
string impl;
|
||||||
|
impl ~= "module " ~ thisModule ~ ";\n";
|
||||||
|
impl ~= "public import " ~ thisModule ~ "_d_interface;\n\n";
|
||||||
|
if(!isInterface) {
|
||||||
|
impl ~= "import arsd.jni : ImportExportImpl;\n";
|
||||||
|
impl ~= "mixin ImportExportImpl!"~lastClassName~";\n";
|
||||||
|
}
|
||||||
|
|
||||||
std.file.write(filename, dco);
|
std.file.write(filename, impl);
|
||||||
|
std.file.write(filename[0 .. $-2] ~ "_d_interface.d", dco);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
string javaSignatureToDTypeString(ref const(char)[] js, ref string[string] javaPackages) {
|
string javaSignatureToDTypeString(ref const(char)[] js, ref string[string] javaPackages) {
|
||||||
|
@ -439,7 +471,7 @@ string javaSignatureToDTypeString(ref const(char)[] js, ref string[string] javaP
|
||||||
if(type == "java/lang/String") {
|
if(type == "java/lang/String") {
|
||||||
type = "string"; // or could be wstring...
|
type = "string"; // or could be wstring...
|
||||||
} else if(type == "java/lang/Object") {
|
} else if(type == "java/lang/Object") {
|
||||||
type = "IJavaObject"; // or could be wstring...
|
type = "IJavaObject";
|
||||||
} else {
|
} else {
|
||||||
// NOTE rughs strings in this file
|
// NOTE rughs strings in this file
|
||||||
type = type.replace("$", "_");
|
type = type.replace("$", "_");
|
||||||
|
@ -1520,7 +1552,7 @@ interface IJavaObject {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static T fromExistingJavaObject(T)(jobject o) if(is(T : IJavaObject) && !is(T == IJavaObject)) {
|
static T fromExistingJavaObject(T)(jobject o) if(is(T : IJavaObject) && !is(T == interface)) {
|
||||||
import core.memory;
|
import core.memory;
|
||||||
auto ptr = GC.malloc(__traits(classInstanceSize, T));
|
auto ptr = GC.malloc(__traits(classInstanceSize, T));
|
||||||
ptr[0 .. __traits(classInstanceSize, T)] = typeid(T).initializer[];
|
ptr[0 .. __traits(classInstanceSize, T)] = typeid(T).initializer[];
|
||||||
|
@ -1529,12 +1561,12 @@ static T fromExistingJavaObject(T)(jobject o) if(is(T : IJavaObject) && !is(T ==
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
static auto fromExistingJavaObject(T)(jobject o) if(is(T == IJavaObject)) {
|
static auto fromExistingJavaObject(T)(jobject o) if(is(T == interface)) {
|
||||||
static class Dummy : IJavaObject {
|
static class Dummy : IJavaObject {
|
||||||
mixin IJavaObjectImplementation!(false);
|
mixin IJavaObjectImplementation!(false);
|
||||||
mixin JavaPackageId!("java.lang", "Object");
|
mixin JavaPackageId!("java.lang", "Object");
|
||||||
}
|
}
|
||||||
return fromExistingJavaObject!Dummy(o);
|
return cast(T) cast(void*) fromExistingJavaObject!Dummy(o); // FIXME this is so wrong
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
5
rpc.d
5
rpc.d
|
@ -6,6 +6,11 @@ module arsd.rpc;
|
||||||
1) integrate with arsd.eventloop
|
1) integrate with arsd.eventloop
|
||||||
2) make it easy to use with other processes; pipe to a process and talk to it that way. perhaps with shared memory too?
|
2) make it easy to use with other processes; pipe to a process and talk to it that way. perhaps with shared memory too?
|
||||||
3) extend the serialization capabilities
|
3) extend the serialization capabilities
|
||||||
|
|
||||||
|
|
||||||
|
@Throws!(List, Of, Exceptions)
|
||||||
|
classes are also RPC proxied
|
||||||
|
stdin/out/err also redirected
|
||||||
*/
|
*/
|
||||||
|
|
||||||
///+ //example usage
|
///+ //example usage
|
||||||
|
|
Loading…
Reference in New Issue