2016-01-26 15:28:31 +01:00
/*
2018-02-26 09:34:12 +01:00
* Copyright ( c ) 2016 , 2018 , Oracle and / or its affiliates . All rights reserved .
2016-01-26 15:28:31 +01:00
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER .
*
* This code is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License version 2 only , as
* published by the Free Software Foundation .
*
* This code is distributed in the hope that it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License
* version 2 for more details ( a copy is included in the LICENSE file that
* accompanied this code ) .
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work ; if not , write to the Free Software Foundation ,
* Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA .
*
* Please contact Oracle , 500 Oracle Parkway , Redwood Shores , CA 94065 USA
* or visit www . oracle . com if you need additional information or have any
* questions .
*
*/
# include "precompiled.hpp"
2016-12-07 13:51:20 +01:00
# include "gc/g1/g1Allocator.inline.hpp"
2016-01-26 15:28:31 +01:00
# include "gc/g1/g1CollectedHeap.hpp"
# include "gc/g1/g1CollectedHeap.inline.hpp"
2018-04-03 12:05:49 +02:00
# include "gc/g1/g1ConcurrentMarkThread.hpp"
2016-01-26 15:28:31 +01:00
# include "gc/g1/g1HeapVerifier.hpp"
2016-03-18 15:20:43 +01:00
# include "gc/g1/g1Policy.hpp"
2016-01-26 15:28:31 +01:00
# include "gc/g1/g1RemSet.hpp"
# include "gc/g1/g1RootProcessor.hpp"
# include "gc/g1/heapRegion.hpp"
# include "gc/g1/heapRegion.inline.hpp"
# include "gc/g1/heapRegionRemSet.hpp"
# include "gc/g1/g1StringDedup.hpp"
2017-07-21 09:50:12 +02:00
# include "logging/log.hpp"
# include "logging/logStream.hpp"
2018-05-26 06:59:49 +02:00
# include "memory/iterator.inline.hpp"
2016-01-26 15:28:31 +01:00
# include "memory/resourceArea.hpp"
2018-03-15 21:24:10 +01:00
# include "oops/access.inline.hpp"
# include "oops/compressedOops.inline.hpp"
2016-01-26 15:28:31 +01:00
# include "oops/oop.inline.hpp"
2018-03-08 19:10:25 +01:00
# include "runtime/handles.inline.hpp"
2016-01-26 15:28:31 +01:00
2018-03-26 12:44:39 +02:00
int G1HeapVerifier : : _enabled_verification_types = G1HeapVerifier : : G1VerifyAll ;
2016-01-26 15:28:31 +01:00
class VerifyRootsClosure : public OopClosure {
private :
G1CollectedHeap * _g1h ;
VerifyOption _vo ;
bool _failures ;
public :
// _vo == UsePrevMarking -> use "prev" marking information,
// _vo == UseNextMarking -> use "next" marking information,
2017-11-14 11:33:23 +01:00
// _vo == UseFullMarking -> use "next" marking bitmap but no TAMS
2016-01-26 15:28:31 +01:00
VerifyRootsClosure ( VerifyOption vo ) :
_g1h ( G1CollectedHeap : : heap ( ) ) ,
_vo ( vo ) ,
_failures ( false ) { }
bool failures ( ) { return _failures ; }
2018-05-26 06:59:49 +02:00
template < class T > void do_oop_work ( T * p ) {
2018-03-15 21:24:10 +01:00
T heap_oop = RawAccess < > : : oop_load ( p ) ;
if ( ! CompressedOops : : is_null ( heap_oop ) ) {
oop obj = CompressedOops : : decode_not_null ( heap_oop ) ;
2016-01-26 15:28:31 +01:00
if ( _g1h - > is_obj_dead_cond ( obj , _vo ) ) {
2016-04-04 09:15:15 +02:00
Log ( gc , verify ) log ;
2017-08-29 05:33:15 -04:00
log . error ( " Root location " PTR_FORMAT " points to dead obj " PTR_FORMAT , p2i ( p ) , p2i ( obj ) ) ;
2016-01-26 15:28:31 +01:00
ResourceMark rm ;
2017-07-21 09:50:12 +02:00
LogStream ls ( log . error ( ) ) ;
obj - > print_on ( & ls ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
}
}
}
2018-05-26 06:59:49 +02:00
void do_oop ( oop * p ) { do_oop_work ( p ) ; }
void do_oop ( narrowOop * p ) { do_oop_work ( p ) ; }
2016-01-26 15:28:31 +01:00
} ;
class G1VerifyCodeRootOopClosure : public OopClosure {
G1CollectedHeap * _g1h ;
OopClosure * _root_cl ;
nmethod * _nm ;
VerifyOption _vo ;
bool _failures ;
template < class T > void do_oop_work ( T * p ) {
// First verify that this root is live
_root_cl - > do_oop ( p ) ;
if ( ! G1VerifyHeapRegionCodeRoots ) {
// We're not verifying the code roots attached to heap region.
return ;
}
// Don't check the code roots during marking verification in a full GC
2017-11-14 11:33:23 +01:00
if ( _vo = = VerifyOption_G1UseFullMarking ) {
2016-01-26 15:28:31 +01:00
return ;
}
// Now verify that the current nmethod (which contains p) is
// in the code root list of the heap region containing the
// object referenced by p.
2018-03-15 21:24:10 +01:00
T heap_oop = RawAccess < > : : oop_load ( p ) ;
if ( ! CompressedOops : : is_null ( heap_oop ) ) {
oop obj = CompressedOops : : decode_not_null ( heap_oop ) ;
2016-01-26 15:28:31 +01:00
// Now fetch the region containing the object
HeapRegion * hr = _g1h - > heap_region_containing ( obj ) ;
HeapRegionRemSet * hrrs = hr - > rem_set ( ) ;
// Verify that the strong code root list for this region
// contains the nmethod
if ( ! hrrs - > strong_code_roots_list_contains ( _nm ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " Code root location " PTR_FORMAT " "
" from nmethod " PTR_FORMAT " not in strong "
" code roots for region [ " PTR_FORMAT " , " PTR_FORMAT " ) " ,
p2i ( p ) , p2i ( _nm ) , p2i ( hr - > bottom ( ) ) , p2i ( hr - > end ( ) ) ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
}
}
}
public :
G1VerifyCodeRootOopClosure ( G1CollectedHeap * g1h , OopClosure * root_cl , VerifyOption vo ) :
2018-08-08 15:31:06 +02:00
_g1h ( g1h ) , _root_cl ( root_cl ) , _nm ( NULL ) , _vo ( vo ) , _failures ( false ) { }
2016-01-26 15:28:31 +01:00
void do_oop ( oop * p ) { do_oop_work ( p ) ; }
void do_oop ( narrowOop * p ) { do_oop_work ( p ) ; }
void set_nmethod ( nmethod * nm ) { _nm = nm ; }
bool failures ( ) { return _failures ; }
} ;
class G1VerifyCodeRootBlobClosure : public CodeBlobClosure {
G1VerifyCodeRootOopClosure * _oop_cl ;
public :
G1VerifyCodeRootBlobClosure ( G1VerifyCodeRootOopClosure * oop_cl ) :
_oop_cl ( oop_cl ) { }
void do_code_blob ( CodeBlob * cb ) {
nmethod * nm = cb - > as_nmethod_or_null ( ) ;
if ( nm ! = NULL ) {
_oop_cl - > set_nmethod ( nm ) ;
nm - > oops_do ( _oop_cl ) ;
}
}
} ;
class YoungRefCounterClosure : public OopClosure {
G1CollectedHeap * _g1h ;
int _count ;
public :
YoungRefCounterClosure ( G1CollectedHeap * g1h ) : _g1h ( g1h ) , _count ( 0 ) { }
void do_oop ( oop * p ) { if ( _g1h - > is_in_young ( * p ) ) { _count + + ; } }
void do_oop ( narrowOop * p ) { ShouldNotReachHere ( ) ; }
int count ( ) { return _count ; }
void reset_count ( ) { _count = 0 ; } ;
} ;
2017-10-03 16:42:04 -04:00
class VerifyCLDClosure : public CLDClosure {
2016-01-26 15:28:31 +01:00
YoungRefCounterClosure _young_ref_counter_closure ;
OopClosure * _oop_closure ;
public :
2017-10-03 16:42:04 -04:00
VerifyCLDClosure ( G1CollectedHeap * g1h , OopClosure * cl ) : _young_ref_counter_closure ( g1h ) , _oop_closure ( cl ) { }
void do_cld ( ClassLoaderData * cld ) {
cld - > oops_do ( _oop_closure , false ) ;
2016-01-26 15:28:31 +01:00
_young_ref_counter_closure . reset_count ( ) ;
2017-10-03 16:42:04 -04:00
cld - > oops_do ( & _young_ref_counter_closure , false ) ;
2016-01-26 15:28:31 +01:00
if ( _young_ref_counter_closure . count ( ) > 0 ) {
2017-10-03 16:42:04 -04:00
guarantee ( cld - > has_modified_oops ( ) , " CLD " PTR_FORMAT " , has young %d refs but is not dirty. " , p2i ( cld ) , _young_ref_counter_closure . count ( ) ) ;
2016-01-26 15:28:31 +01:00
}
}
} ;
2018-06-26 13:54:19 +02:00
class VerifyLivenessOopClosure : public BasicOopIterateClosure {
2016-01-26 15:28:31 +01:00
G1CollectedHeap * _g1h ;
VerifyOption _vo ;
public :
VerifyLivenessOopClosure ( G1CollectedHeap * g1h , VerifyOption vo ) :
_g1h ( g1h ) , _vo ( vo )
{ }
void do_oop ( narrowOop * p ) { do_oop_work ( p ) ; }
void do_oop ( oop * p ) { do_oop_work ( p ) ; }
template < class T > void do_oop_work ( T * p ) {
2018-03-15 21:24:10 +01:00
oop obj = RawAccess < > : : oop_load ( p ) ;
2016-01-26 15:28:31 +01:00
guarantee ( obj = = NULL | | ! _g1h - > is_obj_dead_cond ( obj , _vo ) ,
" Dead object referenced by a not dead object " ) ;
}
} ;
class VerifyObjsInRegionClosure : public ObjectClosure {
private :
G1CollectedHeap * _g1h ;
size_t _live_bytes ;
HeapRegion * _hr ;
VerifyOption _vo ;
public :
// _vo == UsePrevMarking -> use "prev" marking information,
// _vo == UseNextMarking -> use "next" marking information,
2017-11-14 11:33:23 +01:00
// _vo == UseFullMarking -> use "next" marking bitmap but no TAMS.
2016-01-26 15:28:31 +01:00
VerifyObjsInRegionClosure ( HeapRegion * hr , VerifyOption vo )
: _live_bytes ( 0 ) , _hr ( hr ) , _vo ( vo ) {
_g1h = G1CollectedHeap : : heap ( ) ;
}
void do_object ( oop o ) {
VerifyLivenessOopClosure isLive ( _g1h , _vo ) ;
assert ( o ! = NULL , " Huh? " ) ;
if ( ! _g1h - > is_obj_dead_cond ( o , _vo ) ) {
2017-11-14 11:33:23 +01:00
// If the object is alive according to the full gc mark,
2016-01-26 15:28:31 +01:00
// then verify that the marking information agrees.
// Note we can't verify the contra-positive of the
// above: if the object is dead (according to the mark
// word), it may not be marked, or may have been marked
// but has since became dead, or may have been allocated
// since the last marking.
2017-11-14 11:33:23 +01:00
if ( _vo = = VerifyOption_G1UseFullMarking ) {
guarantee ( ! _g1h - > is_obj_dead ( o ) , " Full GC marking and concurrent mark mismatch " ) ;
2016-01-26 15:28:31 +01:00
}
2018-06-26 13:54:19 +02:00
o - > oop_iterate ( & isLive ) ;
2016-01-26 15:28:31 +01:00
if ( ! _hr - > obj_allocated_since_prev_marking ( o ) ) {
size_t obj_size = o - > size ( ) ; // Make sure we don't overflow
_live_bytes + = ( obj_size * HeapWordSize ) ;
}
}
}
size_t live_bytes ( ) { return _live_bytes ; }
} ;
2018-06-26 13:54:19 +02:00
class VerifyArchiveOopClosure : public BasicOopIterateClosure {
2017-08-14 14:32:17 -04:00
HeapRegion * _hr ;
2016-01-26 15:28:31 +01:00
public :
2017-08-14 14:32:17 -04:00
VerifyArchiveOopClosure ( HeapRegion * hr )
: _hr ( hr ) { }
2016-01-26 15:28:31 +01:00
void do_oop ( narrowOop * p ) { do_oop_work ( p ) ; }
void do_oop ( oop * p ) { do_oop_work ( p ) ; }
template < class T > void do_oop_work ( T * p ) {
2018-03-15 21:24:10 +01:00
oop obj = RawAccess < > : : oop_load ( p ) ;
2017-08-14 14:32:17 -04:00
if ( _hr - > is_open_archive ( ) ) {
guarantee ( obj = = NULL | | G1ArchiveAllocator : : is_archive_object ( obj ) ,
" Archive object at " PTR_FORMAT " references a non-archive object at " PTR_FORMAT ,
p2i ( p ) , p2i ( obj ) ) ;
} else {
assert ( _hr - > is_closed_archive ( ) , " should be closed archive region " ) ;
guarantee ( obj = = NULL | | G1ArchiveAllocator : : is_closed_archive_object ( obj ) ,
" Archive object at " PTR_FORMAT " references a non-archive object at " PTR_FORMAT ,
p2i ( p ) , p2i ( obj ) ) ;
}
2016-01-26 15:28:31 +01:00
}
} ;
2017-08-14 14:32:17 -04:00
class VerifyObjectInArchiveRegionClosure : public ObjectClosure {
HeapRegion * _hr ;
2016-01-26 15:28:31 +01:00
public :
2017-08-14 14:32:17 -04:00
VerifyObjectInArchiveRegionClosure ( HeapRegion * hr , bool verbose )
: _hr ( hr ) { }
2016-01-26 15:28:31 +01:00
// Verify that all object pointers are to archive regions.
void do_object ( oop o ) {
2017-08-14 14:32:17 -04:00
VerifyArchiveOopClosure checkOop ( _hr ) ;
2016-01-26 15:28:31 +01:00
assert ( o ! = NULL , " Should not be here for NULL oops " ) ;
2018-06-26 13:54:19 +02:00
o - > oop_iterate ( & checkOop ) ;
2016-01-26 15:28:31 +01:00
}
} ;
2017-08-14 14:32:17 -04:00
// Should be only used at CDS dump time
class VerifyArchivePointerRegionClosure : public HeapRegionClosure {
private :
G1CollectedHeap * _g1h ;
public :
VerifyArchivePointerRegionClosure ( G1CollectedHeap * g1h ) { }
2018-02-09 13:09:55 +01:00
virtual bool do_heap_region ( HeapRegion * r ) {
2017-08-14 14:32:17 -04:00
if ( r - > is_archive ( ) ) {
VerifyObjectInArchiveRegionClosure verify_oop_pointers ( r , false ) ;
r - > object_iterate ( & verify_oop_pointers ) ;
}
return false ;
}
} ;
void G1HeapVerifier : : verify_archive_regions ( ) {
G1CollectedHeap * g1h = G1CollectedHeap : : heap ( ) ;
VerifyArchivePointerRegionClosure cl ( NULL ) ;
g1h - > heap_region_iterate ( & cl ) ;
}
2016-01-26 15:28:31 +01:00
class VerifyRegionClosure : public HeapRegionClosure {
private :
bool _par ;
VerifyOption _vo ;
bool _failures ;
public :
// _vo == UsePrevMarking -> use "prev" marking information,
// _vo == UseNextMarking -> use "next" marking information,
2017-11-14 11:33:23 +01:00
// _vo == UseFullMarking -> use "next" marking bitmap but no TAMS
2016-01-26 15:28:31 +01:00
VerifyRegionClosure ( bool par , VerifyOption vo )
: _par ( par ) ,
_vo ( vo ) ,
_failures ( false ) { }
bool failures ( ) {
return _failures ;
}
2018-02-09 13:09:55 +01:00
bool do_heap_region ( HeapRegion * r ) {
2018-03-26 16:51:43 +02:00
guarantee ( ! r - > is_young ( ) | | r - > rem_set ( ) - > is_complete ( ) , " Remembered set for Young region %u must be complete, is %s " , r - > hrm_index ( ) , r - > rem_set ( ) - > get_state_str ( ) ) ;
// Humongous and old regions regions might be of any state, so can't check here.
guarantee ( ! r - > is_free ( ) | | ! r - > rem_set ( ) - > is_tracked ( ) , " Remembered set for free region %u must be untracked, is %s " , r - > hrm_index ( ) , r - > rem_set ( ) - > get_state_str ( ) ) ;
2018-06-14 13:07:33 +02:00
// Verify that the continues humongous regions' remembered set state matches the
// one from the starts humongous region.
if ( r - > is_continues_humongous ( ) ) {
if ( r - > rem_set ( ) - > get_state_str ( ) ! = r - > humongous_start_region ( ) - > rem_set ( ) - > get_state_str ( ) ) {
log_error ( gc , verify ) ( " Remset states differ: Region %u (%s) remset %s with starts region %u (%s) remset %s " ,
r - > hrm_index ( ) ,
r - > get_short_type_str ( ) ,
r - > rem_set ( ) - > get_state_str ( ) ,
r - > humongous_start_region ( ) - > hrm_index ( ) ,
r - > humongous_start_region ( ) - > get_short_type_str ( ) ,
r - > humongous_start_region ( ) - > rem_set ( ) - > get_state_str ( ) ) ;
_failures = true ;
}
}
2016-01-26 15:28:31 +01:00
// For archive regions, verify there are no heap pointers to
// non-pinned regions. For all others, verify liveness info.
2017-08-14 14:32:17 -04:00
if ( r - > is_closed_archive ( ) ) {
VerifyObjectInArchiveRegionClosure verify_oop_pointers ( r , false ) ;
2016-01-26 15:28:31 +01:00
r - > object_iterate ( & verify_oop_pointers ) ;
return true ;
2017-08-14 14:32:17 -04:00
} else if ( r - > is_open_archive ( ) ) {
VerifyObjsInRegionClosure verify_open_archive_oop ( r , _vo ) ;
r - > object_iterate ( & verify_open_archive_oop ) ;
return true ;
} else if ( ! r - > is_continues_humongous ( ) ) {
2016-01-26 15:28:31 +01:00
bool failures = false ;
r - > verify ( _vo , & failures ) ;
if ( failures ) {
_failures = true ;
} else if ( ! r - > is_starts_humongous ( ) ) {
VerifyObjsInRegionClosure not_dead_yet_cl ( r , _vo ) ;
r - > object_iterate ( & not_dead_yet_cl ) ;
if ( _vo ! = VerifyOption_G1UseNextMarking ) {
if ( r - > max_live_bytes ( ) < not_dead_yet_cl . live_bytes ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " [ " PTR_FORMAT " , " PTR_FORMAT " ] max_live_bytes " SIZE_FORMAT " < calculated " SIZE_FORMAT ,
p2i ( r - > bottom ( ) ) , p2i ( r - > end ( ) ) , r - > max_live_bytes ( ) , not_dead_yet_cl . live_bytes ( ) ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
}
} else {
// When vo == UseNextMarking we cannot currently do a sanity
// check on the live bytes as the calculation has not been
// finalized yet.
}
}
}
return false ; // stop the region iteration if we hit a failure
}
} ;
// This is the task used for parallel verification of the heap regions
class G1ParVerifyTask : public AbstractGangTask {
private :
G1CollectedHeap * _g1h ;
VerifyOption _vo ;
bool _failures ;
HeapRegionClaimer _hrclaimer ;
public :
// _vo == UsePrevMarking -> use "prev" marking information,
// _vo == UseNextMarking -> use "next" marking information,
2017-11-14 11:33:23 +01:00
// _vo == UseFullMarking -> use "next" marking bitmap but no TAMS
2016-01-26 15:28:31 +01:00
G1ParVerifyTask ( G1CollectedHeap * g1h , VerifyOption vo ) :
AbstractGangTask ( " Parallel verify task " ) ,
_g1h ( g1h ) ,
_vo ( vo ) ,
_failures ( false ) ,
_hrclaimer ( g1h - > workers ( ) - > active_workers ( ) ) { }
bool failures ( ) {
return _failures ;
}
void work ( uint worker_id ) {
HandleMark hm ;
VerifyRegionClosure blk ( true , _vo ) ;
2017-11-14 11:33:23 +01:00
_g1h - > heap_region_par_iterate_from_worker_offset ( & blk , & _hrclaimer , worker_id ) ;
2016-01-26 15:28:31 +01:00
if ( blk . failures ( ) ) {
_failures = true ;
}
}
} ;
2017-12-01 08:56:22 +01:00
void G1HeapVerifier : : enable_verification_type ( G1VerifyType type ) {
// First enable will clear _enabled_verification_types.
if ( _enabled_verification_types = = G1VerifyAll ) {
_enabled_verification_types = type ;
} else {
_enabled_verification_types | = type ;
}
}
bool G1HeapVerifier : : should_verify ( G1VerifyType type ) {
return ( _enabled_verification_types & type ) = = type ;
}
2016-01-26 15:28:31 +01:00
void G1HeapVerifier : : verify ( VerifyOption vo ) {
if ( ! SafepointSynchronize : : is_at_safepoint ( ) ) {
log_info ( gc , verify ) ( " Skipping verification. Not at safepoint. " ) ;
}
assert ( Thread : : current ( ) - > is_VM_thread ( ) ,
" Expected to be executed serially by the VM thread at this point " ) ;
log_debug ( gc , verify ) ( " Roots " ) ;
VerifyRootsClosure rootsCl ( vo ) ;
2017-10-03 16:42:04 -04:00
VerifyCLDClosure cldCl ( _g1h , & rootsCl ) ;
2016-01-26 15:28:31 +01:00
// We apply the relevant closures to all the oops in the
// system dictionary, class loader data graph, the string table
// and the nmethods in the code cache.
G1VerifyCodeRootOopClosure codeRootsCl ( _g1h , & rootsCl , vo ) ;
G1VerifyCodeRootBlobClosure blobsCl ( & codeRootsCl ) ;
{
G1RootProcessor root_processor ( _g1h , 1 ) ;
root_processor . process_all_roots ( & rootsCl ,
& cldCl ,
& blobsCl ) ;
}
bool failures = rootsCl . failures ( ) | | codeRootsCl . failures ( ) ;
2018-03-29 14:07:59 +02:00
if ( ! _g1h - > g1_policy ( ) - > collector_state ( ) - > in_full_gc ( ) ) {
2016-01-26 15:28:31 +01:00
// If we're verifying during a full GC then the region sets
// will have been torn down at the start of the GC. Therefore
// verifying the region sets will fail. So we only verify
// the region sets when not in a full GC.
log_debug ( gc , verify ) ( " HeapRegionSets " ) ;
verify_region_sets ( ) ;
}
log_debug ( gc , verify ) ( " HeapRegions " ) ;
if ( GCParallelVerificationEnabled & & ParallelGCThreads > 1 ) {
G1ParVerifyTask task ( _g1h , vo ) ;
_g1h - > workers ( ) - > run_task ( & task ) ;
if ( task . failures ( ) ) {
failures = true ;
}
} else {
VerifyRegionClosure blk ( false , vo ) ;
_g1h - > heap_region_iterate ( & blk ) ;
if ( blk . failures ( ) ) {
failures = true ;
}
}
if ( G1StringDedup : : is_enabled ( ) ) {
log_debug ( gc , verify ) ( " StrDedup " ) ;
G1StringDedup : : verify ( ) ;
}
if ( failures ) {
2018-03-26 16:51:43 +02:00
log_error ( gc , verify ) ( " Heap after failed verification (kind %d): " , vo ) ;
2016-01-26 15:28:31 +01:00
// It helps to have the per-region information in the output to
// help us track down what went wrong. This is why we call
// print_extended_on() instead of print_on().
2016-04-04 09:15:15 +02:00
Log ( gc , verify ) log ;
2016-01-26 15:28:31 +01:00
ResourceMark rm ;
2017-07-21 09:50:12 +02:00
LogStream ls ( log . error ( ) ) ;
_g1h - > print_extended_on ( & ls ) ;
2016-01-26 15:28:31 +01:00
}
guarantee ( ! failures , " there should not have been any failures " ) ;
}
// Heap region set verification
class VerifyRegionListsClosure : public HeapRegionClosure {
private :
HeapRegionSet * _old_set ;
HeapRegionSet * _humongous_set ;
HeapRegionManager * _hrm ;
public :
uint _old_count ;
uint _humongous_count ;
uint _free_count ;
VerifyRegionListsClosure ( HeapRegionSet * old_set ,
HeapRegionSet * humongous_set ,
HeapRegionManager * hrm ) :
_old_set ( old_set ) , _humongous_set ( humongous_set ) , _hrm ( hrm ) ,
_old_count ( ) , _humongous_count ( ) , _free_count ( ) { }
2018-02-09 13:09:55 +01:00
bool do_heap_region ( HeapRegion * hr ) {
2016-01-26 15:28:31 +01:00
if ( hr - > is_young ( ) ) {
// TODO
} else if ( hr - > is_humongous ( ) ) {
assert ( hr - > containing_set ( ) = = _humongous_set , " Heap region %u is humongous but not in humongous set. " , hr - > hrm_index ( ) ) ;
_humongous_count + + ;
} else if ( hr - > is_empty ( ) ) {
assert ( _hrm - > is_free ( hr ) , " Heap region %u is empty but not on the free list. " , hr - > hrm_index ( ) ) ;
_free_count + + ;
} else if ( hr - > is_old ( ) ) {
assert ( hr - > containing_set ( ) = = _old_set , " Heap region %u is old but not in the old set. " , hr - > hrm_index ( ) ) ;
_old_count + + ;
} else {
// There are no other valid region types. Check for one invalid
// one we can identify: pinned without old or humongous set.
assert ( ! hr - > is_pinned ( ) , " Heap region %u is pinned but not old (archive) or humongous. " , hr - > hrm_index ( ) ) ;
ShouldNotReachHere ( ) ;
}
return false ;
}
void verify_counts ( HeapRegionSet * old_set , HeapRegionSet * humongous_set , HeapRegionManager * free_list ) {
guarantee ( old_set - > length ( ) = = _old_count , " Old set count mismatch. Expected %u, actual %u. " , old_set - > length ( ) , _old_count ) ;
guarantee ( humongous_set - > length ( ) = = _humongous_count , " Hum set count mismatch. Expected %u, actual %u. " , humongous_set - > length ( ) , _humongous_count ) ;
guarantee ( free_list - > num_free_regions ( ) = = _free_count , " Free list count mismatch. Expected %u, actual %u. " , free_list - > num_free_regions ( ) , _free_count ) ;
}
} ;
void G1HeapVerifier : : verify_region_sets ( ) {
assert_heap_locked_or_at_safepoint ( true /* should_be_vm_thread */ ) ;
// First, check the explicit lists.
_g1h - > _hrm . verify ( ) ;
// Finally, make sure that the region accounting in the lists is
// consistent with what we see in the heap.
VerifyRegionListsClosure cl ( & _g1h - > _old_set , & _g1h - > _humongous_set , & _g1h - > _hrm ) ;
_g1h - > heap_region_iterate ( & cl ) ;
cl . verify_counts ( & _g1h - > _old_set , & _g1h - > _humongous_set , & _g1h - > _hrm ) ;
}
void G1HeapVerifier : : prepare_for_verify ( ) {
if ( SafepointSynchronize : : is_at_safepoint ( ) | | ! UseTLAB ) {
_g1h - > ensure_parsability ( false ) ;
}
}
2017-12-01 08:56:22 +01:00
double G1HeapVerifier : : verify ( G1VerifyType type , VerifyOption vo , const char * msg ) {
2016-01-26 15:28:31 +01:00
double verify_time_ms = 0.0 ;
2017-12-01 08:56:22 +01:00
if ( should_verify ( type ) & & _g1h - > total_collections ( ) > = VerifyGCStartAt ) {
2016-01-26 15:28:31 +01:00
double verify_start = os : : elapsedTime ( ) ;
HandleMark hm ; // Discard invalid handles created during verification
prepare_for_verify ( ) ;
2017-12-01 08:56:22 +01:00
Universe : : verify ( vo , msg ) ;
2016-01-26 15:28:31 +01:00
verify_time_ms = ( os : : elapsedTime ( ) - verify_start ) * 1000 ;
}
return verify_time_ms ;
}
2017-12-01 08:56:22 +01:00
void G1HeapVerifier : : verify_before_gc ( G1VerifyType type ) {
if ( VerifyBeforeGC ) {
double verify_time_ms = verify ( type , VerifyOption_G1UsePrevMarking , " Before GC " ) ;
_g1h - > g1_policy ( ) - > phase_times ( ) - > record_verify_before_time_ms ( verify_time_ms ) ;
}
2016-01-26 15:28:31 +01:00
}
2017-12-01 08:56:22 +01:00
void G1HeapVerifier : : verify_after_gc ( G1VerifyType type ) {
if ( VerifyAfterGC ) {
double verify_time_ms = verify ( type , VerifyOption_G1UsePrevMarking , " After GC " ) ;
_g1h - > g1_policy ( ) - > phase_times ( ) - > record_verify_after_time_ms ( verify_time_ms ) ;
}
2016-01-26 15:28:31 +01:00
}
# ifndef PRODUCT
class G1VerifyCardTableCleanup : public HeapRegionClosure {
G1HeapVerifier * _verifier ;
public :
2018-02-26 09:34:12 +01:00
G1VerifyCardTableCleanup ( G1HeapVerifier * verifier )
: _verifier ( verifier ) { }
2018-02-09 13:09:55 +01:00
virtual bool do_heap_region ( HeapRegion * r ) {
2016-01-26 15:28:31 +01:00
if ( r - > is_survivor ( ) ) {
_verifier - > verify_dirty_region ( r ) ;
} else {
_verifier - > verify_not_dirty_region ( r ) ;
}
return false ;
}
} ;
void G1HeapVerifier : : verify_card_table_cleanup ( ) {
if ( G1VerifyCTCleanup | | VerifyAfterGC ) {
2018-02-26 09:34:12 +01:00
G1VerifyCardTableCleanup cleanup_verifier ( this ) ;
2016-01-26 15:28:31 +01:00
_g1h - > heap_region_iterate ( & cleanup_verifier ) ;
}
}
void G1HeapVerifier : : verify_not_dirty_region ( HeapRegion * hr ) {
// All of the region should be clean.
2018-02-26 09:34:12 +01:00
G1CardTable * ct = _g1h - > card_table ( ) ;
2016-01-26 15:28:31 +01:00
MemRegion mr ( hr - > bottom ( ) , hr - > end ( ) ) ;
2018-02-26 09:34:12 +01:00
ct - > verify_not_dirty_region ( mr ) ;
2016-01-26 15:28:31 +01:00
}
void G1HeapVerifier : : verify_dirty_region ( HeapRegion * hr ) {
// We cannot guarantee that [bottom(),end()] is dirty. Threads
// dirty allocated blocks as they allocate them. The thread that
// retires each region and replaces it with a new one will do a
// maximal allocation to fill in [pre_dummy_top(),end()] but will
// not dirty that area (one less thing to have to do while holding
// a lock). So we can only verify that [bottom(),pre_dummy_top()]
// is dirty.
2018-02-26 09:34:12 +01:00
G1CardTable * ct = _g1h - > card_table ( ) ;
2016-01-26 15:28:31 +01:00
MemRegion mr ( hr - > bottom ( ) , hr - > pre_dummy_top ( ) ) ;
if ( hr - > is_young ( ) ) {
2018-02-26 09:34:12 +01:00
ct - > verify_g1_young_region ( mr ) ;
2016-01-26 15:28:31 +01:00
} else {
2018-02-26 09:34:12 +01:00
ct - > verify_dirty_region ( mr ) ;
2016-01-26 15:28:31 +01:00
}
}
2016-07-06 11:22:55 +02:00
class G1VerifyDirtyYoungListClosure : public HeapRegionClosure {
private :
G1HeapVerifier * _verifier ;
public :
G1VerifyDirtyYoungListClosure ( G1HeapVerifier * verifier ) : HeapRegionClosure ( ) , _verifier ( verifier ) { }
2018-02-09 13:09:55 +01:00
virtual bool do_heap_region ( HeapRegion * r ) {
2016-07-06 11:22:55 +02:00
_verifier - > verify_dirty_region ( r ) ;
return false ;
2016-01-26 15:28:31 +01:00
}
2016-07-06 11:22:55 +02:00
} ;
2016-01-26 15:28:31 +01:00
void G1HeapVerifier : : verify_dirty_young_regions ( ) {
2016-07-06 11:22:55 +02:00
G1VerifyDirtyYoungListClosure cl ( this ) ;
_g1h - > collection_set ( ) - > iterate ( & cl ) ;
2016-01-26 15:28:31 +01:00
}
2017-08-04 14:15:42 +02:00
bool G1HeapVerifier : : verify_no_bits_over_tams ( const char * bitmap_name , const G1CMBitMap * const bitmap ,
2016-01-26 15:28:31 +01:00
HeapWord * tams , HeapWord * end ) {
guarantee ( tams < = end ,
" tams: " PTR_FORMAT " end: " PTR_FORMAT , p2i ( tams ) , p2i ( end ) ) ;
2017-08-04 14:15:42 +02:00
HeapWord * result = bitmap - > get_next_marked_addr ( tams , end ) ;
2016-01-26 15:28:31 +01:00
if ( result < end ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## wrong marked address on %s bitmap: " PTR_FORMAT , bitmap_name , p2i ( result ) ) ;
log_error ( gc , verify ) ( " ## %s tams: " PTR_FORMAT " end: " PTR_FORMAT , bitmap_name , p2i ( tams ) , p2i ( end ) ) ;
2016-01-26 15:28:31 +01:00
return false ;
}
return true ;
}
bool G1HeapVerifier : : verify_bitmaps ( const char * caller , HeapRegion * hr ) {
2017-10-23 11:46:12 +02:00
const G1CMBitMap * const prev_bitmap = _g1h - > concurrent_mark ( ) - > prev_mark_bitmap ( ) ;
const G1CMBitMap * const next_bitmap = _g1h - > concurrent_mark ( ) - > next_mark_bitmap ( ) ;
2016-01-26 15:28:31 +01:00
HeapWord * ptams = hr - > prev_top_at_mark_start ( ) ;
HeapWord * ntams = hr - > next_top_at_mark_start ( ) ;
HeapWord * end = hr - > end ( ) ;
bool res_p = verify_no_bits_over_tams ( " prev " , prev_bitmap , ptams , end ) ;
bool res_n = true ;
2018-04-04 11:21:14 +02:00
// We cannot verify the next bitmap while we are about to clear it.
if ( ! _g1h - > collector_state ( ) - > clearing_next_bitmap ( ) ) {
2016-01-26 15:28:31 +01:00
res_n = verify_no_bits_over_tams ( " next " , next_bitmap , ntams , end ) ;
}
if ( ! res_p | | ! res_n ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " #### Bitmap verification failed for " HR_FORMAT , HR_FORMAT_PARAMS ( hr ) ) ;
log_error ( gc , verify ) ( " #### Caller: %s " , caller ) ;
2016-01-26 15:28:31 +01:00
return false ;
}
return true ;
}
void G1HeapVerifier : : check_bitmaps ( const char * caller , HeapRegion * hr ) {
2018-04-04 11:21:14 +02:00
if ( ! G1VerifyBitmaps ) {
return ;
}
2016-01-26 15:28:31 +01:00
guarantee ( verify_bitmaps ( caller , hr ) , " bitmap verification " ) ;
}
class G1VerifyBitmapClosure : public HeapRegionClosure {
private :
const char * _caller ;
G1HeapVerifier * _verifier ;
bool _failures ;
public :
G1VerifyBitmapClosure ( const char * caller , G1HeapVerifier * verifier ) :
_caller ( caller ) , _verifier ( verifier ) , _failures ( false ) { }
bool failures ( ) { return _failures ; }
2018-02-09 13:09:55 +01:00
virtual bool do_heap_region ( HeapRegion * hr ) {
2016-01-26 15:28:31 +01:00
bool result = _verifier - > verify_bitmaps ( _caller , hr ) ;
if ( ! result ) {
_failures = true ;
}
return false ;
}
} ;
void G1HeapVerifier : : check_bitmaps ( const char * caller ) {
2018-04-04 11:21:14 +02:00
if ( ! G1VerifyBitmaps ) {
return ;
}
2016-01-26 15:28:31 +01:00
G1VerifyBitmapClosure cl ( caller , this ) ;
_g1h - > heap_region_iterate ( & cl ) ;
guarantee ( ! cl . failures ( ) , " bitmap verification " ) ;
}
class G1CheckCSetFastTableClosure : public HeapRegionClosure {
private :
bool _failures ;
public :
G1CheckCSetFastTableClosure ( ) : HeapRegionClosure ( ) , _failures ( false ) { }
2018-02-09 13:09:55 +01:00
virtual bool do_heap_region ( HeapRegion * hr ) {
2016-01-26 15:28:31 +01:00
uint i = hr - > hrm_index ( ) ;
InCSetState cset_state = ( InCSetState ) G1CollectedHeap : : heap ( ) - > _in_cset_fast_test . get_by_index ( i ) ;
if ( hr - > is_humongous ( ) ) {
if ( hr - > in_collection_set ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## humongous region %u in CSet " , i ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
return true ;
}
if ( cset_state . is_in_cset ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## inconsistent cset state " CSETSTATE_FORMAT " for humongous region %u " , cset_state . value ( ) , i ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
return true ;
}
if ( hr - > is_continues_humongous ( ) & & cset_state . is_humongous ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## inconsistent cset state " CSETSTATE_FORMAT " for continues humongous region %u " , cset_state . value ( ) , i ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
return true ;
}
} else {
if ( cset_state . is_humongous ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## inconsistent cset state " CSETSTATE_FORMAT " for non-humongous region %u " , cset_state . value ( ) , i ) ;
2016-01-26 15:28:31 +01:00
_failures = true ;
return true ;
}
if ( hr - > in_collection_set ( ) ! = cset_state . is_in_cset ( ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## in CSet %d / cset state " CSETSTATE_FORMAT " inconsistency for region %u " ,
2016-01-26 15:28:31 +01:00
hr - > in_collection_set ( ) , cset_state . value ( ) , i ) ;
_failures = true ;
return true ;
}
if ( cset_state . is_in_cset ( ) ) {
if ( hr - > is_young ( ) ! = ( cset_state . is_young ( ) ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## is_young %d / cset state " CSETSTATE_FORMAT " inconsistency for region %u " ,
2016-01-26 15:28:31 +01:00
hr - > is_young ( ) , cset_state . value ( ) , i ) ;
_failures = true ;
return true ;
}
if ( hr - > is_old ( ) ! = ( cset_state . is_old ( ) ) ) {
2016-02-11 08:55:36 +01:00
log_error ( gc , verify ) ( " ## is_old %d / cset state " CSETSTATE_FORMAT " inconsistency for region %u " ,
2016-01-26 15:28:31 +01:00
hr - > is_old ( ) , cset_state . value ( ) , i ) ;
_failures = true ;
return true ;
}
}
}
return false ;
}
bool failures ( ) const { return _failures ; }
} ;
bool G1HeapVerifier : : check_cset_fast_test ( ) {
G1CheckCSetFastTableClosure cl ;
_g1h - > _hrm . iterate ( & cl ) ;
return ! cl . failures ( ) ;
}
# endif // PRODUCT