@@ -126,6 +126,9 @@ static struct {
126
126
127
127
VALUE fake_frame_names [TOTAL_FAKE_FRAMES ];
128
128
VALUE empty_string ;
129
+
130
+ int buffer_count ;
131
+ sample_time_t buffer_time ;
129
132
VALUE frames_buffer [BUF_SIZE ];
130
133
int lines_buffer [BUF_SIZE ];
131
134
} _stackprof ;
@@ -594,9 +597,17 @@ stackprof_record_sample_for_stack(int num, uint64_t sample_timestamp, int64_t ti
594
597
}
595
598
}
596
599
600
+ // buffer the current profile frames
601
+ // This must be async-signal-safe
602
+ // Returns immediately if another set of frames are already in the buffer
597
603
void
598
- stackprof_record_sample ( )
604
+ stackprof_buffer_sample ( void )
599
605
{
606
+ if (_stackprof .buffer_count > 0 ) {
607
+ // Another sample is already pending
608
+ return ;
609
+ }
610
+
600
611
uint64_t start_timestamp = 0 ;
601
612
int64_t timestamp_delta = 0 ;
602
613
int num ;
@@ -606,12 +617,16 @@ stackprof_record_sample()
606
617
start_timestamp = timestamp_usec (& t );
607
618
timestamp_delta = delta_usec (& t , & _stackprof .last_sample_at );
608
619
}
620
+
609
621
num = rb_profile_frames (0 , sizeof (_stackprof .frames_buffer ) / sizeof (VALUE ), _stackprof .frames_buffer , _stackprof .lines_buffer );
610
- stackprof_record_sample_for_stack (num , start_timestamp , timestamp_delta );
622
+
623
+ _stackprof .buffer_count = num ;
624
+ _stackprof .buffer_time .timestamp_usec = start_timestamp ;
625
+ _stackprof .buffer_time .delta_usec = timestamp_delta ;
611
626
}
612
627
613
628
void
614
- stackprof_record_gc_samples ()
629
+ stackprof_record_gc_samples (void )
615
630
{
616
631
int64_t delta_to_first_unrecorded_gc_sample = 0 ;
617
632
uint64_t start_timestamp = 0 ;
@@ -661,20 +676,47 @@ stackprof_record_gc_samples()
661
676
_stackprof .unrecorded_gc_sweeping_samples = 0 ;
662
677
}
663
678
679
+ // record the sample previously buffered by stackprof_buffer_sample
664
680
static void
665
- stackprof_gc_job_handler (void * data )
681
+ stackprof_record_buffer (void )
682
+ {
683
+ stackprof_record_sample_for_stack (_stackprof .buffer_count , _stackprof .buffer_time .timestamp_usec , _stackprof .buffer_time .delta_usec );
684
+
685
+ // reset the buffer
686
+ _stackprof .buffer_count = 0 ;
687
+ }
688
+
689
+ static void
690
+ stackprof_sample_and_record (void )
691
+ {
692
+ stackprof_buffer_sample ();
693
+ stackprof_record_buffer ();
694
+ }
695
+
696
+ static void
697
+ stackprof_job_record_gc (void * data )
666
698
{
667
699
if (!_stackprof .running ) return ;
668
700
669
701
stackprof_record_gc_samples ();
670
702
}
671
703
704
+ #ifdef USE_POSTPONED_JOB
705
+ static void
706
+ stackprof_job_sample_and_record (void * data )
707
+ {
708
+ if (!_stackprof .running ) return ;
709
+
710
+ stackprof_sample_and_record ();
711
+ }
712
+ #endif
713
+
672
714
static void
673
- stackprof_job_handler (void * data )
715
+ stackprof_job_record_buffer (void * data )
674
716
{
675
717
if (!_stackprof .running ) return ;
676
718
677
- stackprof_record_sample ();
719
+ stackprof_record_buffer ();
678
720
}
679
721
680
722
static void
@@ -696,12 +738,15 @@ stackprof_signal_handler(int sig, siginfo_t *sinfo, void *ucontext)
696
738
_stackprof .unrecorded_gc_sweeping_samples ++ ;
697
739
}
698
740
_stackprof .unrecorded_gc_samples ++ ;
699
- rb_postponed_job_register_one (0 , stackprof_gc_job_handler , (void * )0 );
741
+ rb_postponed_job_register_one (0 , stackprof_job_record_gc , (void * )0 );
700
742
} else {
701
743
#ifdef USE_POSTPONED_JOB
702
- rb_postponed_job_register_one (0 , stackprof_job_handler , (void * )0 );
744
+ rb_postponed_job_register_one (0 , stackprof_job_sample_and_record , (void * )0 );
703
745
#else
704
- stackprof_job_handler (0 );
746
+ // Buffer a sample immediately, this may overwrite the existing sample
747
+ stackprof_buffer_sample ();
748
+ // Enqueue a job to record the sample
749
+ rb_postponed_job_register_one (0 , stackprof_job_record_buffer , (void * )0 );
705
750
#endif
706
751
}
707
752
pthread_mutex_unlock (& lock );
@@ -713,7 +758,7 @@ stackprof_newobj_handler(VALUE tpval, void *data)
713
758
_stackprof .overall_signals ++ ;
714
759
if (RTEST (_stackprof .interval ) && _stackprof .overall_signals % NUM2LONG (_stackprof .interval ))
715
760
return ;
716
- stackprof_job_handler ( 0 );
761
+ stackprof_sample_and_record ( );
717
762
}
718
763
719
764
static VALUE
@@ -723,7 +768,7 @@ stackprof_sample(VALUE self)
723
768
return Qfalse ;
724
769
725
770
_stackprof .overall_signals ++ ;
726
- stackprof_job_handler ( 0 );
771
+ stackprof_sample_and_record ( );
727
772
return Qtrue ;
728
773
}
729
774
0 commit comments