Moving from VVT to the L-world value types (LWVT)

Frederic Parain frederic.parain at oracle.com
Wed Jan 31 19:08:24 UTC 2018


Here’s an update of the JVMS draft which includes the semantic
proposed below.

http://cr.openjdk.java.net/~fparain/L-world/L-World-JVMS-3.pdf

Feedback and comments are welcome.

Fred


> On Jan 31, 2018, at 05:54, Remi Forax <forax at univ-mlv.fr> wrote:
> 
> 
> 
> ----- Mail original -----
>> De: "Frederic Parain" <frederic.parain at oracle.com>
>> À: "John Rose" <john.r.rose at oracle.com>
>> Cc: "valhalla-dev" <valhalla-dev at openjdk.java.net>
>> Envoyé: Lundi 29 Janvier 2018 20:31:56
>> Objet: Re: Moving from VVT to the L-world value types (LWVT)
> 
>> John,
>> 
>> Silently transforming null into the default value in case 2-B was
>> just in case it could ease the handling of migrated case. If throwing
>> NPE is OK here, it makes the semantic much cleaner. It also solves
>> some problems we had while trying to add support for the null
>> value in the value class’ equals() method.
>> 
>> During a brainstorming session with Karen this morning, we realized
>> that case 2-A (ACC_FLAT flag set for an object class) should be in
>> fact treated as an error, instead of just ignoring the flag.
> 
> Which means that changing a value type to a reference type is not a backward compatible change,
> i'm Ok with that.
> 
>> 
>> These two modifications makes the semantic mucho simpler and
>> cleaner with only two cases: one where null is always valid, and
>> one where null is never valid.
>> 
>> So here’s a re-write of the semantic:
>> 
>> Fields have a new access flag called ACC_NON_NULLABLE.
>> 
>> The type of a field with the ACC_NON_NULLABLE flag set must
>> be a value class type, otherwise an ICCE is thrown.
>> 
>> 1 - If a field is declared without the ACC_NON_NULLABLE flag set:
>>    - this field is initialized with the null reference
>>    - it is valid to write null to this field
>>    - note: JVMs are unlikely to flatten such field
>> 
>> 2 - If a field is declared with the ACC_NON_NULLABLE flag set:
>>    - this field is initialized with the default value of this field’s value class
>>    - writing null to this field causes a NPE
>>    - JVMs are encouraged to flatten such field
>> 
>> This semantic makes much more sense because being non-nullable
>> is a property of the container more than a property of the field’s type.
>> 
>> Fred
> 
> I agree about the semantics,
> the name of the flag is wrong, it should be ACC_FLAT_THUS_NON_NULLABLE :)
> 
> Rémi
> 
>> 
>>> On Jan 25, 2018, at 17:48, John Rose <john.r.rose at oracle.com> wrote:
>>> 
>>> On Jan 25, 2018, at 1:51 PM, Frederic Parain <frederic.parain at oracle.com> wrote:
>>>> 
>>>> Simply allowing non-flattened field to be null is not a viable
>>>> solution because a consequence would be  that the result of
>>>> reading  a non-initialized field would depend on wether or not
>>>> the field has been flattened.
>>> 
>>> As you note below, it is possible to make the details
>>> of null processing depend on whether the *field* was
>>> declared flat or not.
>>> 
>>>> With Valhalla Value Types, we were able to use the null
>>>> value internally for non-initialized non-flattened field because
>>>> the JVM always knew if a field were a value type or not.
>>> 
>>> Nice trick.
>>> 
>>>> Unfortunately, this cannot be done in the L-world because
>>>> when a non-initialized field is read, there’s no guarantee that
>>>> the class of this field has been loaded yet.
>>>> 
>>>> So, here’s a proposal that provides a semantic for null value
>>>> fields that do not depend on the implementation, and without
>>>> additional runtime checks for non-value fields:
>>>> 
>>>> 1 - If a field is declared without the ACC_FLAT flag set:
>>>>  -> it is never flattened (even if the class is already loaded
>>>>     and it is a value class)
>>>>  -> it is OK to write null on this field
>>>>  -> reading the default value of this field returns null
>>> 
>>> Yes, that's what I had in mind too.
>>> 
>>>> 2 - If a field is declared wit the ACC_FLAT_flag set:
>>>>  -> the class of this field is loaded before computing the layout
>>>>     of the declaring class
>>>>  A - If the field's class is an object class
>>>>      -> same semantic as in 1 is applied
>>> 
>>> Yes.
>>> 
>>>>  B - If the field's class is a value class
>>>>      -> writing null to this field results in setting this field
>>>>         to the default value of its value class
>>> 
>>> We have a choice here:  We can also throw NPE on
>>> such writes.  I think we should.  There's no legitimate
>>> need for writing a null value to a flat field, I think.
>>> 
>>> (But maybe there's a translation strategy, that I'm not
>>> noticing, that could use this behavior?)
>>> 
>>>>      -> reading the default value of this field (uninitialized)
>>>>         returns the default value of its value class
>>> 
>>> If you can't write null w/o NPE, then the JVM can
>>> still use the internal trick of allowing null as a
>>> sentinel for the default value.
>>> 
>>>> Case 1 is the object world as we know it today, no changes, no
>>>> additional runtime checks, values will be handled like objects.
>>>> Case 2-A is the case where the ACC_FLAT has been mis-used
>>>> on an object class.
>>>> Case 2-B is the interesting value type case, and its semantic
>>>> relies only on the presence of the ACC_FLAT flag, not on the
>>>> decision of the JVM to flatten the field or not.
>>> 
>>> Exactly.
>>> 
>>>> Checks to intercept
>>>> the null value are easy to implement, all information they need
>>>> are in the field descriptor and on the execution stack.
>>> 
>>> I agree.  Resolving the CONSTANT_Fieldref to the descriptor
>>> will determine whether the *field* was declared ACC_FLAT.
>>> In that case null writes will fail and null reads (if any) can be
>>> handled in some manner convenient to the JVM.
>>> 
>>>> Bytecode quickening or JIT compilation will help avoiding the ACC_FLAT
>>>> test completely for cases 1 and 2-A.
>>> 
>>> Yes.
>>> 
>>> — John




More information about the valhalla-dev mailing list