[9] RFR (S): 8157181: Compilers accept modification of final fields outside initializer methods
Zoltán Majó
zoltan.majo at oracle.com
Wed May 18 13:25:35 UTC 2016
Hi Vladimir,
thank you for the detailed feedback. Please see comments below.
On 05/18/2016 02:17 PM, Vladimir Ivanov wrote:
>>> Also, instead of bailing out the whole compilation, you can emit an
>>> access as if the field access isn't linked yet (C1 supports that by
>>> setting field offset to -1). The JVM should throw an appropriate
>>> exception when trying to link the field.
>>
>> I tried that (by setting the variable 'offset' in
>> GraphBuilder::access_field to -1. But C1 will generate an access to an
>> offset of -1; executing that code results in data corruption.
> Have you set needs_patching as well?
No, I did not and that was the reason why the access to offset -1 was
generated.
>
> src/share/vm/c1/c1_GraphBuilder.cpp [1]
>
> void GraphBuilder::access_field(Bytecodes::Code code) {
> ...
> const bool needs_patching = !holder->is_loaded() ||
> !field->will_link(method()->holder(), code) ||
> PatchALot;
> ...
> const int offset = !needs_patching ? field->offset() : -1;
> ...
> case Bytecodes::_putstatic: {
> ...
> append(new StoreField(append(obj), offset, field, val, true,
> state_before, needs_patching));
>
> Otherwise, it's a bug in C1 and patching is broken for unlinked field
> accesses.
I did what you suggested and patching works fine. The putstatic (and the
corresponding StoreField from above) is treated as a unlinked (as expected).
However, linking does not generate the appropriate exception. The reason
is that LinkResolver::resolve_field() checks only if the holder of the
field is the same as the holder of the method, but not if the method is
an initializer. So execution continues to bytecode @29 (that is constant
folded by the compiler) and the original failure still appears.
>
>>>> It seems to me that the interpeter is affected by a related problem
>>>> (i.e., the interpreter does not throw an IllegalAccessError if (2)
>>>> does
>>>> not hold). I'd prefer to file a separate bug for that problem.
>>> I'm confused. How do you observe the difference between -Xint & -Xcomp
>>> mode then?
>>
>> Without the patch applied:
>> * With -Xint, the VM does not throw an exception IllegalAccessError (as
>> required by the VM spec). Instead, the VM executes the putstatic to the
>> static final field as any other putstatic.
>> * -Xcomp generates optimized code that folds a constant and results in a
>> NullPointerException being thrown.
>
> It's a different bug then. Compilers should not constant fold final
> field loads (from the same class) in initializers.
>
>> The behavior with -Xint is the behavior expected and accepted by Jython.
>> The problem was there since at least JDK 6, but was never observed "in
>> the wild". The reason is that the method triggering the problem is not
>> hot enough to be compiled (and so it is only interpreted).
>
> I vaguely remember JVMS allowed changing final fields in the same
> class from any code, but then that part was tightened (in 5?).
>
> I'd prefer to avoid discrepancy in behavior between different
> execution modes. Either disable problematic constant folding in
> compilers for now or fix compilers and interpreter at once.
I assume by "problematic constant folding" you mean "constant fold final
field loads (from the same class) in initializers" from above.
My impression is that the compilers can be safe only by completely
disabling folding of final fields. The reason is that, currently, any
method of a class 'C' declaring a static final field 'f' can change the
field 'f' after class 'C' has been initialized. So even if we are
compiling a method 'm' of a class 'X' that is accessing field 'C::f',
'f' can change after 'm' has been compiled.
So folding final fields must be disabled for all compilations. Or am I
missing something?
I agree with you, alternatively we can propose a more generic fix (fix
the interpreter and the compilers as well). The fix would most likely
affect field resolution in LinkResolver::resolve_field() around here:
http://hg.openjdk.java.net/jdk9/hs-comp/hotspot/file/07a5eceac654/src/share/vm/interpreter/linkResolver.cpp#l910
Changing resolve_field() to throw an IllegalAccessError for
inappropriate field access will propagate the exception to the
interpreter. Changing ciField::will_link() will most likely kill (some)
compilations (if, e.g., fields are linked through ciField::will_link()).
I'd prefer to take the second option (changing field resolution), but
there might be some disadvantages related to option I might be overseeing.
Thank you!
Best regards,
Zoltan
>
>> More details about the failure are here [3].
>>
>> With the patch applied, the program always completes as it is expected
>> by Jython, no matter if we use -Xint, -Xcomp, or -Xmixed (because we
>> always interpret methods non-conforming with the VM spec).
>>
>> Here is the updated webrev:
>> http://cr.openjdk.java.net/~zmajo/8157181/webrev.01/
> Bailing out the whole compilation in C2 is even less appropriate. It
> should generate an uncommon trap for such accesses instead (see
> Parse::do_field_access).
>
> Best regards,
> Vladimir Ivanov
>
> [1]
> http://hg.openjdk.java.net/jdk9/hs-comp/hotspot/file/07a5eceac654/src/share/vm/c1/c1_GraphBuilder.cpp#l1595
More information about the hotspot-compiler-dev
mailing list