RFR: 8242038: G1: Lazily initialize RSHashTables

Thomas Schatzl thomas.schatzl at oracle.com
Wed Apr 8 11:26:10 UTC 2020


Hi,

On 08.04.20 06:06, Kim Barrett wrote:
>> On Apr 7, 2020, at 10:52 AM, Claes Redestad <claes.redestad at oracle.com> wrote:
>>
>>
>>
>> On 2020-04-07 10:03, Kim Barrett wrote:
>>>> On Apr 7, 2020, at 3:57 AM, Kim Barrett <kim.barrett at oracle.com> wrote:
>>>> Morover, the RSHashTable constructor calls RSHashTable::clear(). That
>>>> calls SparsePRTEntry::size(), whose value is computed from
>>>> G1RSetSparseRegionEntries, which is a product flag which doesn't have
>>>> its final value at static initialization time. Oops!
>>> Oh, and SparsePRTEntry::size() is also called to size the array.
>>> Probably the thing that makes this not just crash quickly is that the special
>>> empty_table doesn’t seem to get used as anything other than a marker / sentinal.
>>
>> Good catches, Kim,
>>
>> I've reworked the initialization of the empty RSHashTable
>> to avoid calling SparsePRTEntry::size and doing C-heap allocation with a
>> dedicated (private) constructor:

Thanks!

>>
>> http://cr.openjdk.java.net/~redestad/8242038/open.01/
> 
> ------------------------------------------------------------------------------
>   177   RSHashTable();
>   178 public:
> 
> Mostly in HotSpot I think we have a blank line between code and a
> following access-specifier.
> 
> ------------------------------------------------------------------------------
> src/hotspot/share/gc/g1/sparsePRT.cpp
>    90 static int empty_buckets[1];
> ...
>   103   _buckets[0] = NullEntry;
> 
> How about
> 
>    90 static int empty_buckets[] = { RSHashTable::NullEntry };
> 
> and remove line 103.
> 
> ------------------------------------------------------------------------------
> src/hotspot/share/gc/g1/sparsePRT.cpp
>   122     assert(_entries != NULL, "INV");
> 
> I'd prefer "invariant" be spelled out.  I see there are a couple of
> other nearby uses of the same abbreviation, but elsewhere in the file
> it is spelled out.
> 
> ------------------------------------------------------------------------------
> src/hotspot/share/gc/g1/sparsePRT.cpp
>   262 SparsePRT::~SparsePRT() {
>   263   delete _table;
>   264 }
> 
> This needs to be updated to not delete when _table == &empty_table.
> 
> ------------------------------------------------------------------------------
> src/hotspot/share/gc/g1/sparsePRT.cpp
>   300   if (last->capacity() < InitialCapacity) {
> 
> I think clearer here would be to check if last == &empty_table.
> Otherwise it's not obvious why we don't need to delete last.
Agree with all above suggestions. Although in this case I'd tend to 
invert the condition (i.e. != &empty_table), but there are no particular 
measurements behind that.

> 
> ------------------------------------------------------------------------------
> src/hotspot/share/gc/g1/sparsePRT.hpp
>    int _tbl_ind;         // [-1, 0.._rsht->_capacity)
> 
> [pre-existing]
> 
> I see no sign of -1 (or any other negative value) as a possibility for
> this member. It is initialized to 0 in the constructor and is only
> ever incremented. As such, I think it's type could be unsigned, and
> the cast when comparing with _rsht->capacity() in RSHTBIter::has_next
> would not be needed.
> 
> ------------------------------------------------------------------------------

The iteration completely changed in the changes how the card scanning is 
done in jdk14, this is a leftover.

Thanks,
   Thomas



More information about the hotspot-gc-dev mailing list