RFR: 8138920: Refactor the sampling thread from ConcurrentG1RefineThread
Derek White
derek.white at oracle.com
Wed Oct 21 17:03:43 UTC 2015
Hi Per,
On 10/21/15 6:42 AM, Per Liden wrote:
> Hi Derek,
>
> On 2015-10-20 21:56, Derek White wrote:
>> 4th Webrev for review please:
>>
>> This version includes the cleanups suggested by Per.
>>
>> RFE: JDK-8138920 <https://bugs.openjdk.java.net/browse/JDK-8138920>
>> Refactor the sampling thread from ConcurrentG1RefineThread
>>
>> Webrev: http://cr.openjdk.java.net/~drwhite/8138920/webrev.04/
>
>
> Looks good over all. Just a few minor things I noticed.
>
> concurrentG1Refine.cpp
> ----------------------
>
> 92 vm_shutdown_during_initialization("Could not create
> ConcurrentSampleThread");
>
> Should be "G1YoungListSampleThread" now.
OK.
>
> 160 void ConcurrentG1Refine::print_worker_threads_on(outputStream*
> st) const {
> 161 for (uint i = 0; i < _n_worker_threads; ++i) {
> 162 _threads[i]->print_on(st);
> 163 st->cr();
> 164 }
> 165 _sample_thread->print_on(st);
> 166 st->cr();
>
> You didn't introduce this, but shouldn't there be a if (_threads !=
> NULL) here around this, similar to the other functions which access
> _threads?
Deleted tests as suggested by Thomas.
> concurrentG1RefineThread.cpp
> ----------------------------
>
> 199 void ConcurrentG1RefineThread::stop_service() {
> 200 {
> 201 MutexLockerEx x(_monitor, Mutex::_no_safepoint_check_flag);
> 202 _monitor->notify();
> 203 }
> 204 }
>
> The extra scope here seems superfluous.
OK.
> concurrentG1RefineThread.hpp
> ----------------------------
>
> 71 virtual void run_service();
> 72 virtual void stop_service();
>
> Please make these non-virtual.
OK. I'll revirtualize in 8140257
<https://bugs.openjdk.java.net/browse/JDK-8140257>.
>
>
> g1YoungListSampleThread.hpp
> ---------------------------
>
> 30 class G1YoungListSampleThread: public ConcurrentGCThread {
> 31 Monitor* _monitor;
>
> Please add a "private:" here.
OK. BTW, I noticed pretty inconsistent indentation of "private:" in the
G1 code, especially right after the class name. Some cases are zero
indent and some are one (1!). I'm assuming that zero is correct.
> g1YoungListSampleThread.cpp
> ---------------------------
>
> 59 _monitor = new Monitor(Mutex::nonleaf,
> 60 "Sample thread monitor",
> 61 true,
> 62 Monitor::_safepoint_check_never);
>
> It looks like _monitor could be a value member instead of a pointer,
> to avoid the extra allocation here.
>
> 66 set_name("G1 Sample");
>
> This should probably be updated to reflect the new name.
OK.
>
> cheers,
> /Per
>
Thanks Per!
- Derek
>>
>> jprt: waiting for results (which might take a long time given current
>> jprt status).
>>
>> Thanks for looking!
>>
>> - Derek
>>
>> On 10/20/15 1:41 PM, Derek White wrote:
>>> Thanks for the review Per! Comments below...
>>>
>>> On 10/20/15 2:56 AM, Per Liden wrote:
>>>> Hi Derek,
>>>>
>>>> On 2015-10-19 22:08, Derek White wrote:
>>>>> Thanks Bengt (and Per indirectly). Comments below.
>>>>>
>>>>> On 10/19/15 3:17 PM, Bengt Rutisson wrote:
>>>>>> Hi Derek,
>>>>>>
>>>>>> On 2015-10-17 03:32, Derek White wrote:
>>>>>>> 3rd Webrev for review please:
>>>>>>>
>>>>>>> This version does away with the common abstract base case
>>>>>>> ConcurrentServiceThread, and pushes the code down to the concrete
>>>>>>> classes. This may get cleaned up in a later fix to
>>>>>>> ConcurrentGCThread.
>>>>>>>
>>>>>>>
>>>>>>> RFE: JDK-8138920 <https://bugs.openjdk.java.net/browse/JDK-8138920>
>>>>>>> Refactor the sampling thread from ConcurrentG1RefineThread
>>>>>>>
>>>>>>> Webrev: http://cr.openjdk.java.net/~drwhite/8138920/webrev.03/
>>>>>>
>>>>>> I think this version is much good.
>>>>>>
>>>>>> I talked a bit to Per about the latest webrev and he pointed out
>>>>>> that
>>>>>> we don't really need the run_service()/stop_service() abstraction
>>>>>> now
>>>>>> that the threads implement all this themselves. The change would be
>>>>>> smaller if we left that change out.
>>>>>>
>>>>>> I'll leave it up to you if you want to keep it in there or not.
>>>>>> Do you
>>>>>> plan on following this work up by starting to work on JDK-8138920
>>>>>> pretty soon? In that case I think the run_service()/stop_service()
>>>>>> abstraction will have to be handled there anyway which makes it a
>>>>>> smaller issue now in my opinion.
>>>>> I went back and forth about including
>>>>> run_service()/stop_service(), but
>>>>> decided that the refactoring had already been done, and I'd like to
>>>>> push
>>>>> run()/stop() up to ConcurrentGCThread like we talked about soon.
>>>>> There
>>>>> are a few extra complications that will also need to be handled (like
>>>>> G1StringDedupThread already has a static stop() method).
>>>>
>>>> I'm ok with keeping it like this if the ConcurrentGCThread cleanup is
>>>> something we'll get to soon-ish.
>>>>
>>>> concurrentSampleThread.hpp/cpp
>>>> ------------------------------
>>>>
>>>> * I'd suggest we name the new thread it G1SampleThread (or maybe
>>>> G1YoungListSampleThread, or something similar). The fact that it's
>>>> concurrent doesn't seem important enough to make it part of the name,
>>>> better to try to say something about what it does.
>>> OK.
>>>
>>>> * sleepBeforeNextCycle() should be sleep_before_next_cycle(). I know
>>>> concurrentMarkThread and maybe some other thread also has a
>>>> sleepBeforeNextCycle() and I think that should be changed too at some
>>>> point (not saying you need to do that in this change).
>>>>
>>> OK
>>>> * The protected members should be private, i.e. :
>>>>
>>>> 30 class ConcurrentSampleThread: public ConcurrentGCThread {
>>>> 31 protected:
>>>> 32 Monitor* _monitor;
>>>> 33
>>>> 34 void sample_young_list_rs_lengths();
>>>> 35
>>>> 36 void run_service();
>>>> 37 void stop_service();
>>>> 38
>>>> 39 void sleepBeforeNextCycle();
>>>> 40
>>>> 41 double _vtime_accum; // Accumulated virtual time.
>>> OK
>>>> * I could be wrong here, but I can see any immediate need to include
>>>> these, or?
>>>>
>>>> 30 #include "memory/resourceArea.hpp"
>>>> 31 #include "runtime/handles.inline.hpp"
>>>>
>>> Not likely. I'll Check.
>>>
>>>> concurrentG1Refine.hpp/cpp
>>>> --------------------------
>>>>
>>>> * As Thomas mentioned, I think the sample thread ownership should
>>>> move from the ConcurrentG1Refine to to G1CollectedHeap. That seems to
>>>> make more sense, no?
>>>
>>> Yes, but I'd like to leave that as a separate fix. One that actually
>>> allows concurrent refinement to be disabled. It would take me a bit to
>>> figure out how to that correctly (and test).
>>>
>>> Thanks again, new webrev in a bit...
>>>
>>> - Derek
>>>>
>>>> cheers,
>>>> /Per
>>>>
>>>>>
>>>>> - Derek
>>>>>>
>>>>>> Thanks,
>>>>>> Bengt
>>>>>>
>>>>>>>
>>>>>>> Passed jprt (finally!).
>>>>>>>
>>>>>>> Thanks for looking!
>>>>>>>
>>>>>>> - Derek
>>>>>>>
>>>>>>> On 10/9/15 12:01 PM, Bengt Rutisson wrote:
>>>>>>>>
>>>>>>>> Hi Derek,
>>>>>>>>
>>>>>>>> Comments inlined.
>>>>>>>>
>>>>>>>> On 2015-10-09 00:29, Derek White wrote:
>>>>>>>>> Another call for review:
>>>>>>>>>
>>>>>>>>> 2nd webrev:
>>>>>>>>> http://cr.openjdk.java.net/~drwhite/8138920/webrev.02/
>>>>>>>>>
>>>>>>>>> See changes and comments below:
>>>>>>>>>
>>>>>>>>> On 10/8/15 2:47 AM, Bengt Rutisson wrote:
>>>>>>>>>>
>>>>>>>>>> Hi Derek,
>>>>>>>>>>
>>>>>>>>>> On 2015-10-07 17:19, Derek White wrote:
>>>>>>>>>>> Hi Bengt,
>>>>>>>>>>>
>>>>>>>>>>> On 10/7/15 4:02 AM, Bengt Rutisson wrote:
>>>>>>>>>>>> Hi Derek,
>>>>>>>>>>>>
>>>>>>>>>>>> Thanks for fixing this!
>>>>>>>>>>>>
>>>>>>>>>>>> On 2015-10-06 19:51, Derek White wrote:
>>>>>>>>>>>>> Refactor and cleanup the G1 concurrent thread classes:
>>>>>>>>>>>>> - Pull out a sampling thread class (now
>>>>>>>>>>>>> ConcurrentG1SampleThread) from ConcurrentG1RefineThread.
>>>>>>>>>>>>> - Factor out an abstract base class
>>>>>>>>>>>>> ConcurrentG1ServiceThread
>>>>>>>>>>>>> that is used by:
>>>>>>>>>>>>> - ConcurrentG1RefineThread
>>>>>>>>>>>>> - ConcurrentG1SampleThread
>>>>>>>>>>>>> - ConcurrentMarkThread
>>>>>>>>>>>>> - Made the handling of the "primary" refinement thread more
>>>>>>>>>>>>> explicit.
>>>>>>>>>>>>> - Updated obsolete and confusing comments
>>>>>>>>>>>>>
>>>>>>>>>>>>> This is tech debt that also will allow disabling concurrent
>>>>>>>>>>>>> refinement (if desired) and also fixes a P4 bug.
>>>>>>>>>>>>> Patch started by Thomas and improved and/or mangled myself.
>>>>>>>>>>>>>
>>>>>>>>>>>>> RFE: JDK-8138920
>>>>>>>>>>>>> <https://bugs.openjdk.java.net/browse/JDK-8138920>
>>>>>>>>>>>>> Refactor the
>>>>>>>>>>>>> sampling thread from ConcurrentG1RefineThread
>>>>>>>>>>>>>
>>>>>>>>>>>>> Webrev:
>>>>>>>>>>>>> http://cr.openjdk.java.net/~drwhite/8138920/webrev.01/
>>>>>>>>>>>>
>>>>>>>>>>>> Overall this looks really good to me.
>>>>>>>>>>>>
>>>>>>>>>>>> Some comments:
>>>>>>>>>>>>
>>>>>>>>>>>> No one seems to depend on
>>>>>>>>>>>> ConcurrentG1ServiceThread::vtime_accum(). All uses have the
>>>>>>>>>>>> specific subclass available. So, I don't think the pure
>>>>>>>>>>>> virtual
>>>>>>>>>>>> declaration in ConcurrentG1ServiceThread is needed. I'd just
>>>>>>>>>>>> remove that and also make the corresponding methods in the
>>>>>>>>>>>> subclasses non-virtual.
>>>>>>>>>>>
>>>>>>>>>>> OK. At some point we need to systematic rewrite of timing, but
>>>>>>>>>>> that can wait.
>>>>>>>>>>
>>>>>>>>>> Quite agree.
>>>>>>>>>>
>>>>>>>>>>>> That more or less only leaves the run() and stop() methods in
>>>>>>>>>>>> ConcurrentG1ServiceThread. It is kind of nice for the
>>>>>>>>>>>> subclasses
>>>>>>>>>>>> to get help with this, but I wonder if it is not possible to
>>>>>>>>>>>> improve the ConcurrentGCThread class to help with this
>>>>>>>>>>>> instead.
>>>>>>>>>>>> Basically I guess I am a little unsure if the extra class
>>>>>>>>>>>> ConcurrentG1ServiceThread is really needed.
>>>>>>>>>>>>
>>>>>>>>>>> I'll look at ConcurrentGCThread to see how well it could cover
>>>>>>>>>>> these cases.
>>>>>>>>>>
>>>>>>>>>> Thanks. I think it is worth a try. If it doesn't turn out
>>>>>>>>>> well we
>>>>>>>>>> can keep the intermediate class, but I think it is worth
>>>>>>>>>> exploring.
>>>>>>>>>
>>>>>>>>> I looked at pushing ConcurrentServiceThread up into
>>>>>>>>> ConcurrentGCThread, but things got a little complicated.
>>>>>>>>> ConcurrentG1RefineThread, ConcurrentMarkThread, and
>>>>>>>>> ConcurrentSampleThread have a very "regularized"
>>>>>>>>> implementation of
>>>>>>>>> the "termination protocol". G1StringDedupThread is slightly off
>>>>>>>>> from this, and ConcurrentMarkSweepThread more so. Pushing the
>>>>>>>>> shared code up into ConcurrentGCThread but not using it in
>>>>>>>>> G1StringDedupThread and ConcurrentMarkSweepThread seems
>>>>>>>>> confusing.
>>>>>>>>>
>>>>>>>>> There's a tension between providing a framework that handles all
>>>>>>>>> shared factorizable code, but can become a straight jacket for
>>>>>>>>> future code, and everyone doing everything separately and
>>>>>>>>> differently. This webrev is somewhere in the middle. Some of the
>>>>>>>>> changes between webrev.01 and .02 are to make the duplicated code
>>>>>>>>> more similar, even though it's not shared.
>>>>>>>>
>>>>>>>> I see. Thanks for investigating it!
>>>>>>>>
>>>>>>>> I think I agree with Per, though. The value of
>>>>>>>> ConcurrentServiceThread in its current form is IMHO not really
>>>>>>>> worth
>>>>>>>> the extra complexity of having it. I would prefer to just
>>>>>>>> duplicate
>>>>>>>> the code in ConcurrentG1RefineThread, ConcurrentMarkThread, and
>>>>>>>> ConcurrentSampleThread for now and then have a separate change to
>>>>>>>> try to clean this part up.
>>>>>>>>
>>>>>>>>>
>>>>>>>>>>>> Naming. The naming in G1 is a bit inconsistent. Some files and
>>>>>>>>>>>> classes are prefixed with G1 and some are not. But if they are
>>>>>>>>>>>> called something with G1 it is normally a prefix. So, I would
>>>>>>>>>>>> prefer the new classes to be called G1Concurrent* instead of
>>>>>>>>>>>> ConcurrentG1*.
>>>>>>>>>>>
>>>>>>>>>>> So we have:
>>>>>>>>>>> - ConcurrentG1RefineThread
>>>>>>>>>>> - ConcurrentMarkThread
>>>>>>>>>>>
>>>>>>>>>>> And I added:
>>>>>>>>>>> - ConcurrentG1SampleThread
>>>>>>>>>>> - ConcurrentG1ServiceThread
>>>>>>>>>>>
>>>>>>>>>>> And maybe I'm removing ConcurrentG1ServiceThread. In that case
>>>>>>>>>>> I'd be inclined to rename:
>>>>>>>>>>> ConcurrentG1SampleThread => ConcurrentSampleThread
>>>>>>>>>>
>>>>>>>>>> Sounds good. The ConcurrentG1Refine* classes are really the only
>>>>>>>>>> oddly named G1 classes, so I think it is better not to let that
>>>>>>>>>> naming spread.
>>>>>>>>> This version includes the class renaming.
>>>>>>>>
>>>>>>>> Thanks for fixing this.
>>>>>>>>
>>>>>>>> If we do decide to keep ConcurrentServiceThread around I think it
>>>>>>>> would be better to move ConcurrentSampleThread into its own
>>>>>>>> file. It
>>>>>>>> is a separate enough entity to warrant its own file I think.
>>>>>>>>
>>>>>>>>>>
>>>>>>>>>>>> You write "and also fixes a P4 bug". Which bug is that?
>>>>>>>>>>> JDK-8136856
>>>>>>>>>>> <https://bugs.openjdk.java.net/browse/JDK-8136856> G1
>>>>>>>>>>> makes two concurrent refinement threads with
>>>>>>>>>>> -XX:G1ConcRefinementThreads=1
>>>>>>>>>>> (because the sampling thread "is-a" concurrent refinement
>>>>>>>>>>> thread.
>>>>>>>>>>
>>>>>>>>>> Ah. I see. Makes sense. Thanks.
>>>>>>>>>>
>>>>>>>>>> But it is still not possible to turn refinement off by setting
>>>>>>>>>> -XX:G1ConcRefinementThreads=0 since that is considered the
>>>>>>>>>> default, right?
>>>>>>>>>
>>>>>>>>> I'm not sure about this. If I recall correctly, Thomas implied
>>>>>>>>> that
>>>>>>>>> it was hard to disable concurrent refinement without disabling
>>>>>>>>> the
>>>>>>>>> sampling thread too.
>>>>>>>>
>>>>>>>> I'm thinking about this code in Arguments::set_g1_gc_flags():
>>>>>>>>
>>>>>>>> if (G1ConcRefinementThreads == 0) {
>>>>>>>> FLAG_SET_DEFAULT(G1ConcRefinementThreads, ParallelGCThreads);
>>>>>>>> }
>>>>>>>>
>>>>>>>> Could we change that, so that you could turn off refinement by
>>>>>>>> setting -XX:G1ConcRefinementThreads=0 ? It should probably be
>>>>>>>> handled as a separate fix though.
>>>>>>>>
>>>>>>>> Thanks,
>>>>>>>> Bengt
>>>>>>>>
>>>>>>>>>
>>>>>>>>>>> I cant' recall how to mark a bug as blocking another bug.
>>>>>>>>>>
>>>>>>>>>> You add a link (More->Link) to the other bug and choose
>>>>>>>>>> "block" or
>>>>>>>>>> "is blocked by".
>>>>>>>>>>
>>>>>>>>>> Thanks,
>>>>>>>>>> Bengt
>>>>>>>>> Thanks for the tip!
>>>>>>>>>
>>>>>>>>> - Derek
>>>>>>>>>
>>>>>>>>
>>>>>>>
>>>>>>
>>>>>
>>>
>>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/hotspot-gc-dev/attachments/20151021/7a019474/attachment.htm>
More information about the hotspot-gc-dev
mailing list