@@ -381,7 +381,9 @@ enum {
381
381
class DumpWriter : public StackObj {
382
382
private:
383
383
enum {
384
- io_buffer_size = 8 *M
384
+ io_buffer_max_size = 8 *M,
385
+ io_buffer_min_size = 64 *K,
386
+ dump_segment_header_size = 9
385
387
};
386
388
387
389
int _fd; // file descriptor (-1 if dump file not open)
@@ -391,13 +393,19 @@ class DumpWriter : public StackObj {
391
393
size_t _size;
392
394
size_t _pos;
393
395
394
- jlong _dump_start;
396
+ bool _in_dump_segment; // Are we currently in a dump segment?
397
+ bool _is_huge_sub_record; // Are we writing a sub-record larger than the buffer size?
398
+ DEBUG_ONLY (size_t _sub_record_left;) // The bytes not written for the current sub-record.
399
+ DEBUG_ONLY (bool _sub_record_ended;) // True if we have called the end_sub_record().
395
400
396
401
char * _error; // error message when I/O fails
397
402
398
403
void set_file_descriptor (int fd) { _fd = fd; }
399
404
int file_descriptor () const { return _fd; }
400
405
406
+ bool is_open () const { return file_descriptor () >= 0 ; }
407
+ void flush ();
408
+
401
409
char * buffer () const { return _buffer; }
402
410
size_t buffer_size () const { return _size; }
403
411
size_t position () const { return _pos; }
@@ -413,28 +421,12 @@ class DumpWriter : public StackObj {
413
421
~DumpWriter ();
414
422
415
423
void close ();
416
- bool is_open () const { return file_descriptor () >= 0 ; }
417
- void flush ();
418
-
419
- jlong dump_start () const { return _dump_start; }
420
- void set_dump_start (jlong pos);
421
- julong current_record_length ();
422
424
423
425
// total number of bytes written to the disk
424
426
julong bytes_written () const { return _bytes_written; }
425
427
426
- // adjust the number of bytes written to disk (used to keep the count
427
- // of the number of bytes written in case of rewrites)
428
- void adjust_bytes_written (jlong n) { _bytes_written += n; }
429
-
430
- // number of (buffered) bytes as yet unwritten to the dump file
431
- size_t bytes_unwritten () const { return position (); }
432
-
433
428
char * error () const { return _error; }
434
429
435
- jlong current_offset ();
436
- void seek_to_offset (jlong pos);
437
-
438
430
// writer functions
439
431
void write_raw (void * s, size_t len);
440
432
void write_u1 (u1 x) { write_raw ((void *)&x, 1 ); }
@@ -445,38 +437,43 @@ class DumpWriter : public StackObj {
445
437
void write_symbolID (Symbol* o);
446
438
void write_classID (Klass* k);
447
439
void write_id (u4 x);
440
+
441
+ // Start a new sub-record. Starts a new heap dump segment if needed.
442
+ void start_sub_record (u1 tag, u4 len);
443
+ // Ends the current sub-record.
444
+ void end_sub_record ();
445
+ // Finishes the current dump segment if not already finished.
446
+ void finish_dump_segment ();
448
447
};
449
448
450
- DumpWriter::DumpWriter (const char * path) {
449
+ DumpWriter::DumpWriter (const char * path) : _fd(-1 ), _bytes_written(0 ), _pos(0 ),
450
+ _in_dump_segment(false ), _error(NULL ) {
451
451
// try to allocate an I/O buffer of io_buffer_size. If there isn't
452
452
// sufficient memory then reduce size until we can allocate something.
453
- _size = io_buffer_size ;
453
+ _size = io_buffer_max_size ;
454
454
do {
455
455
_buffer = (char *)os::malloc (_size, mtInternal);
456
456
if (_buffer == NULL ) {
457
457
_size = _size >> 1 ;
458
458
}
459
- } while (_buffer == NULL && _size > 0 );
460
- assert ((_size > 0 && _buffer != NULL ) || (_size == 0 && _buffer == NULL ), " sanity check " );
461
- _pos = 0 ;
462
- _error = NULL ;
463
- _bytes_written = 0L ;
464
- _dump_start = (jlong)- 1 ;
465
- _fd = os::create_binary_file (path, false ); // don't replace existing file
466
-
467
- // if the open failed we record the error
468
- if (_fd < 0 ) {
469
- _error = ( char *) os::strdup ( os::strerror (errno));
459
+ } while (_buffer == NULL && _size >= io_buffer_min_size );
460
+
461
+ if (_buffer == NULL ) {
462
+ set_error ( " Could not allocate buffer memory for heap dump " ) ;
463
+ } else {
464
+ _fd = os::create_binary_file (path, false ); // don't replace existing file
465
+
466
+ // if the open failed we record the error
467
+ if (_fd < 0 ) {
468
+ set_error ( os::strerror (errno));
469
+ }
470
470
}
471
471
}
472
472
473
473
DumpWriter::~DumpWriter () {
474
- // flush and close dump file
475
- if (is_open ()) {
476
- close ();
477
- }
478
- if (_buffer != NULL ) os::free (_buffer);
479
- if (_error != NULL ) os::free (_error);
474
+ close ();
475
+ os::free (_buffer);
476
+ os::free (_error);
480
477
}
481
478
482
479
// closes dump file (if open)
@@ -489,29 +486,13 @@ void DumpWriter::close() {
489
486
}
490
487
}
491
488
492
- // sets the dump starting position
493
- void DumpWriter::set_dump_start (jlong pos) {
494
- _dump_start = pos;
495
- }
496
-
497
- julong DumpWriter::current_record_length () {
498
- if (is_open ()) {
499
- // calculate the size of the dump record
500
- julong dump_end = bytes_written () + bytes_unwritten ();
501
- assert (dump_end == (size_t )current_offset (), " checking" );
502
- julong dump_len = dump_end - dump_start () - 4 ;
503
- return dump_len;
504
- }
505
- return 0 ;
506
- }
507
-
508
489
// write directly to the file
509
490
void DumpWriter::write_internal (void * s, size_t len) {
510
491
if (is_open ()) {
511
492
const char * pos = (char *)s;
512
493
ssize_t n = 0 ;
513
494
while (len > 0 ) {
514
- uint tmp = (uint )MIN2 (len, (size_t )UINT_MAX );
495
+ uint tmp = (uint )MIN2 (len, (size_t )INT_MAX );
515
496
n = os::write (file_descriptor (), pos, tmp);
516
497
517
498
if (n < 0 ) {
@@ -531,53 +512,30 @@ void DumpWriter::write_internal(void* s, size_t len) {
531
512
532
513
// write raw bytes
533
514
void DumpWriter::write_raw (void * s, size_t len) {
534
- if (is_open ()) {
535
- // flush buffer to make room
536
- if ((position () + len) >= buffer_size ()) {
537
- flush ();
538
- }
515
+ assert (!_in_dump_segment || (_sub_record_left >= len), " sub-record too large" );
516
+ debug_only (_sub_record_left -= len);
517
+
518
+ // flush buffer to make room
519
+ if (len > buffer_size () - position ()) {
520
+ assert (!_in_dump_segment || _is_huge_sub_record, " Cannot overflow in non-huge sub-record." );
521
+ flush ();
539
522
540
- // buffer not available or too big to buffer it
541
- if (( buffer () == NULL ) || ( len >= buffer_size () )) {
523
+ // If larger than the buffer, just write it directly.
524
+ if (len > buffer_size ()) {
542
525
write_internal (s, len);
543
- } else {
544
- // Should optimize this for u1/u2/u4/u8 sizes.
545
- memcpy (buffer () + position (), s, len);
546
- set_position (position () + len);
526
+
527
+ return ;
547
528
}
548
529
}
530
+
531
+ memcpy (buffer () + position (), s, len);
532
+ set_position (position () + len);
549
533
}
550
534
551
535
// flush any buffered bytes to the file
552
536
void DumpWriter::flush () {
553
- if (is_open () && position () > 0 ) {
554
- write_internal (buffer (), position ());
555
- set_position (0 );
556
- }
557
- }
558
-
559
- jlong DumpWriter::current_offset () {
560
- if (is_open ()) {
561
- // the offset is the file offset plus whatever we have buffered
562
- jlong offset = os::current_file_offset (file_descriptor ());
563
- assert (offset >= 0 , " lseek failed" );
564
- return offset + position ();
565
- } else {
566
- return (jlong)-1 ;
567
- }
568
- }
569
-
570
- void DumpWriter::seek_to_offset (jlong off) {
571
- assert (off >= 0 , " bad offset" );
572
-
573
- // need to flush before seeking
574
- flush ();
575
-
576
- // may be closed due to I/O error
577
- if (is_open ()) {
578
- jlong n = os::seek_to_file_offset (file_descriptor (), off);
579
- assert (n >= 0 , " lseek failed" );
580
- }
537
+ write_internal (buffer (), position ());
538
+ set_position (0 );
581
539
}
582
540
583
541
void DumpWriter::write_u2 (u2 x) {
@@ -629,7 +587,58 @@ void DumpWriter::write_classID(Klass* k) {
629
587
write_objectID (k->java_mirror ());
630
588
}
631
589
590
+ void DumpWriter::finish_dump_segment () {
591
+ if (_in_dump_segment) {
592
+ assert (_sub_record_left == 0 , " Last sub-record not written completely" );
593
+ assert (_sub_record_ended, " sub-record must have ended" );
594
+
595
+ // Fix up the dump segment length if we haven't written a huge sub-record last
596
+ // (in which case the segment length was already set to the correct value initially).
597
+ if (!_is_huge_sub_record) {
598
+ assert (position () > dump_segment_header_size, " Dump segment should have some content" );
599
+ Bytes::put_Java_u4 ((address) (buffer () + 5 ), (u4) (position () - dump_segment_header_size));
600
+ }
601
+
602
+ flush ();
603
+ _in_dump_segment = false ;
604
+ }
605
+ }
606
+
607
+ void DumpWriter::start_sub_record (u1 tag, u4 len) {
608
+ if (!_in_dump_segment) {
609
+ if (position () > 0 ) {
610
+ flush ();
611
+ assert (position () == 0 , " Must be at the start" );
612
+ }
632
613
614
+ write_u1 (HPROF_HEAP_DUMP_SEGMENT);
615
+ write_u4 (0 ); // timestamp
616
+ // Will be fixed up later if we add more sub-records. If this is a huge sub-record,
617
+ // this is already the correct length, since we don't add more sub-records.
618
+ write_u4 (len);
619
+ _in_dump_segment = true ;
620
+ _is_huge_sub_record = len > buffer_size () - dump_segment_header_size;
621
+ } else if (_is_huge_sub_record || (len > buffer_size () - position ())) {
622
+ // This object will not fit in completely or the last sub-record was huge.
623
+ // Finish the current segement and try again.
624
+ finish_dump_segment ();
625
+ start_sub_record (tag, len);
626
+
627
+ return ;
628
+ }
629
+
630
+ debug_only (_sub_record_left = len);
631
+ debug_only (_sub_record_ended = false );
632
+
633
+ write_u1 (tag);
634
+ }
635
+
636
+ void DumpWriter::end_sub_record () {
637
+ assert (_in_dump_segment, " must be in dump segment" );
638
+ assert (_sub_record_left == 0 , " sub-record not written completely" );
639
+ assert (!_sub_record_ended, " Must not have ended yet" );
640
+ debug_only (_sub_record_ended = true );
641
+ }
633
642
634
643
// Support class with a collection of functions used when dumping the heap
635
644
@@ -643,6 +652,8 @@ class DumperSupport : AllStatic {
643
652
static hprofTag sig2tag (Symbol* sig);
644
653
// returns hprof tag for the given basic type
645
654
static hprofTag type2tag (BasicType type);
655
+ // Returns the size of the data to write.
656
+ static u4 sig2size (Symbol* sig);
646
657
647
658
// returns the size of the instance of the given class
648
659
static u4 instance_size (Klass* k);
@@ -653,10 +664,14 @@ class DumperSupport : AllStatic {
653
664
static void dump_double (DumpWriter* writer, jdouble d);
654
665
// dumps the raw value of the given field
655
666
static void dump_field_value (DumpWriter* writer, char type, oop obj, int offset);
667
+ // returns the size of the static fields; also counts the static fields
668
+ static u4 get_static_fields_size (InstanceKlass* ik, u2& field_count);
656
669
// dumps static fields of the given class
657
670
static void dump_static_fields (DumpWriter* writer, Klass* k);
658
671
// dump the raw values of the instance fields of the given object
659
672
static void dump_instance_fields (DumpWriter* writer, oop o);
673
+ // get the count of the instance fields for a given class
674
+ static u2 get_instance_fields_count (InstanceKlass* ik);
660
675
// dumps the definition of the instance fields for a given class
661
676
static void dump_instance_field_descriptors (DumpWriter* writer, Klass* k);
662
677
// creates HPROF_GC_INSTANCE_DUMP record for the given object
@@ -678,12 +693,6 @@ class DumperSupport : AllStatic {
678
693
// check if we need to truncate an array
679
694
static int calculate_array_max_length (DumpWriter* writer, arrayOop array, short header_size);
680
695
681
- // writes a HPROF_HEAP_DUMP_SEGMENT record
682
- static void write_dump_header (DumpWriter* writer);
683
-
684
- // fixes up the length of the current dump record
685
- static void write_current_dump_record_length (DumpWriter* writer);
686
-
687
696
// fixes up the current dump record and writes HPROF_HEAP_DUMP_END record
688
697
static void end_of_dump (DumpWriter* writer);
689
698
@@ -736,6 +745,22 @@ hprofTag DumperSupport::type2tag(BasicType type) {
736
745
}
737
746
}
738
747
748
+ u4 DumperSupport::sig2size (Symbol* sig) {
749
+ switch (sig->char_at (0 )) {
750
+ case JVM_SIGNATURE_CLASS:
751
+ case JVM_SIGNATURE_ARRAY: return sizeof (address);
752
+ case JVM_SIGNATURE_BOOLEAN:
753
+ case JVM_SIGNATURE_BYTE: return 1 ;
754
+ case JVM_SIGNATURE_SHORT:
755
+ case JVM_SIGNATURE_CHAR: return 2 ;
756
+ case JVM_SIGNATURE_INT:
757
+ case JVM_SIGNATURE_FLOAT: return 4 ;
758
+ case JVM_SIGNATURE_LONG:
759
+ case JVM_SIGNATURE_DOUBLE: return 8 ;
760
+ default : ShouldNotReachHere (); /* to shut up compiler */ return 0 ;
761
+ }
762
+ }
763
+
739
764
// dump a jfloat
740
765
void DumperSupport::dump_float (DumpWriter* writer, jfloat f) {
741
766
if (g_isnan (f)) {
@@ -833,44 +858,26 @@ void DumperSupport::dump_field_value(DumpWriter* writer, char type, oop obj, int
833
858
u4 DumperSupport::instance_size (Klass* k) {
834
859
HandleMark hm;
835
860
InstanceKlass* ik = InstanceKlass::cast (k);
836
-
837
861
u4 size = 0 ;
838
862
839
863
for (FieldStream fld (ik, false , false ); !fld.eos (); fld.next ()) {
840
864
if (!fld.access_flags ().is_static ()) {
841
- Symbol* sig = fld.signature ();
842
- switch (sig->char_at (0 )) {
843
- case JVM_SIGNATURE_CLASS :
844
- case JVM_SIGNATURE_ARRAY : size += oopSize; break ;
845
-
846
- case JVM_SIGNATURE_BYTE :
847
- case JVM_SIGNATURE_BOOLEAN : size += 1 ; break ;
848
-
849
- case JVM_SIGNATURE_CHAR :
850
- case JVM_SIGNATURE_SHORT : size += 2 ; break ;
851
-
852
- case JVM_SIGNATURE_INT :
853
- case JVM_SIGNATURE_FLOAT : size += 4 ; break ;
854
-
855
- case JVM_SIGNATURE_LONG :
856
- case JVM_SIGNATURE_DOUBLE : size += 8 ; break ;
857
-
858
- default : ShouldNotReachHere ();
859
- }
865
+ size += sig2size (fld.signature ());
860
866
}
861
867
}
862
868
return size;
863
869
}
864
870
865
- // dumps static fields of the given class
866
- void DumperSupport::dump_static_fields (DumpWriter* writer, Klass* k) {
871
+ u4 DumperSupport::get_static_fields_size (InstanceKlass* ik, u2& field_count) {
867
872
HandleMark hm;
868
- InstanceKlass* ik = InstanceKlass::cast (k);
873
+ field_count = 0 ;
874
+ u4 size = 0 ;
869
875
870
- // pass 1 - count the static fields
871
- u2 field_count = 0 ;
872
876
for (FieldStream fldc (ik, true , true ); !fldc.eos (); fldc.next ()) {
873
- if (fldc.access_flags ().is_static ()) field_count++;
877
+ if (fldc.access_flags ().is_static ()) {
878
+ field_count++;
879
+ size += sig2size (fldc.signature ());
880
+ }
874
881
}
875
882
876
883
// Add in resolved_references which is referenced by the cpCache
@@ -879,12 +886,14 @@ void DumperSupport::dump_static_fields(DumpWriter* writer, Klass* k) {
879
886
oop resolved_references = ik->constants ()->resolved_references_or_null ();
880
887
if (resolved_references != NULL ) {
881
888
field_count++;
889
+ size += sizeof (address);
882
890
883
891
// Add in the resolved_references of the used previous versions of the class
884
892
// in the case of RedefineClasses
885
893
InstanceKlass* prev = ik->previous_versions ();
886
894
while (prev != NULL && prev->constants ()->resolved_references_or_null () != NULL ) {
887
895
field_count++;
896
+ size += sizeof (address);
888
897
prev = prev->previous_versions ();
889
898
}
890
899
}
@@ -894,11 +903,19 @@ void DumperSupport::dump_static_fields(DumpWriter* writer, Klass* k) {
894
903
oop init_lock = ik->init_lock ();
895
904
if (init_lock != NULL ) {
896
905
field_count++;
906
+ size += sizeof (address);
897
907
}
898
908
899
- writer->write_u2 (field_count);
909
+ // We write the value itself plus a name and a one byte type tag per field.
910
+ return size + field_count * (sizeof (address) + 1 );
911
+ }
900
912
901
- // pass 2 - dump the field descriptors and raw values
913
+ // dumps static fields of the given class
914
+ void DumperSupport::dump_static_fields (DumpWriter* writer, Klass* k) {
915
+ HandleMark hm;
916
+ InstanceKlass* ik = InstanceKlass::cast (k);
917
+
918
+ // dump the field descriptors and raw values
902
919
for (FieldStream fld (ik, true , true ); !fld.eos (); fld.next ()) {
903
920
if (fld.access_flags ().is_static ()) {
904
921
Symbol* sig = fld.signature ();
@@ -912,6 +929,7 @@ void DumperSupport::dump_static_fields(DumpWriter* writer, Klass* k) {
912
929
}
913
930
914
931
// Add resolved_references for each class that has them
932
+ oop resolved_references = ik->constants ()->resolved_references_or_null ();
915
933
if (resolved_references != NULL ) {
916
934
writer->write_symbolID (vmSymbols::resolved_references_name ()); // name
917
935
writer->write_u1 (sig2tag (vmSymbols::object_array_signature ())); // type
@@ -928,6 +946,7 @@ void DumperSupport::dump_static_fields(DumpWriter* writer, Klass* k) {
928
946
}
929
947
930
948
// Add init lock to the end if the class is not yet initialized
949
+ oop init_lock = ik->init_lock ();
931
950
if (init_lock != NULL ) {
932
951
writer->write_symbolID (vmSymbols::init_lock_name ()); // name
933
952
writer->write_u1 (sig2tag (vmSymbols::int_array_signature ())); // type
@@ -949,19 +968,23 @@ void DumperSupport::dump_instance_fields(DumpWriter* writer, oop o) {
949
968
}
950
969
951
970
// dumps the definition of the instance fields for a given class
952
- void DumperSupport::dump_instance_field_descriptors (DumpWriter* writer, Klass* k ) {
971
+ u2 DumperSupport::get_instance_fields_count (InstanceKlass* ik ) {
953
972
HandleMark hm;
954
- InstanceKlass* ik = InstanceKlass::cast (k);
955
-
956
- // pass 1 - count the instance fields
957
973
u2 field_count = 0 ;
974
+
958
975
for (FieldStream fldc (ik, true , true ); !fldc.eos (); fldc.next ()) {
959
976
if (!fldc.access_flags ().is_static ()) field_count++;
960
977
}
961
978
962
- writer->write_u2 (field_count);
979
+ return field_count;
980
+ }
963
981
964
- // pass 2 - dump the field descriptors
982
+ // dumps the definition of the instance fields for a given class
983
+ void DumperSupport::dump_instance_field_descriptors (DumpWriter* writer, Klass* k) {
984
+ HandleMark hm;
985
+ InstanceKlass* ik = InstanceKlass::cast (k);
986
+
987
+ // dump the field descriptors
965
988
for (FieldStream fld (ik, true , true ); !fld.eos (); fld.next ()) {
966
989
if (!fld.access_flags ().is_static ()) {
967
990
Symbol* sig = fld.signature ();
@@ -974,20 +997,24 @@ void DumperSupport::dump_instance_field_descriptors(DumpWriter* writer, Klass* k
974
997
975
998
// creates HPROF_GC_INSTANCE_DUMP record for the given object
976
999
void DumperSupport::dump_instance (DumpWriter* writer, oop o) {
977
- Klass* k = o->klass ();
1000
+ InstanceKlass* ik = InstanceKlass::cast (o->klass ());
1001
+ u4 is = instance_size (ik);
1002
+ u4 size = 1 + sizeof (address) + 4 + sizeof (address) + 4 + is;
978
1003
979
- writer->write_u1 (HPROF_GC_INSTANCE_DUMP);
1004
+ writer->start_sub_record (HPROF_GC_INSTANCE_DUMP, size );
980
1005
writer->write_objectID (o);
981
1006
writer->write_u4 (STACK_TRACE_ID);
982
1007
983
1008
// class ID
984
- writer->write_classID (k );
1009
+ writer->write_classID (ik );
985
1010
986
1011
// number of bytes that follow
987
- writer->write_u4 (instance_size (k) );
1012
+ writer->write_u4 (is );
988
1013
989
1014
// field values
990
1015
dump_instance_fields (writer, o);
1016
+
1017
+ writer->end_sub_record ();
991
1018
}
992
1019
993
1020
// creates HPROF_GC_CLASS_DUMP record for the given class and each of
@@ -1002,7 +1029,13 @@ void DumperSupport::dump_class_and_array_classes(DumpWriter* writer, Klass* k) {
1002
1029
return ;
1003
1030
}
1004
1031
1005
- writer->write_u1 (HPROF_GC_CLASS_DUMP);
1032
+ u2 static_fields_count = 0 ;
1033
+ u4 static_size = get_static_fields_size (ik, static_fields_count);
1034
+ u2 instance_fields_count = get_instance_fields_count (ik);
1035
+ u4 instance_fields_size = instance_fields_count * (sizeof (address) + 1 );
1036
+ u4 size = 1 + sizeof (address) + 4 + 6 * sizeof (address) + 4 + 2 + 2 + static_size + 2 + instance_fields_size;
1037
+
1038
+ writer->start_sub_record (HPROF_GC_CLASS_DUMP, size);
1006
1039
1007
1040
// class ID
1008
1041
writer->write_classID (ik);
@@ -1025,29 +1058,33 @@ void DumperSupport::dump_class_and_array_classes(DumpWriter* writer, Klass* k) {
1025
1058
writer->write_objectID (oop (NULL ));
1026
1059
1027
1060
// instance size
1028
- writer->write_u4 (DumperSupport::instance_size (k ));
1061
+ writer->write_u4 (DumperSupport::instance_size (ik ));
1029
1062
1030
1063
// size of constant pool - ignored by HAT 1.1
1031
1064
writer->write_u2 (0 );
1032
1065
1033
- // number of static fields
1034
- dump_static_fields (writer, k);
1066
+ // static fields
1067
+ writer->write_u2 (static_fields_count);
1068
+ dump_static_fields (writer, ik);
1035
1069
1036
1070
// description of instance fields
1037
- dump_instance_field_descriptors (writer, k);
1071
+ writer->write_u2 (instance_fields_count);
1072
+ dump_instance_field_descriptors (writer, ik);
1073
+
1074
+ writer->end_sub_record ();
1038
1075
1039
1076
// array classes
1040
- k = k ->array_klass_or_null ();
1077
+ k = ik ->array_klass_or_null ();
1041
1078
while (k != NULL ) {
1042
- Klass* klass = k;
1043
- assert (klass->is_objArray_klass (), " not an ObjArrayKlass" );
1079
+ assert (k->is_objArray_klass (), " not an ObjArrayKlass" );
1044
1080
1045
- writer->write_u1 (HPROF_GC_CLASS_DUMP);
1046
- writer->write_classID (klass);
1081
+ u4 size = 1 + sizeof (address) + 4 + 6 * sizeof (address) + 4 + 2 + 2 + 2 ;
1082
+ writer->start_sub_record (HPROF_GC_CLASS_DUMP, size);
1083
+ writer->write_classID (k);
1047
1084
writer->write_u4 (STACK_TRACE_ID);
1048
1085
1049
1086
// super class of array classes is java.lang.Object
1050
- java_super = klass ->java_super ();
1087
+ java_super = k ->java_super ();
1051
1088
assert (java_super != NULL , " checking" );
1052
1089
writer->write_classID (java_super);
1053
1090
@@ -1062,8 +1099,10 @@ void DumperSupport::dump_class_and_array_classes(DumpWriter* writer, Klass* k) {
1062
1099
writer->write_u2 (0 ); // static fields
1063
1100
writer->write_u2 (0 ); // instance fields
1064
1101
1102
+ writer->end_sub_record ();
1103
+
1065
1104
// get the array class for the next rank
1066
- k = klass ->array_klass_or_null ();
1105
+ k = k ->array_klass_or_null ();
1067
1106
}
1068
1107
}
1069
1108
@@ -1074,7 +1113,8 @@ void DumperSupport::dump_basic_type_array_class(DumpWriter* writer, Klass* k) {
1074
1113
while (k != NULL ) {
1075
1114
Klass* klass = k;
1076
1115
1077
- writer->write_u1 (HPROF_GC_CLASS_DUMP);
1116
+ u4 size = 1 + sizeof (address) + 4 + 6 * sizeof (address) + 4 + 2 + 2 + 2 ;
1117
+ writer->start_sub_record (HPROF_GC_CLASS_DUMP, size);
1078
1118
writer->write_classID (klass);
1079
1119
writer->write_u4 (STACK_TRACE_ID);
1080
1120
@@ -1094,6 +1134,8 @@ void DumperSupport::dump_basic_type_array_class(DumpWriter* writer, Klass* k) {
1094
1134
writer->write_u2 (0 ); // static fields
1095
1135
writer->write_u2 (0 ); // instance fields
1096
1136
1137
+ writer->end_sub_record ();
1138
+
1097
1139
// get the array class for the next rank
1098
1140
k = klass->array_klass_or_null ();
1099
1141
}
@@ -1115,23 +1157,8 @@ int DumperSupport::calculate_array_max_length(DumpWriter* writer, arrayOop array
1115
1157
}
1116
1158
1117
1159
size_t length_in_bytes = (size_t )length * type_size;
1160
+ uint max_bytes = max_juint - header_size;
1118
1161
1119
- // Create a new record if the current record is non-empty and the array can't fit.
1120
- julong current_record_length = writer->current_record_length ();
1121
- if (current_record_length > 0 &&
1122
- (current_record_length + header_size + length_in_bytes) > max_juint) {
1123
- write_current_dump_record_length (writer);
1124
- write_dump_header (writer);
1125
-
1126
- // We now have an empty record.
1127
- current_record_length = 0 ;
1128
- }
1129
-
1130
- // Calculate max bytes we can use.
1131
- uint max_bytes = max_juint - (header_size + current_record_length);
1132
-
1133
- // Array too long for the record?
1134
- // Calculate max length and return it.
1135
1162
if (length_in_bytes > max_bytes) {
1136
1163
length = max_bytes / type_size;
1137
1164
length_in_bytes = (size_t )length * type_size;
@@ -1146,10 +1173,10 @@ int DumperSupport::calculate_array_max_length(DumpWriter* writer, arrayOop array
1146
1173
void DumperSupport::dump_object_array (DumpWriter* writer, objArrayOop array) {
1147
1174
// sizeof(u1) + 2 * sizeof(u4) + sizeof(objectID) + sizeof(classID)
1148
1175
short header_size = 1 + 2 * 4 + 2 * sizeof (address);
1149
-
1150
1176
int length = calculate_array_max_length (writer, array, header_size);
1177
+ u4 size = header_size + length * sizeof (address);
1151
1178
1152
- writer->write_u1 (HPROF_GC_OBJ_ARRAY_DUMP);
1179
+ writer->start_sub_record (HPROF_GC_OBJ_ARRAY_DUMP, size );
1153
1180
writer->write_objectID (array);
1154
1181
writer->write_u4 (STACK_TRACE_ID);
1155
1182
writer->write_u4 (length);
@@ -1169,6 +1196,8 @@ void DumperSupport::dump_object_array(DumpWriter* writer, objArrayOop array) {
1169
1196
o = mask_dormant_archived_object (o);
1170
1197
writer->write_objectID (o);
1171
1198
}
1199
+
1200
+ writer->end_sub_record ();
1172
1201
}
1173
1202
1174
1203
#define WRITE_ARRAY (Array, Type, Size, Length ) \
@@ -1184,15 +1213,17 @@ void DumperSupport::dump_prim_array(DumpWriter* writer, typeArrayOop array) {
1184
1213
int length = calculate_array_max_length (writer, array, header_size);
1185
1214
int type_size = type2aelembytes (type);
1186
1215
u4 length_in_bytes = (u4)length * type_size;
1216
+ u4 size = header_size + length_in_bytes;
1187
1217
1188
- writer->write_u1 (HPROF_GC_PRIM_ARRAY_DUMP);
1218
+ writer->start_sub_record (HPROF_GC_PRIM_ARRAY_DUMP, size );
1189
1219
writer->write_objectID (array);
1190
1220
writer->write_u4 (STACK_TRACE_ID);
1191
1221
writer->write_u4 (length);
1192
1222
writer->write_u1 (type2tag (type));
1193
1223
1194
1224
// nothing to copy
1195
1225
if (length == 0 ) {
1226
+ writer->end_sub_record ();
1196
1227
return ;
1197
1228
}
1198
1229
@@ -1262,6 +1293,8 @@ void DumperSupport::dump_prim_array(DumpWriter* writer, typeArrayOop array) {
1262
1293
}
1263
1294
default : ShouldNotReachHere ();
1264
1295
}
1296
+
1297
+ writer->end_sub_record ();
1265
1298
}
1266
1299
1267
1300
// create a HPROF_FRAME record of the given Method* and bci
@@ -1337,10 +1370,12 @@ void JNILocalsDumper::do_oop(oop* obj_p) {
1337
1370
// ignore null handles
1338
1371
oop o = *obj_p;
1339
1372
if (o != NULL ) {
1340
- writer ()->write_u1 (HPROF_GC_ROOT_JNI_LOCAL);
1373
+ u4 size = 1 + sizeof (address) + 4 + 4 ;
1374
+ writer ()->start_sub_record (HPROF_GC_ROOT_JNI_LOCAL, size);
1341
1375
writer ()->write_objectID (o);
1342
1376
writer ()->write_u4 (_thread_serial_num);
1343
1377
writer ()->write_u4 ((u4)_frame_num);
1378
+ writer ()->end_sub_record ();
1344
1379
}
1345
1380
}
1346
1381
@@ -1368,9 +1403,11 @@ void JNIGlobalsDumper::do_oop(oop* obj_p) {
1368
1403
1369
1404
// we ignore global ref to symbols and other internal objects
1370
1405
if (o->is_instance () || o->is_objArray () || o->is_typeArray ()) {
1371
- writer ()->write_u1 (HPROF_GC_ROOT_JNI_GLOBAL);
1406
+ u4 size = 1 + 2 * sizeof (address);
1407
+ writer ()->start_sub_record (HPROF_GC_ROOT_JNI_GLOBAL, size);
1372
1408
writer ()->write_objectID (o);
1373
1409
writer ()->write_objectID ((oopDesc*)obj_p); // global ref ID
1410
+ writer ()->end_sub_record ();
1374
1411
}
1375
1412
};
1376
1413
@@ -1386,8 +1423,10 @@ class MonitorUsedDumper : public OopClosure {
1386
1423
_writer = writer;
1387
1424
}
1388
1425
void do_oop (oop* obj_p) {
1389
- writer ()->write_u1 (HPROF_GC_ROOT_MONITOR_USED);
1426
+ u4 size = 1 + sizeof (address);
1427
+ writer ()->start_sub_record (HPROF_GC_ROOT_MONITOR_USED, size);
1390
1428
writer ()->write_objectID (*obj_p);
1429
+ writer ()->end_sub_record ();
1391
1430
}
1392
1431
void do_oop (narrowOop* obj_p) { ShouldNotReachHere (); }
1393
1432
};
@@ -1406,10 +1445,12 @@ class StickyClassDumper : public KlassClosure {
1406
1445
void do_klass (Klass* k) {
1407
1446
if (k->is_instance_klass ()) {
1408
1447
InstanceKlass* ik = InstanceKlass::cast (k);
1409
- writer ()->write_u1 (HPROF_GC_ROOT_STICKY_CLASS);
1410
- writer ()->write_classID (ik);
1411
- }
1448
+ u4 size = 1 + sizeof (address);
1449
+ writer ()->start_sub_record (HPROF_GC_ROOT_STICKY_CLASS, size);
1450
+ writer ()->write_classID (ik);
1451
+ writer ()->end_sub_record ();
1412
1452
}
1453
+ }
1413
1454
};
1414
1455
1415
1456
@@ -1425,9 +1466,6 @@ class HeapObjectDumper : public ObjectClosure {
1425
1466
VM_HeapDumper* dumper () { return _dumper; }
1426
1467
DumpWriter* writer () { return _writer; }
1427
1468
1428
- // used to indicate that a record has been writen
1429
- void mark_end_of_record ();
1430
-
1431
1469
public:
1432
1470
HeapObjectDumper (VM_HeapDumper* dumper, DumpWriter* writer) {
1433
1471
_dumper = dumper;
@@ -1454,15 +1492,12 @@ void HeapObjectDumper::do_object(oop o) {
1454
1492
if (o->is_instance ()) {
1455
1493
// create a HPROF_GC_INSTANCE record for each object
1456
1494
DumperSupport::dump_instance (writer (), o);
1457
- mark_end_of_record ();
1458
1495
} else if (o->is_objArray ()) {
1459
1496
// create a HPROF_GC_OBJ_ARRAY_DUMP record for each object array
1460
1497
DumperSupport::dump_object_array (writer (), objArrayOop (o));
1461
- mark_end_of_record ();
1462
1498
} else if (o->is_typeArray ()) {
1463
1499
// create a HPROF_GC_PRIM_ARRAY_DUMP record for each type array
1464
1500
DumperSupport::dump_prim_array (writer (), typeArrayOop (o));
1465
- mark_end_of_record ();
1466
1501
}
1467
1502
}
1468
1503
@@ -1553,8 +1588,6 @@ class VM_HeapDumper : public VM_GC_Operation {
1553
1588
}
1554
1589
1555
1590
VMOp_Type type () const { return VMOp_HeapDumper; }
1556
- // used to mark sub-record boundary
1557
- void check_segment_length ();
1558
1591
void doit ();
1559
1592
};
1560
1593
@@ -1565,72 +1598,13 @@ bool VM_HeapDumper::skip_operation() const {
1565
1598
return false ;
1566
1599
}
1567
1600
1568
- // writes a HPROF_HEAP_DUMP_SEGMENT record
1569
- void DumperSupport::write_dump_header (DumpWriter* writer) {
1570
- if (writer->is_open ()) {
1571
- writer->write_u1 (HPROF_HEAP_DUMP_SEGMENT);
1572
- writer->write_u4 (0 ); // current ticks
1573
-
1574
- // record the starting position for the dump (its length will be fixed up later)
1575
- writer->set_dump_start (writer->current_offset ());
1576
- writer->write_u4 (0 );
1577
- }
1578
- }
1579
-
1580
- // fixes up the length of the current dump record
1581
- void DumperSupport::write_current_dump_record_length (DumpWriter* writer) {
1582
- if (writer->is_open ()) {
1583
- julong dump_end = writer->bytes_written () + writer->bytes_unwritten ();
1584
- julong dump_len = writer->current_record_length ();
1585
-
1586
- // record length must fit in a u4
1587
- if (dump_len > max_juint) {
1588
- warning (" record is too large" );
1589
- }
1590
-
1591
- // seek to the dump start and fix-up the length
1592
- assert (writer->dump_start () >= 0 , " no dump start recorded" );
1593
- writer->seek_to_offset (writer->dump_start ());
1594
- writer->write_u4 ((u4)dump_len);
1595
-
1596
- // adjust the total size written to keep the bytes written correct.
1597
- writer->adjust_bytes_written (-((jlong) sizeof (u4)));
1598
-
1599
- // seek to dump end so we can continue
1600
- writer->seek_to_offset (dump_end);
1601
-
1602
- // no current dump record
1603
- writer->set_dump_start ((jlong)-1 );
1604
- }
1605
- }
1606
-
1607
- // used on a sub-record boundary to check if we need to start a
1608
- // new segment.
1609
- void VM_HeapDumper::check_segment_length () {
1610
- if (writer ()->is_open ()) {
1611
- julong dump_len = writer ()->current_record_length ();
1612
-
1613
- if (dump_len > 2UL *G) {
1614
- DumperSupport::write_current_dump_record_length (writer ());
1615
- DumperSupport::write_dump_header (writer ());
1616
- }
1617
- }
1618
- }
1619
-
1620
1601
// fixes up the current dump record and writes HPROF_HEAP_DUMP_END record
1621
1602
void DumperSupport::end_of_dump (DumpWriter* writer) {
1622
- if (writer->is_open ()) {
1623
- write_current_dump_record_length (writer);
1603
+ writer->finish_dump_segment ();
1624
1604
1625
- writer->write_u1 (HPROF_HEAP_DUMP_END);
1626
- writer->write_u4 (0 );
1627
- writer->write_u4 (0 );
1628
- }
1629
- }
1630
-
1631
- // marks sub-record boundary
1632
- void HeapObjectDumper::mark_end_of_record () {
1633
- dumper ()->check_segment_length ();
1605
+ writer->write_u1 (HPROF_HEAP_DUMP_END);
1606
+ writer->write_u4 (0 );
1607
+ writer->write_u4 (0 );
1634
1608
}
1635
1609
1636
1610
// writes a HPROF_LOAD_CLASS record for the class (and each of its
@@ -1720,10 +1694,12 @@ int VM_HeapDumper::do_thread(JavaThread* java_thread, u4 thread_serial_num) {
1720
1694
oop o = locals->obj_at (slot)();
1721
1695
1722
1696
if (o != NULL ) {
1723
- writer ()->write_u1 (HPROF_GC_ROOT_JAVA_FRAME);
1697
+ u4 size = 1 + sizeof (address) + 4 + 4 ;
1698
+ writer ()->start_sub_record (HPROF_GC_ROOT_JAVA_FRAME, size);
1724
1699
writer ()->write_objectID (o);
1725
1700
writer ()->write_u4 (thread_serial_num);
1726
1701
writer ()->write_u4 ((u4) (stack_depth + extra_frames));
1702
+ writer ()->end_sub_record ();
1727
1703
}
1728
1704
}
1729
1705
}
@@ -1732,10 +1708,12 @@ int VM_HeapDumper::do_thread(JavaThread* java_thread, u4 thread_serial_num) {
1732
1708
if (exprs->at (index )->type () == T_OBJECT) {
1733
1709
oop o = exprs->obj_at (index )();
1734
1710
if (o != NULL ) {
1735
- writer ()->write_u1 (HPROF_GC_ROOT_JAVA_FRAME);
1711
+ u4 size = 1 + sizeof (address) + 4 + 4 ;
1712
+ writer ()->start_sub_record (HPROF_GC_ROOT_JAVA_FRAME, size);
1736
1713
writer ()->write_objectID (o);
1737
1714
writer ()->write_u4 (thread_serial_num);
1738
1715
writer ()->write_u4 ((u4) (stack_depth + extra_frames));
1716
+ writer ()->end_sub_record ();
1739
1717
}
1740
1718
}
1741
1719
}
@@ -1783,10 +1761,12 @@ void VM_HeapDumper::do_threads() {
1783
1761
oop threadObj = thread->threadObj ();
1784
1762
u4 thread_serial_num = i+1 ;
1785
1763
u4 stack_serial_num = thread_serial_num + STACK_TRACE_ID;
1786
- writer ()->write_u1 (HPROF_GC_ROOT_THREAD_OBJ);
1764
+ u4 size = 1 + sizeof (address) + 4 + 4 ;
1765
+ writer ()->start_sub_record (HPROF_GC_ROOT_THREAD_OBJ, size);
1787
1766
writer ()->write_objectID (threadObj);
1788
1767
writer ()->write_u4 (thread_serial_num); // thread number
1789
1768
writer ()->write_u4 (stack_serial_num); // stack trace serial number
1769
+ writer ()->end_sub_record ();
1790
1770
int num_frames = do_thread (thread, thread_serial_num);
1791
1771
assert (num_frames == _stack_traces[i]->get_stack_depth (),
1792
1772
" total number of Java frames not matched" );
@@ -1864,16 +1844,12 @@ void VM_HeapDumper::doit() {
1864
1844
// this must be called after _klass_map is built when iterating the classes above.
1865
1845
dump_stack_traces ();
1866
1846
1867
- // write HPROF_HEAP_DUMP_SEGMENT
1868
- DumperSupport::write_dump_header (writer ());
1869
-
1870
1847
// Writes HPROF_GC_CLASS_DUMP records
1871
1848
{
1872
1849
LockedClassesDo locked_dump_class (&do_class_dump);
1873
1850
ClassLoaderDataGraph::classes_do (&locked_dump_class);
1874
1851
}
1875
1852
Universe::basic_type_classes_do (&do_basic_type_array_class_dump);
1876
- check_segment_length ();
1877
1853
1878
1854
// writes HPROF_GC_INSTANCE_DUMP records.
1879
1855
// After each sub-record is written check_segment_length will be invoked
@@ -1886,27 +1862,24 @@ void VM_HeapDumper::doit() {
1886
1862
1887
1863
// HPROF_GC_ROOT_THREAD_OBJ + frames + jni locals
1888
1864
do_threads ();
1889
- check_segment_length ();
1890
1865
1891
1866
// HPROF_GC_ROOT_MONITOR_USED
1892
1867
MonitorUsedDumper mon_dumper (writer ());
1893
1868
ObjectSynchronizer::oops_do (&mon_dumper);
1894
- check_segment_length ();
1895
1869
1896
1870
// HPROF_GC_ROOT_JNI_GLOBAL
1897
1871
JNIGlobalsDumper jni_dumper (writer ());
1898
1872
JNIHandles::oops_do (&jni_dumper);
1899
1873
Universe::oops_do (&jni_dumper); // technically not jni roots, but global roots
1900
1874
// for things like preallocated throwable backtraces
1901
- check_segment_length ();
1902
1875
1903
1876
// HPROF_GC_ROOT_STICKY_CLASS
1904
1877
// These should be classes in the NULL class loader data, and not all classes
1905
1878
// if !ClassUnloading
1906
1879
StickyClassDumper class_dumper (writer ());
1907
1880
ClassLoaderData::the_null_class_loader_data ()->classes_do (&class_dumper);
1908
1881
1909
- // fixes up the length of the dump record and writes the HPROF_HEAP_DUMP_END record.
1882
+ // Writes the HPROF_HEAP_DUMP_END record.
1910
1883
DumperSupport::end_of_dump (writer ());
1911
1884
1912
1885
// Now we clear the global variables, so that a future dumper might run.
@@ -1980,7 +1953,7 @@ int HeapDumper::dump(const char* path, outputStream* out) {
1980
1953
1981
1954
// create the dump writer. If the file can be opened then bail
1982
1955
DumpWriter writer (path);
1983
- if (! writer.is_open () ) {
1956
+ if (writer.error () != NULL ) {
1984
1957
set_error (writer.error ());
1985
1958
if (out != NULL ) {
1986
1959
out->print_cr (" Unable to create %s: %s" , path,
0 commit comments