[External] : Re: Feedback / query on jextract for Windows 10

Duncan Gittins duncan.gittins at gmail.com
Wed Jan 27 14:36:09 UTC 2021


Thanks, that's very helpful. Use of repeated --filter option solves my 
problem with huge jars - now down to 800k and cuts the first-time 
startup delay vs handcoded to around 50-150ms depending on the calls made.

It might be helpful if there was a command line option to set the output 
class name so the top level classname isn't derived from the header file 
name. I avoid static imports, and would prefer to call say 
"Ole32.CoInitializeEx()" than "shlobj_core.CoInitializeEx()" for code 
derived from shlobj_core.h.

This would also help for projects that had one master header file which 
set up #define for other includes. eg. Consider winuser.h:

    set "WINKIT=c:\Program Files (x86)\Windows Kits\10\Include\10.0.18362.0"
    jextract -luser32 -t duncan.win.user32 -d winuser.classes --filter 
winuser.h "%WINKIT%/um/winuser.h"
    => Fails error: "No Target Architecture"

    jextract -luser32 -t duncan.win.user32 -d winuser.classes --filter 
winuser.h "%WINKIT%/um/shlobj_core.h"
    => Works fine, but the top level class name is shlobj_core_h which 
would same class name as for this jextract:

    jextract -lshell32 -t duncan.win.shell32 -d shlobj_core.classes 
--filter shlobj_core.h "%WINKIT%/um/shlobj_core.h"

However there is a very simple workaround: I can create a User32.h file 
which contains "#include <shlobj_core.h>"

     jextract -luser32 -t duncan.win.user32 -d winuser.classes --filter 
winuser.h  -I "%WINKIT%/um" User32.h

Duncan

