Windows OLE/COM with jextract code generation
Maurizio Cimadamore
maurizio.cimadamore at oracle.com
Fri Mar 5 21:44:11 UTC 2021
Hi Duncan,
I've just integrated a fix for this:
https://github.com/openjdk/panama-foreign/pull/456
Now, if you have a global variable, or a struct field whose type is a
function pointer type, an extra accessor will be generated which returns
an instance of the corresponding functional interface.
Let us know if this helps (we had a couple of possible strategies to go
through, described in the PR - in the end we opted for the functional
interface accessor).
Cheers
Maurizio
On 17/02/2021 17:59, Duncan Gittins wrote:
> On 17/02/2021 12:58, Jorn Vernee wrote:
>> Hi Duncan,
>>
>> Thanks for bringing this up. Virtual calls are an interesting
>> use-case, and a while ago we added support for linking virtual calls
>> to CLinker, where you get back a method handle that takes an
>> additional leading Addressable parameter that represents the function
>> that should be called.
>>
>> On the jextract front though, there is a relatively low-hanging fruit
>> that we could also grab for this. IUnknownVtbl is essentially a
>> struct with a bunch of function pointers. A simplified example would be:
>>
>> struct Foo {
>> void (*cb)(int);
>> };
>>
>> Now maybe you get one of these structs back from a native call, and
>> then you want to call 'cb'.
>>
>> Jextract currently generates getters and setters for those kinds of
>> function pointer fields, which would allow you to manually get the
>> function address and then manually link a MethodHandle to call it, as
>> you've found out. But, we could, in the case of function pointer
>> types, also add another kind of accessor to call the function
>> directly, and wire up the right MethodHandle behind the scenes.
>>
>> So, your case could theoretically be turned from:
>>
>> Addressable vtFuncPtr = (Addressable)
>> Ole32_h.IUnknownVtbl.QueryInterface$VH().get(vtable);
>> MethodType mt= MethodType.methodType(int.class,
>> MemoryAddress.class, MemoryAddress.class, MemoryAddress.class);
>> FunctionDescriptor fd = FunctionDescriptor.of(CLinker.C_LONG,
>> CLinker.C_POINTER, CLinker.C_POINTER, CLinker.C_POINTER);
>> MethodHandle queryInterfaceMH =
>> CLinker.getInstance().downcallHandle(vtFuncPtr, mt, fd);
>>
>> Into something like:
>>
>> MemoryAddress hresult = Ole32_h.IUnknownVtbl.QueryInterface(vtable);
>>
> That extra code generation would be helpful. I used the method handle
> from vtable in above code to create version of the interface added by
> jextract:
>
> Ole32_h.IUnknownVtbl.QueryInterface queryInterface =
> (This,riid,ppvObject) -> {
> var mh$ = queryInterfaceMH;
> try {
> return (int)mh$.invokeExact(This, riid, ppvObject);
> } catch (Throwable ex$) {
> throw new AssertionError("should not reach here", ex$);
> }
> };
>
> ... and then call QI was simple:
>
> int hRes = queryInterface.apply(comObj.address(),
> riid.address(), queryRes.address());
>
> Duncan
>
>> Maurizio has filed an issue for this [1].
>>
>> Thanks,
>> Jorn
>>
>> [1] : https://bugs.openjdk.java.net/browse/JDK-8261906
>>
>> On 16/02/2021 19:13, Duncan Gittins wrote:
>>> This may be outside the scope of jextract (too Windows specific), or
>>> a dumb question (then I'll send it to the usual place for dumb
>>> questions: stackoverflow).
>>>
>>> I'm converting my handwritten Panama Java application over to use
>>> the corresponding data structures generated by jextract, but can't
>>> find definitions which could simplify the MethodHandle lookups for
>>> referencing Windows OLE/COM API calls. The jextract output of
>>> Ole32.h (which contains only "#include <objbase.h>") used is
>>> generated from:
>>>
>>> set "WINKIT=c:\Program Files (x86)\Windows
>>> Kits\10\Include\10.0.18362.0"
>>> jextract -source -lole32 -t duncan.win -d java -I
>>> "%WINKIT%\um" Ole32.h
>>>
>>> Retrieving any COM object IUnknown is easy with appropriate values
>>> for class+iid guids as follows:
>>>
>>> MemoryAddress comObj =
>>> Ole32_h.CoCreateInstance(rclsid.address(), pUnkOuter, dwClsContext,
>>> riid.address(), ptrComObj.address());
>>>
>>> ... and read the COM object vTable by following IUnknown.lpVtbl
>>> pointer to IUnknownVTbl:
>>>
>>> MemorySegment memseg =
>>> Ole32_h.IUnknown.ofAddressRestricted(comObj);
>>> MemoryAddress pVTableAddr = Ole32_h.IUnknown.lpVtbl$get(memseg);
>>> MemorySegment vtable =
>>> Ole32_h.IUnknownVtbl.ofAddressRestricted(pVTableAddr);
>>>
>>> At this point I need downcallHandle for every COM method pointer in
>>> vtable - QueryInterface / AddRef... I have this for "QueryInterface"
>>> but is there a way to avoid my hardcoded mt/fd declarations for all
>>> each method handle?
>>>
>>> Addressable vtFuncPtr = (Addressable)
>>> Ole32_h.IUnknownVtbl.QueryInterface$VH().get(vtable);
>>> MethodType mt= MethodType.methodType(int.class,
>>> MemoryAddress.class, MemoryAddress.class, MemoryAddress.class);
>>> FunctionDescriptor fd = FunctionDescriptor.of(CLinker.C_LONG,
>>> CLinker.C_POINTER, CLinker.C_POINTER, CLinker.C_POINTER);
>>> MethodHandle queryInterfaceMH =
>>> CLinker.getInstance().downcallHandle(vtFuncPtr, mt, fd);
>>>
>>> The interface IUnknownVtbl.QueryInterface has support for upcalls (-
>>> so maybe I could implement QueryInterface for a COM object in Java),
>>> and has "fd" internally as xxx_constants.QueryInterface $FUNC, but
>>> no equivalent to "mt" or downcalls of this specific comObj - say as
>>> "IUnknownVtbl.QueryInterface$MH(vtFuncPtr)".
>>>
>>> Kind regards
>>>
>>> Duncan Gittins
>
>
More information about the panama-dev
mailing list