Skip to content

Commit afc4441

Browse files
author
Markus Grönlund
committedDec 14, 2020
8258094: AIX build fails after 8257602
Reviewed-by: rrich, stuefe, egahlin
1 parent 164c8a6 commit afc4441

File tree

5 files changed

+159
-83
lines changed

5 files changed

+159
-83
lines changed
 

‎src/hotspot/share/gc/shared/allocTracer.cpp

+2-81
Original file line numberDiff line numberDiff line change
@@ -24,111 +24,32 @@
2424

2525
#include "precompiled.hpp"
2626
#include "gc/shared/allocTracer.hpp"
27-
#include "gc/shared/threadLocalAllocBuffer.inline.hpp"
2827
#include "jfr/jfrEvents.hpp"
2928
#include "utilities/globalDefinitions.hpp"
3029
#include "utilities/macros.hpp"
3130
#if INCLUDE_JFR
3231
#include "jfr/support/jfrAllocationTracer.hpp"
3332
#endif
3433

35-
static THREAD_LOCAL int64_t _last_allocated_bytes = 0;
36-
37-
inline void send_allocation_sample(const Klass* klass, int64_t allocated_bytes) {
38-
assert(allocated_bytes > 0, "invariant");
39-
EventObjectAllocationSample event;
40-
if (event.should_commit()) {
41-
const size_t weight = allocated_bytes - _last_allocated_bytes;
42-
assert(weight > 0, "invariant");
43-
event.set_objectClass(klass);
44-
event.set_weight(weight);
45-
event.commit();
46-
_last_allocated_bytes = allocated_bytes;
47-
}
48-
}
49-
50-
inline bool send_allocation_sample_with_result(const Klass* klass, int64_t allocated_bytes) {
51-
assert(allocated_bytes > 0, "invariant");
52-
EventObjectAllocationSample event;
53-
if (event.should_commit()) {
54-
const size_t weight = allocated_bytes - _last_allocated_bytes;
55-
assert(weight > 0, "invariant");
56-
event.set_objectClass(klass);
57-
event.set_weight(weight);
58-
event.commit();
59-
_last_allocated_bytes = allocated_bytes;
60-
return true;
61-
}
62-
return false;
63-
}
64-
65-
inline intptr_t estimate_tlab_size_bytes(Thread* thread) {
66-
assert(thread != NULL, "invariant");
67-
const size_t desired_tlab_size_bytes = thread->tlab().desired_size() * HeapWordSize;
68-
const size_t alignment_reserve_bytes = thread->tlab().alignment_reserve_in_bytes();
69-
assert(desired_tlab_size_bytes > alignment_reserve_bytes, "invariant");
70-
return static_cast<intptr_t>(desired_tlab_size_bytes - alignment_reserve_bytes);
71-
}
72-
73-
inline int64_t load_allocated_bytes(Thread* thread) {
74-
const int64_t allocated_bytes = thread->allocated_bytes();
75-
if (allocated_bytes < _last_allocated_bytes) {
76-
// A hw thread can detach and reattach to the VM, and when it does,
77-
// it gets a new JavaThread representation. The thread local variable
78-
// tracking _last_allocated_bytes is mapped to the existing hw thread,
79-
// so it needs to be reset.
80-
_last_allocated_bytes = 0;
81-
}
82-
return allocated_bytes == _last_allocated_bytes ? 0 : allocated_bytes;
83-
}
84-
85-
// To avoid large objects from being undersampled compared to the regular TLAB samples,
86-
// the data amount is normalized as if it was a TLAB, giving a number of TLAB sampling attempts to the large object.
87-
static void normalize_as_tlab_and_send_allocation_samples(Klass* klass, intptr_t obj_alloc_size_bytes, Thread* thread) {
88-
const int64_t allocated_bytes = load_allocated_bytes(thread);
89-
assert(allocated_bytes > 0, "invariant"); // obj_alloc_size_bytes is already attributed to allocated_bytes at this point.
90-
if (!UseTLAB) {
91-
send_allocation_sample(klass, allocated_bytes);
92-
return;
93-
}
94-
const intptr_t tlab_size_bytes = estimate_tlab_size_bytes(thread);
95-
if (allocated_bytes - _last_allocated_bytes < tlab_size_bytes) {
96-
return;
97-
}
98-
assert(obj_alloc_size_bytes > 0, "invariant");
99-
do {
100-
if (send_allocation_sample_with_result(klass, allocated_bytes)) {
101-
return;
102-
}
103-
obj_alloc_size_bytes -= tlab_size_bytes;
104-
} while (obj_alloc_size_bytes > 0);
105-
}
106-
10734
void AllocTracer::send_allocation_outside_tlab(Klass* klass, HeapWord* obj, size_t alloc_size, Thread* thread) {
108-
JFR_ONLY(JfrAllocationTracer tracer(obj, alloc_size, thread);)
35+
JFR_ONLY(JfrAllocationTracer tracer(klass, obj, alloc_size, true, thread);)
10936
EventObjectAllocationOutsideTLAB event;
11037
if (event.should_commit()) {
11138
event.set_objectClass(klass);
11239
event.set_allocationSize(alloc_size);
11340
event.commit();
11441
}
115-
normalize_as_tlab_and_send_allocation_samples(klass, static_cast<intptr_t>(alloc_size), thread);
11642
}
11743