On 26/01/2021 17:25, Maurizio Cimadamore wrote:
> Another fly by comment - have you tried using the --filter option so 
> that jextract will just extract the header you want?
>
> e.g. you might try adding `--filter shlobj_core` and see what happens 
> (but it might be too narrow of a filter).
>
> Maurizio
>
>
> On 26/01/2021 15:43, Duncan Gittins wrote:
>> I don't know what causes the speed difference, which may be down to 
>> the jar size as you suggest. I'll re-test after the time that 
>> "--source" generates same code as seen in the .class files when it 
>> may be easier to understand why there is a difference. I am using 
>> 16-panama+3-385.
>>
>> This isn't a very robust test, but I also tried deleted all the 
>> un-necessary .class files from the 11MB jextract generated classes 
>> (not produced using --source) down to 2MB jar, then I moved my 
>> handcoded method into a separate class and added a dependency on a 
>> different 2MB jar. Neither of those changes affected the typical 
>> speeds of first call times - I still consistently see that the hand 
>> coded method handle version of my app is ~300-500ms faster on first 
>> call, and no difference for 2nd+ calls.
>>
>> Duncan
>>
>> On 25/01/2021 16:18, Maurizio Cimadamore wrote:
>>>
>>> On 25/01/2021 16:13, Duncan Gittins wrote:
>>>> Thanks for the information.  As you suspected that bug must be 
>>>> related and there is a minor difference in the --source output 
>>>> because the problem does go away when removing the jextract 
>>>> "--source" flag and making jar directly from generated classes 
>>>> rather than just compiling from the generated source code.
>>>>
>>>> My code samples using jextract bindings are ~300ms slower at 
>>>> startup time (but same speed runtime). I'm not certain if this 
>>>> overhead is down to initialisation of static constants or simply to 
>>>> be expected when adding 11MB jar footprint (compared to 56k jar for 
>>>> my handwritten MH bindings). I will follow-up if I find out more.
>>> If you see the startup slowdown even when omitting --source, then 
>>> the issue is with loading the jar, rather than initializing - as 
>>> explaining no initializing should happen outside for stuff that is 
>>> actually used by your code - so I'd expect that initialization 
>>> overhead for jextract code and your manually written versions should 
>>> be roughly the same.
>>>>
>>>> Looking at the generated code for "xxx$constants$nn.java" there 
>>>> seems to be a degree of unnecessary initialisation per 
>>>> xxx$constants$nn class as it contains static variables for 
>>>> FunctionDescriptor / MemoryLayout which would be initialised every 
>>>> time that particular class was loaded and not used in my code. I 
>>>> felt that my own jar was slow starting so switched foreign linker 
>>>> references to a lazy load Supplier for every library / symbol / mh 
>>>> / vh so that only the referenced types get created.
>>>
>>> Sure - but that's only when --source is used - when you generated 
>>> classes directly, everything is lazy (and in a more general way than 
>>> using Suppliers).
>>>
>>> So, can you please confirm, to be clear, that you still see startup 
>>> regression even when NOT using "--source" ?
>>>
>>> Thanks!
>>> Maurizio
>>>
>>>>
>>>> I will have a look at future builds - this project is extremely 
>>>> useful so I hope it gets into a full JDK release soon. 
>>>> Unfortunately my issue with a huge jar in Eclipse means I can't 
>>>> easily develop using the generated code variant right now, but in 
>>>> the meantime the jextract code is still very helpful to work out 
>>>> how to call Windows APIs correctly.
>>>>
>>>> Duncan
>>>>
>>>> On 25/01/2021 10:20, Maurizio Cimadamore wrote:
>>>>>
>>>>> On 23/01/2021 18:34, Duncan Gittins wrote:
>>>>>> I've been using Panama on Windows 10 to eliminate my C# apps 
>>>>>> which I would normally call via ProcessBuilder. I'm very 
>>>>>> impressed so far and looking forward to newer versions: 
>>>>>> everything I have written in Windows C# is now replaced with 
>>>>>> foreign memory / linker calls to native code using the latest 
>>>>>> Java 16 EA.  I have examples for various Win32 API and COM 
>>>>>> objects, HWND objects and screensaver. However I'm struggling to 
>>>>>> get the code running with output from jextract build. Perhaps I 
>>>>>> overlooked something very obvious in how to setup?
>>>>>>
>>>>>> One example app is attached - this calls SystemParametersInfoA to 
>>>>>> set the Windows desktop background. There are two implementations 
>>>>>> of setImage: one with hardwired MethodHandle works fine, but the 
>>>>>> jextract version gives UnsupportedOperationException for layout 
>>>>>> unrelated to SystemParametersInfoA. If I edit the generated code 
>>>>>> to comment out all lines unrelated to SystemParametersInfoA then 
>>>>>> both versions of setImage work.
>>>>>
>>>>> I think the culprit is this:
>>>>>
>>>>> ```
>>>>> Caused by: java.lang.UnsupportedOperationException: Invalid 
>>>>> alignment requirements for layout 
>>>>> b64(ArbitraryUserPointer)[abi/kind=POINTER,layout/name=ArbitraryUserPointer]
>>>>>         at 
>>>>> jdk.incubator.foreign/jdk.internal.foreign.LayoutPath.checkAlignment(LayoutPath.java:273)
>>>>>         at 
>>>>> jdk.incubator.foreign/jdk.internal.foreign.LayoutPath.dereferenceHandle(LayoutPath.java:159)
>>>>>         at 
>>>>> jdk.incubator.foreign/jdk.incubator.foreign.MemoryLayout.lambda$varHandle$2(MemoryLayout.java:488)
>>>>>         at 
>>>>> jdk.incubator.foreign/jdk.incubator.foreign.MemoryLayout.computePathOp(MemoryLayout.java:534)
>>>>>         at 
>>>>> jdk.incubator.foreign/jdk.incubator.foreign.MemoryLayout.varHandle(MemoryLayout.java:488)
>>>>>         at 
>>>>> duncan.win32.shobjidl_core.shlobj_core_h$constants$16.<clinit>(shlobj_core_h$constants$16.java:1844)
>>>>> ```
>>>>>
>>>>> This seems a duplicate of another issue that has been reported 
>>>>> last week:
>>>>>
>>>>> https://bugs.openjdk.java.net/browse/JDK-8259832
>>>>>
>>>>> E.g. jextract ignoring alignment requirements. Note that jextract 
>>>>> generates lots of constants, and it takes only one bad one to make 
>>>>> initialization fail.
>>>>>
>>>>>>
>>>>>> The jextract command on Windows 10 (with fresh Vis Studio 
>>>>>> installation) expands shlobj_core.h because I my app also uses 
>>>>>> IShellLink.
>>>>>>
>>>>>>      set "WINKIT=c:\Program Files (x86)\Windows 
>>>>>> Kits\10\Include\10.0.18362.0"
>>>>>>      jextract --source -t duncan.win32.shobjidl_core -d 
>>>>>> shlobj_core.h/java     "%WINKIT%/um/shlobj_core.h"
>>>>>>
>>>>>> Another issue is that this single windows header file gives rise 
>>>>>> to 11MB jar and is unusable inside Eclipse (on my slow PC), and 
>>>>>> is slow startup at runtime. Have you considered changing the 
>>>>>> generated code to use lazy Supplier<MethodHandle> instead of 
>>>>>> MethodHandle to ensure only required handles are created on-demand?
>>>>>
>>>>> Uhm... we are already using dynamic constants, so you should only 
>>>>> pay for the MHs and VHs you use - e.g. in fact I'm even surprised 
>>>>> you get the failure inside the <clinit> above - since that 
>>>>> constant should not be initialized unless it's used. I wonder if 
>>>>> there's a regression here... are you using the `--source` flag? 
>>>>> With that flag we revert back to a monolithic header, so that will 
>>>>> explain both failures. If so, try removing that flag - and see 
>>>>> what happens (if everything is working as it should I don't think 
>>>>> you should even see the error above, assuming you never use that 
>>>>> layout).
>>>>>
>>>>> Maurizio
>>>>>
>>>>>>
>>>>>> Kind regards
>>>>>>
>>>>>> Duncan
>>>>>>
>>>>>>
>>>>
>>



More information about the panama-dev mailing list