VaList is buggy on Java 17?

Maurizio Cimadamore maurizio.cimadamore at oracle.com
Thu Apr 20 09:19:16 UTC 2023


Hi Mark,
the implementation of VaList has been ironed out a lot around 19 (when 
the API became preview) - that said, I don't recall a specific fix for 
the issue you mention.

Also, as mentioned here [1, 2], VaList is no longer part of the API 
since 21, as the interaction with a VaList is hopelessly platform 
dependent (since different platforms use different encoding for 
va_list), meaning that it is very problematic, from a javadoc 
perspective, to specify which exception will be issued and when.

Regarding int vs. long, the C language defines a set of default argument 
promotion that should be applied when using variadic functions and 
va_lists. The builder methods and getters of the VaList class were 
designed with these constraints in mind, which means the calls to the 
valist builder have to "match" the calls to the getters (or undefined 
behavior will ensue). From the Scala code you paste below, it is not 
clear to me is such a mismatch occurs (I do no see `long` anywhere, but 
I do see `get[CInt]`).

Maurizio

[1] - 
https://mail.openjdk.org/pipermail/panama-dev/2022-November/018096.html
[2] - https://git.openjdk.org/panama-foreign/pull/763


On 20/04/2023 08:00, Mark Hammons wrote:
> Hi all,
>
> I know that the panama project is pushing forward on java 20 and 21 
> right now, and java 17 is probably not a great concern for you all at 
> the moment, but I was testing VaList support in my library, Slinc, and 
> it seems to be flaky on Java 17.0.5. The test in question goes like so:
>
>   property("varargs can be embedded in structs"):
>       forAll: (ints: Seq[CInt]) =>
>         val numTake = 8
>         Scope.confined:
>           val va = VarArgsBuilder.fromIterable
>             (
>               ints.map(a => a: Variadic).take(numTake)
>             )
>             .build
>
>           val x = va.copy()
>
>           val p = Ptr.copy(E(x))
>
>           ints.take(numTake).foreach: value =>
>             assertEquals(va.get[CInt], value, "conversion test")
>
>           ints.take(numTake).foreach: value =>
>             assertEquals((!p).list.get[CInt], value)
> Originally in this test I was testing if I could write a VaList to a 
> struct and get it back out and pull the values out of it 
> appropriately. What I noticed is that if the number of integer values 
> in a row passed into VaList is 8 or greater, my test fails. 7 or less 
> ints and the test passes. On the Java 19 implementation, this does not 
> happen, nor does this happen when I pass a great deal more parameters 
> (but more varied in types, some longs, some pointers, some structs). 
> This test passes if I randomly generate Long values and pass those 
> into the VaList.
>
> This is not a critical bug by any means. I can sidestep the issue by 
> encoding all lower integral types into Long in the java 17 runtime for 
> my library. I just wanted to bring this to your attention because it's 
> a weird issue that java 17 users of the foreign api may hit.
>
> Thanks for your time,
> Mark Hammons


More information about the panama-dev mailing list