mirror of https://github.com/adamdruppe/arsd.git
Merge branch 'master' of github.com:adamdruppe/arsd
This commit is contained in:
commit
e8894916db
121
com.d
121
com.d
|
@ -45,15 +45,28 @@
|
|||
//writeln(obj.opDispatch!("test", int)());
|
||||
}
|
||||
---
|
||||
|
||||
I'll show a COM server example later. It is cool to call D objects
|
||||
from JScript and such.
|
||||
+/
|
||||
module arsd.com;
|
||||
|
||||
// for arrays to/from IDispatch use SAFEARRAY
|
||||
// see https://stackoverflow.com/questions/295067/passing-an-array-using-com
|
||||
|
||||
// for exceptions
|
||||
// see: https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-erref/705fb797-2175-4a90-b5a3-3918024b10b8
|
||||
// see: https://docs.microsoft.com/en-us/openspecs/windows_protocols/ms-erref/0c0bcf55-277e-4120-b5dc-f6115fc8dc38
|
||||
|
||||
/+
|
||||
see: program\comtest.d on the laptop.
|
||||
see: program\cs\comtest.d on the laptop.
|
||||
|
||||
as administrator: from program\cs
|
||||
c:\Windows\Microsoft.NEt\Framework64\v4.0.30319\regasm.exe /regfile /codebase test.dll
|
||||
|
||||
note: use the 64 bit register for 64 bit programs (Framework64)
|
||||
use 32 for 32 bit program (\Framework\)
|
||||
|
||||
sn -k key.snk
|
||||
program\cs\makefile
|
||||
|
||||
|
@ -69,8 +82,19 @@ module arsd.com;
|
|||
and then fully dynamic can be done with opDispatch for teh lulz.
|
||||
+/
|
||||
|
||||
/+
|
||||
createComObject returns the wrapped one
|
||||
wrapping can go dynamic if it is wrapping IDispatch
|
||||
some other IUnknown gets minimal wrapping (Translate formats)
|
||||
all wrappers can return lower level stuff on demand. like LL!string maybe is actually an RAII BSTR.
|
||||
|
||||
i also want variant to jsvar and stuff like that.
|
||||
createRawComObject returns the IUnknown raw one
|
||||
+/
|
||||
|
||||
import core.sys.windows.windows;
|
||||
import core.sys.windows.com;
|
||||
import core.sys.windows.wtypes;
|
||||
import core.sys.windows.oaidl;
|
||||
|
||||
import core.stdc.string;
|
||||
|
@ -167,7 +191,7 @@ template Dify(T) {
|
|||
import std.traits;
|
||||
|
||||
///
|
||||
struct ComClient(DVersion, ComVersion) {
|
||||
struct ComClient(DVersion, ComVersion = IDispatch) {
|
||||
ComVersion innerComObject_;
|
||||
this(ComVersion t) {
|
||||
this.innerComObject_ = t;
|
||||
|
@ -209,7 +233,8 @@ struct ComClient(DVersion, ComVersion) {
|
|||
static if(args.length) {
|
||||
VARIANT[args.length] vargs;
|
||||
foreach(idx, arg; args) {
|
||||
vargs[idx] = toComVariant(arg);
|
||||
// lol it is put in backwards way to explain MSFT
|
||||
vargs[$ - 1 - idx] = toComVariant(arg);
|
||||
}
|
||||
|
||||
disp_params.rgvarg = vargs.ptr;
|
||||
|
@ -217,16 +242,47 @@ struct ComClient(DVersion, ComVersion) {
|
|||
}
|
||||
|
||||
VARIANT result;
|
||||
EXCEPINFO einfo;
|
||||
uint argError;
|
||||
|
||||
ComCheck(innerComObject_.Invoke(
|
||||
//ComCheck(innerComObject_.Invoke(
|
||||
auto hr =innerComObject_.Invoke(
|
||||
dispid,
|
||||
&GUID_NULL, LOCALE_SYSTEM_DEFAULT, // whatever
|
||||
DISPATCH_METHOD,
|
||||
&disp_params,
|
||||
&result,
|
||||
null, // exception info
|
||||
null // arg error
|
||||
), "Invoke");
|
||||
&einfo, // exception info
|
||||
&argError // arg error
|
||||
);//, "Invoke");
|
||||
|
||||
import std.conv;
|
||||
if(FAILED(hr)) {
|
||||
if(hr == DISP_E_EXCEPTION) {
|
||||
auto code = einfo.scode ? einfo.scode : einfo.wCode;
|
||||
string source;
|
||||
string description;
|
||||
if(einfo.bstrSource) {
|
||||
// this is really a wchar[] but it needs to be freed so....
|
||||
source = einfo.bstrSource[0 .. SysStringLen(einfo.bstrSource)].to!string;
|
||||
SysFreeString(einfo.bstrSource);
|
||||
}
|
||||
if(einfo.bstrDescription) {
|
||||
description = einfo.bstrDescription[0 .. SysStringLen(einfo.bstrDescription)].to!string;
|
||||
SysFreeString(einfo.bstrDescription);
|
||||
}
|
||||
if(einfo.bstrHelpFile) {
|
||||
// FIXME: we could prolly use this too
|
||||
SysFreeString(einfo.bstrHelpFile);
|
||||
// and dwHelpContext
|
||||
}
|
||||
|
||||
throw new ComException(code, description ~ " (from com source " ~ source ~ ")");
|
||||
|
||||
} else {
|
||||
throw new ComException(hr, "Call failed");
|
||||
}
|
||||
}
|
||||
|
||||
return getFromVariant!(typeof(return))(result);
|
||||
} else {
|
||||
|
@ -338,14 +394,63 @@ T getFromVariant(T)(VARIANT arg) {
|
|||
static if(is(T == int)) {
|
||||
if(arg.vt == 3)
|
||||
return arg.intVal;
|
||||
} else static if(is(T == bool)) {
|
||||
if(arg.vt == 11)
|
||||
return arg.boolVal ? true : false;
|
||||
} else static if(is(T == string)) {
|
||||
if(arg.vt == 8) {
|
||||
auto str = arg.bstrVal;
|
||||
scope(exit) SysFreeString(str);
|
||||
return to!string(str[0 .. SysStringLen(str)]);
|
||||
}
|
||||
} else static if(is(T == IDispatch)) {
|
||||
if(arg.vt == 9)
|
||||
return arg.pdispVal;
|
||||
} else static if(is(T : IUnknown)) {
|
||||
// if(arg.vt == 13)
|
||||
static assert(0);
|
||||
} else static if(is(T == ComClient!(D, I), D, I)) {
|
||||
if(arg.vt == 9)
|
||||
return ComClient!(D, I)(arg.pdispVal);
|
||||
} else static if(is(T == E[], E)) {
|
||||
if(arg.vt & 0x2000) {
|
||||
auto elevt = arg.vt & ~0x2000;
|
||||
auto a = arg.parray;
|
||||
scope(exit) SafeArrayDestroy(a);
|
||||
|
||||
auto bounds = a.rgsabound.ptr[0 .. a.cDims];
|
||||
|
||||
auto hr = SafeArrayLock(a);
|
||||
if(SUCCEEDED(hr)) {
|
||||
scope(exit) SafeArrayUnlock(a);
|
||||
|
||||
// BTW this is where things get interesting with the
|
||||
// mid-level wrapper. it can avoid these copies
|
||||
|
||||
// maybe i should check bounds.lLbound too.....
|
||||
|
||||
static if(is(E == int)) {
|
||||
if(elevt == 3) {
|
||||
assert(a.cbElements == E.sizeof);
|
||||
return (cast(E*)a.pvData)[0 .. bounds[0].cElements].dup;
|
||||
}
|
||||
} else static if(is(E == string)) {
|
||||
if(elevt == 8) {
|
||||
//assert(a.cbElements == E.sizeof);
|
||||
//return (cast(E*)a.pvData)[0 .. bounds[0].cElements].dup;
|
||||
|
||||
string[] ret;
|
||||
foreach(item; (cast(BSTR*) a.pvData)[0 .. bounds[0].cElements]) {
|
||||
auto str = item;
|
||||
scope(exit) SysFreeString(str);
|
||||
ret ~= to!string(str[0 .. SysStringLen(str)]);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
throw new Exception("Type mismatch, needed "~ T.stringof ~"got " ~ to!string(arg.vt));
|
||||
assert(0);
|
||||
|
@ -432,7 +537,7 @@ mixin template IDispatchImpl() {
|
|||
} catch(Throwable e) {
|
||||
// FIXME: fill in the exception info
|
||||
if(except !is null) {
|
||||
except.wCode = 1;
|
||||
except.sCode = 1;
|
||||
import std.utf;
|
||||
except.bstrDescription = SysAllocString(toUTFz!(wchar*)(e.toString()));
|
||||
except.bstrSource = SysAllocString("amazing"w.ptr);
|
||||
|
|
|
@ -0,0 +1,178 @@
|
|||
/++
|
||||
A declarative file/stream loader/saver. You define structs with a handful of annotations, this read and writes them to/from files.
|
||||
+/
|
||||
module arsd.declarativeloader;
|
||||
|
||||
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 LittleEndian;
|
||||
enum VariableLength;
|
||||
struct NumBytes(alias field) {}
|
||||
struct NumElements(alias field) {}
|
||||
struct Tagged(alias field) {}
|
||||
struct TagStruct(T) { T t; }
|
||||
auto Tag(T)(T t) {
|
||||
return TagStruct!T(t);
|
||||
}
|
||||
enum NotSaved;
|
||||
struct MustBeStruct(T) { T t; }
|
||||
auto MustBe(T)(T t) {
|
||||
return MustBeStruct!T(t);
|
||||
}
|
||||
|
||||
static bool fieldSaved(alias a)() {
|
||||
bool saved;
|
||||
static if(is(typeof(a.offsetof))) {
|
||||
saved = true;
|
||||
static foreach(attr; __traits(getAttributes, a))
|
||||
static if(is(attr == NotSaved))
|
||||
saved = false;
|
||||
}
|
||||
return saved;
|
||||
}
|
||||
|
||||
static bool bigEndian(alias a)(bool def) {
|
||||
bool be = def;
|
||||
static foreach(attr; __traits(getAttributes, a)) {
|
||||
static if(is(attr == BigEndian))
|
||||
be = true;
|
||||
else static if(is(attr == LittleEndian))
|
||||
be = false;
|
||||
}
|
||||
return be;
|
||||
}
|
||||
|
||||
static auto getTag(alias a)() {
|
||||
static foreach(attr; __traits(getAttributes, a)) {
|
||||
static if(is(typeof(attr) == TagStruct!T, T)) {
|
||||
return attr.t;
|
||||
}
|
||||
}
|
||||
assert(0);
|
||||
}
|
||||
|
||||
union N(ty) {
|
||||
ty member;
|
||||
ubyte[ty.sizeof] bytes;
|
||||
}
|
||||
|
||||
// input range of ubytes...
|
||||
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
||||
int bytesConsumed;
|
||||
ubyte next() {
|
||||
auto bfr = r.front;
|
||||
r.popFront;
|
||||
bytesConsumed++;
|
||||
return bfr;
|
||||
}
|
||||
|
||||
bool endianness = bigEndian!T(assumeBigEndian);
|
||||
static foreach(memberName; __traits(allMembers, T)) {{
|
||||
static if(is(typeof(__traits(getMember, T, memberName)))) {
|
||||
alias f = __traits(getMember, T, memberName);
|
||||
alias ty = typeof(f);
|
||||
static if(fieldSaved!f) {
|
||||
endianness = bigEndian!f(endianness);
|
||||
// FIXME VariableLength
|
||||
static if(is(ty : ulong) || is(ty : double)) {
|
||||
N!ty n;
|
||||
if(endianness) {
|
||||
foreach(i; 0 .. ty.sizeof) {
|
||||
version(BigEndian)
|
||||
n.bytes[i] = next();
|
||||
else
|
||||
n.bytes[$ - 1 - i] = next();
|
||||
}
|
||||
} else {
|
||||
foreach(i; 0 .. ty.sizeof) {
|
||||
version(BigEndian)
|
||||
n.bytes[$ - 1 - i] = next();
|
||||
else
|
||||
n.bytes[i] = next();
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: MustBe
|
||||
|
||||
__traits(getMember, t, memberName) = n.member;
|
||||
} else static if(is(ty == struct)) {
|
||||
bytesConsumed += loadFrom(__traits(getMember, t, memberName), r, endianness);
|
||||
} else static if(is(ty == union)) {
|
||||
static foreach(attr; __traits(getAttributes, ty))
|
||||
static if(is(attr == Tagged!Field, alias Field))
|
||||
enum tagField = __traits(identifier, Field);
|
||||
static assert(is(typeof(tagField)), "Unions need a Tagged UDA on the union type (not the member) indicating the field that identifies the union");
|
||||
|
||||
auto tag = __traits(getMember, t, tagField);
|
||||
// find the child of the union matching the tag...
|
||||
static foreach(um; __traits(allMembers, ty)) {
|
||||
if(tag == getTag!(__traits(getMember, ty, um))) {
|
||||
bytesConsumed += loadFrom(__traits(getMember, __traits(getMember, t, memberName), um), r, endianness);
|
||||
}
|
||||
}
|
||||
} else static if(is(ty == E[], E)) {
|
||||
static foreach(attr; __traits(getAttributes, f)) {
|
||||
static if(is(attr == NumBytes!Field, alias Field))
|
||||
ulong numBytesRemaining = __traits(getMember, t, __traits(identifier, Field));
|
||||
else static if(is(attr == NumElements!Field, alias Field)) {
|
||||
ulong numElementsRemaining = __traits(getMember, t, __traits(identifier, Field));
|
||||
}
|
||||
}
|
||||
|
||||
static if(is(typeof(numBytesRemaining))) {
|
||||
static if(is(E : const(ubyte)) || is(E : const(char))) {
|
||||
while(numBytesRemaining) {
|
||||
__traits(getMember, t, memberName) ~= next;
|
||||
numBytesRemaining--;
|
||||
}
|
||||
} else {
|
||||
while(numBytesRemaining) {
|
||||
E piece;
|
||||
auto by = loadFrom(e, r, endianness);
|
||||
numBytesRemaining -= by;
|
||||
bytesConsumed += by;
|
||||
__traits(getMember, t, memberName) ~= piece;
|
||||
}
|
||||
}
|
||||
} else static if(is(typeof(numElementsRemaining))) {
|
||||
static if(is(E : const(ubyte)) || is(E : const(char))) {
|
||||
while(numElementsRemaining) {
|
||||
__traits(getMember, t, memberName) ~= next;
|
||||
numElementsRemaining--;
|
||||
}
|
||||
} else static if(is(E : const(ushort))) {
|
||||
while(numElementsRemaining) {
|
||||
ushort n;
|
||||
n = next << 8;
|
||||
n |= next;
|
||||
// FIXME all of this filth
|
||||
__traits(getMember, t, memberName) ~= n;
|
||||
numElementsRemaining--;
|
||||
}
|
||||
} else {
|
||||
while(numElementsRemaining) {
|
||||
E piece;
|
||||
auto by = loadFrom(piece, r, endianness);
|
||||
numElementsRemaining--;
|
||||
bytesConsumed += by;
|
||||
__traits(getMember, t, memberName) ~= piece;
|
||||
}
|
||||
}
|
||||
} else static assert(0, "no way to identify length... " ~ memberName);
|
||||
|
||||
} else static assert(0, ty.stringof);
|
||||
}
|
||||
}
|
||||
}}
|
||||
|
||||
return bytesConsumed;
|
||||
}
|
179
jni.d
179
jni.d
|
@ -191,35 +191,115 @@ private string getJavaName(alias a)() {
|
|||
return name;
|
||||
}
|
||||
|
||||
/+
|
||||
/+ Java class file definitions { +/
|
||||
// see: https://docs.oracle.com/javase/specs/jvms/se8/html/jvms-4.html#jvms-4.6
|
||||
|
||||
version(WithClassLoadSupport) {
|
||||
import arsd.declarativeloader;
|
||||
|
||||
struct cp_info {
|
||||
|
||||
enum CONSTANT_Class = 7; // sizeof = 2
|
||||
struct CONSTANT_Class_info {
|
||||
@BigEndian:
|
||||
ushort name_index;
|
||||
}
|
||||
enum CONSTANT_Fieldref = 9; // sizeof = 4
|
||||
struct CONSTANT_Fieldref_info {
|
||||
@BigEndian:
|
||||
ushort class_index;
|
||||
ushort name_and_type_index;
|
||||
}
|
||||
enum CONSTANT_Methodref = 10; // sizeof = 4
|
||||
struct CONSTANT_Methodref_info {
|
||||
@BigEndian:
|
||||
ushort class_index;
|
||||
ushort name_and_type_index;
|
||||
}
|
||||
enum CONSTANT_InterfaceMethodref = 11; // sizeof = 4
|
||||
struct CONSTANT_InterfaceMethodref_info {
|
||||
@BigEndian:
|
||||
ushort class_index;
|
||||
ushort name_and_type_index;
|
||||
}
|
||||
enum CONSTANT_String = 8; // sizeof = 2
|
||||
struct CONSTANT_String_info {
|
||||
@BigEndian:
|
||||
ushort string_index;
|
||||
}
|
||||
enum CONSTANT_Integer = 3; // sizeof = 4
|
||||
struct CONSTANT_Integer_info {
|
||||
@BigEndian:
|
||||
int bytes;
|
||||
}
|
||||
enum CONSTANT_Float = 4; // sizeof = 4
|
||||
struct CONSTANT_Float_info {
|
||||
@BigEndian:
|
||||
float bytes;
|
||||
}
|
||||
enum CONSTANT_Long = 5; // sizeof = 8, but eats two slots
|
||||
struct CONSTANT_Long_info {
|
||||
@BigEndian:
|
||||
long bytes;
|
||||
}
|
||||
enum CONSTANT_Double = 6; // sizeof = 8, but eats two slots
|
||||
struct CONSTANT_Double_info {
|
||||
@BigEndian:
|
||||
double bytes;
|
||||
}
|
||||
enum CONSTANT_NameAndType = 12; // sizeof = 2
|
||||
struct CONSTANT_NameAndType_info {
|
||||
@BigEndian:
|
||||
ushort name_index;
|
||||
ushort descriptor_index;
|
||||
}
|
||||
enum CONSTANT_Utf8 = 1; // sizeof = 2 + length
|
||||
struct CONSTANT_Utf8_info {
|
||||
@BigEndian:
|
||||
ushort length;
|
||||
@NumElements!length char[] bytes; // actually modified UTF-8 but meh
|
||||
}
|
||||
enum CONSTANT_MethodHandle = 15; // sizeof = 3
|
||||
struct CONSTANT_MethodHandle_info {
|
||||
@BigEndian:
|
||||
ubyte reference_kind;
|
||||
ushort reference_index;
|
||||
}
|
||||
enum CONSTANT_MethodType = 16; // sizeof = 2; descriptor index
|
||||
struct CONSTANT_MethodType_info {
|
||||
@BigEndian:
|
||||
ushort descriptor_index;
|
||||
}
|
||||
enum CONSTANT_InvokeDynamic = 18; // sizeof = 4
|
||||
struct CONSTANT_InvokeDynamic_info {
|
||||
@BigEndian:
|
||||
ushort bootstrap_method_attr_index;
|
||||
ushort name_and_type_index;
|
||||
}
|
||||
|
||||
ubyte tag;
|
||||
union {
|
||||
|
||||
@Tagged!(tag)
|
||||
union Info {
|
||||
@Tag(CONSTANT_Class) CONSTANT_Class_info class_info;
|
||||
@Tag(CONSTANT_Fieldref) CONSTANT_Fieldref_info fieldref_info;
|
||||
@Tag(CONSTANT_Methodref) CONSTANT_Methodref_info methodref_info;
|
||||
@Tag(CONSTANT_InterfaceMethodref) CONSTANT_InterfaceMethodref_info interfaceMethodref_info;
|
||||
@Tag(CONSTANT_String) CONSTANT_String_info string_info;
|
||||
@Tag(CONSTANT_Integer) CONSTANT_Integer_info integer_info;
|
||||
@Tag(CONSTANT_Float) CONSTANT_Float_info float_info;
|
||||
@Tag(CONSTANT_Long) CONSTANT_Long_info long_info;
|
||||
@Tag(CONSTANT_Double) CONSTANT_Double_info double_info;
|
||||
@Tag(CONSTANT_NameAndType) CONSTANT_NameAndType_info nameAndType_info;
|
||||
@Tag(CONSTANT_Utf8) CONSTANT_Utf8_info utf8_info;
|
||||
@Tag(CONSTANT_MethodHandle) CONSTANT_MethodHandle_info methodHandle_info;
|
||||
@Tag(CONSTANT_MethodType) CONSTANT_MethodType_info methodType_info;
|
||||
@Tag(CONSTANT_InvokeDynamic) CONSTANT_InvokeDynamic_info invokeDynamic_info;
|
||||
}
|
||||
ubyte[] info;
|
||||
Info info;
|
||||
}
|
||||
|
||||
struct field_info {
|
||||
@BigEndian:
|
||||
|
||||
enum ACC_PUBLIC = 0x0001;
|
||||
enum ACC_PRIVATE = 0x0002;
|
||||
|
@ -235,15 +315,16 @@ struct field_info {
|
|||
ushort name_index;
|
||||
ushort descriptor_index;
|
||||
ushort attributes_count;
|
||||
attribute_info attributes[attributes_count];
|
||||
@NumElements!attributes_count attribute_info[] attributes;
|
||||
}
|
||||
|
||||
struct method_info {
|
||||
@BigEndian:
|
||||
ushort access_flags;
|
||||
ushort name_index;
|
||||
ushort descriptor_index;
|
||||
ushort attributes_count;
|
||||
attribute_info attributes[attributes_count];
|
||||
@NumElements!attributes_count attribute_info[] attributes;
|
||||
|
||||
enum ACC_PUBLIC = 0x0001;
|
||||
enum ACC_PRIVATE = 0x0002;
|
||||
|
@ -260,12 +341,14 @@ struct method_info {
|
|||
}
|
||||
|
||||
struct attribute_info {
|
||||
@BigEndian:
|
||||
ushort attribute_name_index;
|
||||
uint attribute_length;
|
||||
ubyte[attribute_length] info;
|
||||
@NumBytes!attribute_length ubyte[] info;
|
||||
}
|
||||
|
||||
struct ClassFile {
|
||||
@BigEndian:
|
||||
|
||||
|
||||
enum ACC_PUBLIC = 0x0001;
|
||||
|
@ -277,27 +360,60 @@ struct ClassFile {
|
|||
enum ACC_ANNOTATION = 0x2000;
|
||||
enum ACC_ENUM = 0x4000;
|
||||
|
||||
const(char)[] className() {
|
||||
return this.constant(this.constant(this.this_class).info.class_info.name_index).info.utf8_info.bytes;
|
||||
}
|
||||
|
||||
uint magic;
|
||||
const(char)[] superclassName() {
|
||||
return this.constant(this.constant(this.super_class).info.class_info.name_index).info.utf8_info.bytes;
|
||||
}
|
||||
|
||||
Method[] methodsListing() {
|
||||
Method[] ms;
|
||||
foreach(met; this.methods) {
|
||||
Method m;
|
||||
m.name = this.constant(met.name_index).info.utf8_info.bytes;
|
||||
m.signature = this.constant(met.descriptor_index).info.utf8_info.bytes;
|
||||
m.flags = met.access_flags;
|
||||
ms ~= m;
|
||||
}
|
||||
return ms;
|
||||
}
|
||||
|
||||
static struct Method {
|
||||
const(char)[] name;
|
||||
const(char)[] signature;
|
||||
ushort flags;
|
||||
}
|
||||
|
||||
|
||||
@MustBe(0xcafebabe) uint magic;
|
||||
ushort minor_version;
|
||||
ushort major_version;
|
||||
ushort constant_pool_count;
|
||||
cp_info constant_pool[constant_pool_count-1];
|
||||
ushort constant_pool_count_;
|
||||
// the zeroth item of the constant pool is null, but not actually in the file.
|
||||
ushort constant_pool_count() { return cast(ushort)(constant_pool_count_ - 1); }
|
||||
auto constant(ushort number) {
|
||||
if(number == 0) throw new Exception("invalid");
|
||||
return constant_pool[number - 1];
|
||||
}
|
||||
@NumElements!constant_pool_count cp_info[] constant_pool;
|
||||
ushort access_flags;
|
||||
ushort this_class;
|
||||
ushort super_class;
|
||||
ushort interfaces_count;
|
||||
ushort interfaces[interfaces_count];
|
||||
@NumElements!interfaces_count ushort[] interfaces;
|
||||
ushort fields_count;
|
||||
field_info fields[fields_count];
|
||||
@NumElements!fields_count field_info[] fields;
|
||||
ushort methods_count;
|
||||
method_info methods[methods_count];
|
||||
@NumElements!methods_count method_info[] methods;
|
||||
ushort attributes_count;
|
||||
attribute_info attributes[attributes_count];
|
||||
@NumElements!attributes_count attribute_info[] attributes;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/+ } end java class file definitions +/
|
||||
+/
|
||||
|
||||
// semi-FIXME: java.lang.CharSequence is the interface for String. We should support that just as well.
|
||||
// possibly other boxed types too, like Integer.
|
||||
|
@ -560,6 +676,10 @@ private enum ImportImplementationString = q{
|
|||
auto ret = (*env).CallSTATICIntMethod(env, jobj, _jmethodID, DDataToJni(env, args).args);
|
||||
exceptionCheck(env);
|
||||
return ret;
|
||||
} else static if(is(typeof(return) : IJavaObject)) {
|
||||
auto ret = (*env).CallSTATICObjectMethod(env, jobj, _jmethodID, DDataToJni(env, args).args);
|
||||
exceptionCheck(env);
|
||||
return typeof(return).fromExistingJavaObject(ret);
|
||||
} else static if(is(typeof(return) == long)) {
|
||||
auto ret = (*env).CallSTATICLongMethod(env, jobj, _jmethodID, DDataToJni(env, args).args);
|
||||
exceptionCheck(env);
|
||||
|
@ -1085,6 +1205,15 @@ class JavaClass(string javaPackage, CRTP) : IJavaObject {
|
|||
enum Import; /// UDA to indicate you are importing the method from Java. Do NOT put a body on these methods.
|
||||
enum Export; /// UDA to indicate you are exporting the method to Java. Put a D implementation body on these.
|
||||
|
||||
static CRTP fromExistingJavaObject(jobject o) {
|
||||
import core.memory;
|
||||
auto ptr = GC.malloc(__traits(classInstanceSize, CRTP));
|
||||
ptr[0 .. __traits(classInstanceSize, CRTP)] = typeid(CRTP).initializer[];
|
||||
auto obj = cast(CRTP) ptr;
|
||||
obj.internalJavaHandle_ = o;
|
||||
return obj;
|
||||
}
|
||||
|
||||
/+
|
||||
/++
|
||||
D constructors on Java objects don't work right, so this is disabled to ensure
|
||||
|
@ -1120,8 +1249,20 @@ class JavaClass(string javaPackage, CRTP) : IJavaObject {
|
|||
|
||||
import core.stdc.stdio;
|
||||
auto internalJavaClassHandle_ = (*env).FindClass(env, (_javaParameterString[1 .. $-1] ~ "\0").ptr);
|
||||
|
||||
/+
|
||||
if(!internalJavaClassHandle_) {
|
||||
fprintf(stderr, ("Cannot find Java class for " ~ CRTP.stringof));
|
||||
static if(CRTP.stringof == "ssTest2") {
|
||||
import std.file;
|
||||
auto bytes = cast(byte[]) read("Test2.class");
|
||||
auto loader = ClassLoader.getSystemClassLoader().getJavaHandle();
|
||||
internalJavaClassHandle_ = (*env).DefineClass(env, "wtf/Test2", loader, bytes.ptr, cast(int) bytes.length);
|
||||
}
|
||||
}
|
||||
+/
|
||||
|
||||
if(!internalJavaClassHandle_) {
|
||||
fprintf(stderr, ("Cannot find Java class for " ~ CRTP.stringof ~ "\n"));
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
@ -1146,6 +1287,12 @@ class JavaClass(string javaPackage, CRTP) : IJavaObject {
|
|||
__gshared /* immutable */ int function(JNIEnv* env)[] classInitializers_;
|
||||
|
||||
|
||||
/+
|
||||
final class ClassLoader : JavaClass!("java.lang", ClassLoader) {
|
||||
@Import static ClassLoader getSystemClassLoader();
|
||||
}
|
||||
+/
|
||||
|
||||
|
||||
|
||||
|
||||
|
|
Loading…
Reference in New Issue