RFR: 8221477: Inject os/cpu-specific constants into Unsafe from JVM

David Holmes david.holmes at oracle.com
Mon Apr 1 04:27:04 UTC 2019


Hi Andrew,

On 29/03/2019 8:40 pm, Andrew Dinn wrote:
> Hi David,
> 
> Thanks very much for reviewing this patch.
> 
> On 29/03/2019 01:25, David Holmes wrote:
>> This seems fine in general but I have a few queries on some details:
>>
>> src/hotspot/share/classfile/javaClasses.hpp
>>
>>      f(java_lang_Thread) \
>> +   f(jdk_internal_misc_UnsafeConstants) \
>>      f(java_lang_ThreadGroup) \
>>
>> Is there a reason this needs to be shoved in there? Similarly with
>> src/hotspot/share/classfile/systemDictionary.hpp:
>>
>>     do_klass(Thread_klass,
>> java_lang_Thread                                      ) \
>> +   do_klass(UnsafeConstants_klass,
>> jdk_internal_misc_UnsafeConstants                     ) \
>>      do_klass(ThreadGroup_klass,
>> java_lang_ThreadGroup                                 ) \
>>
>> ?
> 
> I'm not sure what you are asking here. Are you talking about the
> positioning of these entries? If so then the reason for choosing those
> positions was because they match the position of the corresponding class
> initialization calls in the file thread.cpp. As to that choice ... see
> below.

Yes I'm asking about the positioning ...

>> src/hotspot/share/runtime/thread.cpp
>>
>>      main_thread->set_threadObj(thread_object);
>> +
>> +   // initialize the hardware-specific constants needed by Unsafe
>> +   initialize_class(vmSymbols::jdk_internal_misc_UnsafeConstants(),
>> CHECK);
>> +   jdk_internal_misc_UnsafeConstants::set_unsafe_constants();
>> +
>>      // Set thread status to running since main thread has
>>      // been started and running.
>>      java_lang_Thread::set_thread_status(thread_object,
>>
>> That seems a very odd place to insert the new initialization. I can't
>> see any reason you'd need to split the Thread object initialization like
>> that. ??
> 
> Well, yes, indeed :-). I was not sure where this init needed to go so I
> simply put it in as early as I thought was safe. Clearly, it is an
> interloper into intitialise_java_lang_classes() but I was not sure where
> else was more appropriate.
> 
> I don't think it matters too much where this init happens so long as it
> is early enough to precede any clinit dependencies on Unsafe (see
> below). I'm very happy to take advice on this (indeed I was hoping/
> expecting it would be brought up at review) and also on where the
> entries in the headers would best be placed.
> 
>> More generally exactly when do you need to initialize this new class by
>> and how does the initialization order change before/after this fix? (I'm
>> expecting only to see the new class inserted where needed without any
>> other perturbations.)
> 
> As I said, I put the class initialization in at this early point simply
> to guarantee that the constants are available before Unsafe or any class
> which might recursively initialize Unsafe could end up needing them. I
> am sure they could move later and also earlier, although the latter
> would probably not make any sense. The important thing is that they
> don't really have any hard dependencies on other class inits apart,
> perhaps, from 'magic' classes like Object, Class etc which need to exist
> in order to init /any/ other class.

I did some analysis of the class loading and initialization sequence and 
added my suggestions to bug report. In summary loading seems somewhat 
immaterial so I suggest:

- javaClasses.hpp: Add UnsafeConstants at the end of 
BASIC_JAVA_CLASSES_DO_PART2
- systemDictionary.hpp: Add UnsafeConstants immediately before Unsafe

Then for init:
- thread.cpp: initialize UnsafeConstants immediately after j.l.Module

It would be desirable to detect if we happen to execute the <clinit> 
earlier (by accident) so I suggest adding a "not initialized" assertion 
prior to your code calling initialize_class. Actually that might be a 
useful addition to the initialize_class method, as if it fires it means 
we're not initializing in the expected order ... I'll run a little 
adding that ...

Thanks,
David

P.S. This missing space in javaClasses.cpp was reported by Thomas but 
hasn't been fixed yet:

4026     }else {


> I deliberately factored these constants out of Unsafe into a separate
> all static class UnsafeConstants so as to decouple this init from any
> current or future dependencies on Unsafe (and also to make them more
> clearly visible). Since the fields only hold os/hw specific constants
> they can be set any time after VM_Version/CPU-specific init and
> OS-specific init has completed.
> 
>> src/hotspot/share/classfile/vmSymbols.hpp
>>
>> +   template(big_endian_name,                           "BIG_ENDIAN")
>>                             \
>> +   template(use_unaligned_access_name,
>> "UNALIGNED_ACCESS")                        \
>>
>> Nit: There's an extra space before "UNALIGNED...
> 
> Thanks. Thomas Stuefe already spotted that and I have updated the webrev
> in place to fix it.
> 
>> src/java.base/share/classes/jdk/internal/misc/UnsafeConstants.java
>>
>> 31  * package-protected  ...
>>
>> s/protected/private/
> 
> Thanks, will correct it.
> 
>>   37  * The JVM injects values into all the static fields of this class
>>   ...
>>   43  * hardware-specific constants installed by the JVM.
>>
>> I get this gist of this note but still found it rather difficult to
>> intepret. There are I think two key points:
>>
>> 1. The fields must not be constant variables, hence the need for the
>> static initialization block.
>> 2. The true value of each field is injected by the VM.
>>
>> How about:
>>
>> * The JVM injects hardware-specific values into all the static fields
>> * of this class during JVM initialization. The static initialization
>> * block exists to prevent the fields from being considered constant
>> * variables, so the field values will be not be compiled directly into
>> * any class that uses them.
> 
> Yes, I agree that is clearer.
> 
>> 60      * @implNote
>> 61      * The actual value for this field is injected by JVM. A static
>> 62      * initialization block is used to set the value here to
>> 63      * communicate that this static final field is not statically
>> 64      * foldable, and to avoid any possible circular dependency during
>> 65      * vm initialization.
>>
>> I think all you need for each field is just:
>>
>> * @implNote
>> * The actual value for this field is injected by _the_ JVM.
> 
> Yes, also better.
> 
>> 85      * flag whose value ...
>> 92      * flag whose value ...
>>
>> s/flag/Flag/
> 
> Thanks, will correct this.
> 
> regards,
> 
> 
> Andrew Dinn
> -----------
> Senior Principal Software Engineer
> Red Hat UK Ltd
> Registered in England and Wales under Company Registration No. 03798903
> Directors: Michael Cunningham, Michael ("Mike") O'Neill, Eric Shander
> 


More information about the core-libs-dev mailing list