11844
void AllocTracer::send_allocation_in_new_tlab(Klass* klass, HeapWord* obj, size_t tlab_size, size_t alloc_size, Thread* thread) {
119-
JFR_ONLY(JfrAllocationTracer tracer(obj, alloc_size, thread);)
45+
JFR_ONLY(JfrAllocationTracer tracer(klass, obj, alloc_size, false, thread);)
12046
EventObjectAllocationInNewTLAB event;
12147
if (event.should_commit()) {
12248
event.set_objectClass(klass);
12349
event.set_allocationSize(alloc_size);
12450
event.set_tlabSize(tlab_size);
12551
event.commit();
12652
}
127-
const int64_t allocated_bytes = load_allocated_bytes(thread);
128-
if (allocated_bytes == 0) {
129-
return;
130-
}
131-
send_allocation_sample(klass, allocated_bytes);
13253
}
13354

13455
void AllocTracer::send_allocation_requiring_gc_event(size_t size, uint gcId) {

‎src/hotspot/share/jfr/support/jfrAllocationTracer.cpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -25,10 +25,12 @@
2525
#include "precompiled.hpp"
2626
#include "jfr/leakprofiler/leakProfiler.hpp"
2727
#include "jfr/support/jfrAllocationTracer.hpp"
28+
#include "jfr/support/jfrObjectAllocationSample.hpp"
2829
#include "jfr/support/jfrThreadLocal.hpp"
2930
#include "runtime/thread.hpp"
3031

31-
JfrAllocationTracer::JfrAllocationTracer(HeapWord* obj, size_t alloc_size, Thread* thread) : _tl(NULL) {
32+
JfrAllocationTracer::JfrAllocationTracer(const Klass* klass, HeapWord* obj, size_t alloc_size, bool outside_tlab, Thread* thread) : _tl(NULL) {
33+
JfrObjectAllocationSample::send_event(klass, alloc_size, outside_tlab, thread);
3234
if (LeakProfiler::is_running()) {
3335
_tl = thread->jfr_thread_local();
3436
LeakProfiler::sample(obj, alloc_size, thread->as_Java_thread());

‎src/hotspot/share/jfr/support/jfrAllocationTracer.hpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -27,13 +27,15 @@
2727

2828
#include "memory/allocation.hpp"
2929

30+
class Klass;
3031
class JfrThreadLocal;
32+
class Thread;
3133

3234
class JfrAllocationTracer : public StackObj {
3335
private:
3436
JfrThreadLocal* _tl;
3537
public:
36-
JfrAllocationTracer(HeapWord* obj, size_t alloc_size, Thread* thread);
38+
JfrAllocationTracer(const Klass* klass, HeapWord* obj, size_t alloc_size, bool outside_tlab, Thread* thread);
3739
~JfrAllocationTracer();
3840
};
3941

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
#include "precompiled.hpp"
26+
#include "gc/shared/threadLocalAllocBuffer.inline.hpp"
27+
#include "jfr/jfrEvents.hpp"
28+
#include "jfr/support/jfrObjectAllocationSample.hpp"
29+
#include "utilities/globalDefinitions.hpp"
30+
31+
static THREAD_LOCAL int64_t _last_allocated_bytes = 0;
32+
33+
inline void send_allocation_sample(const Klass* klass, int64_t allocated_bytes) {
34+
assert(allocated_bytes > 0, "invariant");
35+
EventObjectAllocationSample event;
36+
if (event.should_commit()) {
37+
const size_t weight = allocated_bytes - _last_allocated_bytes;
38+
assert(weight > 0, "invariant");
39+
event.set_objectClass(klass);
40+
event.set_weight(weight);
41+
event.commit();
42+
_last_allocated_bytes = allocated_bytes;
43+
}
44+
}
45+
46+
inline bool send_allocation_sample_with_result(const Klass* klass, int64_t allocated_bytes) {
47+
assert(allocated_bytes > 0, "invariant");
48+
EventObjectAllocationSample event;
49+
if (event.should_commit()) {
50+
const size_t weight = allocated_bytes - _last_allocated_bytes;
51+
assert(weight > 0, "invariant");
52+
event.set_objectClass(klass);
53+
event.set_weight(weight);
54+
event.commit();
55+
_last_allocated_bytes = allocated_bytes;
56+
return true;
57+
}
58+
return false;
59+
}
60+
61+
inline intptr_t estimate_tlab_size_bytes(Thread* thread) {
62+
const size_t desired_tlab_size_bytes = thread->tlab().desired_size() * HeapWordSize;
63+
const size_t alignment_reserve_bytes = thread->tlab().alignment_reserve_in_bytes();
64+
assert(desired_tlab_size_bytes > alignment_reserve_bytes, "invariant");
65+
return static_cast<intptr_t>(desired_tlab_size_bytes - alignment_reserve_bytes);
66+
}
67+
68+
inline int64_t load_allocated_bytes(Thread* thread) {
69+
assert(thread != NULL, "invariant");
70+
const int64_t allocated_bytes = thread->allocated_bytes();
71+
if (allocated_bytes < _last_allocated_bytes) {
72+
// A hw thread can detach and reattach to the VM, and when it does,
73+
// it gets a new JavaThread representation. The thread local variable
74+
// tracking _last_allocated_bytes is mapped to the existing hw thread,
75+
// so it needs to be reset.
76+
_last_allocated_bytes = 0;
77+
}
78+
return allocated_bytes == _last_allocated_bytes ? 0 : allocated_bytes;
79+
}
80+
81+
// To avoid large objects from being undersampled compared to the regular TLAB samples,
82+
// the data amount is normalized as if it was a TLAB, giving a number of TLAB sampling attempts to the large object.
83+
static void normalize_as_tlab_and_send_allocation_samples(const Klass* klass, intptr_t obj_alloc_size_bytes, Thread* thread) {
84+
const int64_t allocated_bytes = load_allocated_bytes(thread);
85+
assert(allocated_bytes > 0, "invariant"); // obj_alloc_size_bytes is already attributed to allocated_bytes at this point.
86+
if (!UseTLAB) {
87+
send_allocation_sample(klass, allocated_bytes);
88+
return;
89+
}
90+
const intptr_t tlab_size_bytes = estimate_tlab_size_bytes(thread);
91+
if (allocated_bytes - _last_allocated_bytes < tlab_size_bytes) {
92+
return;
93+
}
94+
assert(obj_alloc_size_bytes > 0, "invariant");
95+
do {
96+
if (send_allocation_sample_with_result(klass, allocated_bytes)) {
97+
return;
98+
}
99+
obj_alloc_size_bytes -= tlab_size_bytes;
100+
} while (obj_alloc_size_bytes > 0);
101+
}
102+
103+
void JfrObjectAllocationSample::send_event(const Klass* klass, size_t alloc_size, bool outside_tlab, Thread* thread) {
104+
if (outside_tlab) {
105+
normalize_as_tlab_and_send_allocation_samples(klass, static_cast<intptr_t>(alloc_size), thread);
106+
return;
107+
}
108+
const int64_t allocated_bytes = load_allocated_bytes(thread);
109+
if (allocated_bytes == 0) {
110+
return;
111+
}
112+
send_allocation_sample(klass, allocated_bytes);
113+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
/*
2+
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
#ifndef SHARE_JFR_SUPPORT_JFROBJECTALLOCATIONSAMPLE_HPP
26+
#define SHARE_JFR_SUPPORT_JFROBJECTALLOCATIONSAMPLE_HPP
27+
28+
#include "memory/allocation.hpp"
29+
30+
class Klass;
31+
class Thread;
32+
33+
class JfrObjectAllocationSample : AllStatic {
34+
friend class JfrAllocationTracer;
35+
static void send_event(const Klass* klass, size_t alloc_size, bool outside_tlab, Thread* thread);
36+
};
37+
38+
#endif // SHARE_JFR_SUPPORT_JFROBJECTALLOCATIONSAMPLE_HPP

0 commit comments

Comments
 (0)
Please sign in to comment.