1 /*
2 * Copyright (c) 2020, 2024, 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_CLASSFILE_FIELDLAYOUTBUILDER_HPP
26 #define SHARE_CLASSFILE_FIELDLAYOUTBUILDER_HPP
27
28 #include "classfile/classFileParser.hpp"
29 #include "classfile/classLoaderData.hpp"
30 #include "memory/allocation.hpp"
31 #include "oops/fieldStreams.hpp"
32 #include "utilities/growableArray.hpp"
33
34 // Classes below are used to compute the field layout of classes.
35
36
37 // A LayoutRawBlock describes an element of a layout.
38 // Each field is represented by a LayoutRawBlock.
39 // LayoutRawBlocks can also represent elements injected by the JVM:
40 // padding, empty blocks, inherited fields, etc.
41 // All LayoutRawBlocks must have a size and an alignment. The size is the
42 // exact size of the field expressed in bytes. The alignment is
43 // the alignment constraint of the field (1 for byte, 2 for short,
44 // 4 for int, 8 for long, etc.)
45 //
46 // LayoutRawBlock are designed to be used in two data structures:
47 // - a linked list in a layout (using _next_block, _prev_block)
48 // - a GrowableArray in field group (the growable array contains pointers to LayoutRawBlocks)
49 //
50 // next/prev pointers are included in the LayoutRawBlock class to narrow
51 // the number of allocation required during the computation of a layout.
52 //
53 class LayoutRawBlock : public ResourceObj {
54 public:
55 // Some code relies on the order of values below.
56 enum Kind {
57 EMPTY, // empty slot, space is taken from this to allocate fields
58 RESERVED, // reserved for JVM usage (for instance object header)
59 PADDING, // padding (because of alignment constraints or @Contended)
60 REGULAR, // primitive or oop field (including non-flattened inline fields)
61 FLATTENED, // flattened field
62 INHERITED // field(s) inherited from super classes
63 };
64
65 private:
66 LayoutRawBlock* _next_block;
67 LayoutRawBlock* _prev_block;
68 Kind _kind;
69 int _offset;
70 int _alignment;
71 int _size;
72 int _field_index;
73 bool _is_reference;
74
75 public:
76 LayoutRawBlock(Kind kind, int size);
77 LayoutRawBlock(int index, Kind kind, int size, int alignment, bool is_reference = false);
78 LayoutRawBlock* next_block() const { return _next_block; }
79 void set_next_block(LayoutRawBlock* next) { _next_block = next; }
80 LayoutRawBlock* prev_block() const { return _prev_block; }
81 void set_prev_block(LayoutRawBlock* prev) { _prev_block = prev; }
82 Kind kind() const { return _kind; }
83 int offset() const {
84 assert(_offset >= 0, "Must be initialized");
85 return _offset;
86 }
87 void set_offset(int offset) { _offset = offset; }
88 int alignment() const { return _alignment; }
89 int size() const { return _size; }
90 void set_size(int size) { _size = size; }
91 int field_index() const {
92 assert(_field_index != -1, "Must be initialized");
93 return _field_index;
94 }
95 bool is_reference() const { return _is_reference; }
96
97 bool fit(int size, int alignment);
98
99 static int compare_offset(LayoutRawBlock** x, LayoutRawBlock** y) { return (*x)->offset() - (*y)->offset(); }
100 // compare_size_inverted() returns the opposite of a regular compare method in order to
101 // sort fields in decreasing order.
102 // Note: with line types, the comparison should include alignment constraint if sizes are equals
103 static int compare_size_inverted(LayoutRawBlock** x, LayoutRawBlock** y) {
104 int diff = (*y)->size() - (*x)->size();
105 // qsort() may reverse the order of fields with the same size.
106 // The extension is to ensure stable sort.
107 if (diff == 0) {
108 diff = (*x)->field_index() - (*y)->field_index();
109 }
110 return diff;
111 }
112
113 };
114
115 // A Field group represents a set of fields that have to be allocated together,
116 // this is the way the @Contended annotation is supported.
117 // Inside a FieldGroup, fields are sorted based on their kind: primitive,
118 // oop, or flattened.
119 //
120 class FieldGroup : public ResourceObj {
121
122 private:
123 FieldGroup* _next;
124 GrowableArray<LayoutRawBlock*>* _primitive_fields;
125 GrowableArray<LayoutRawBlock*>* _oop_fields;
126 int _contended_group;
127 int _oop_count;
128 static const int INITIAL_LIST_SIZE = 16;
129
130 public:
131 FieldGroup(int contended_group = -1);
132
133 FieldGroup* next() const { return _next; }
134 void set_next(FieldGroup* next) { _next = next; }
135 GrowableArray<LayoutRawBlock*>* primitive_fields() const { return _primitive_fields; }
136 GrowableArray<LayoutRawBlock*>* oop_fields() const { return _oop_fields; }
137 int contended_group() const { return _contended_group; }
138 int oop_count() const { return _oop_count; }
139
140 void add_primitive_field(int idx, BasicType type);
141 void add_oop_field(int idx);
142 void sort_by_size();
143 };
144
145 // The FieldLayout class represents a set of fields organized
146 // in a layout.
147 // An instance of FieldLayout can either represent the layout
148 // of non-static fields (used in an instance object) or the
149 // layout of static fields (to be included in the class mirror).
150 //
151 // _block is a pointer to a list of LayoutRawBlock ordered by increasing
152 // offsets.
153 // _start points to the LayoutRawBlock with the first offset that can
154 // be used to allocate fields of the current class
155 // _last points to the last LayoutRawBlock of the list. In order to
156 // simplify the code, the LayoutRawBlock list always ends with an
157 // EMPTY block (the kind of LayoutRawBlock from which space is taken
158 // to allocate fields) with a size big enough to satisfy all
159 // field allocations.
160 //
161 class FieldLayout : public ResourceObj {
162 private:
163 GrowableArray<FieldInfo>* _field_info;
164 ConstantPool* _cp;
165 LayoutRawBlock* _blocks; // the layout being computed
166 LayoutRawBlock* _start; // points to the first block where a field can be inserted
167 LayoutRawBlock* _last; // points to the last block of the layout (big empty block)
168
169 public:
170 FieldLayout(GrowableArray<FieldInfo>* field_info, ConstantPool* cp);
171 void initialize_static_layout();
172 void initialize_instance_layout(const InstanceKlass* ik);
173
174 LayoutRawBlock* first_empty_block() {
175 LayoutRawBlock* block = _start;
176 while (block->kind() != LayoutRawBlock::EMPTY) {
177 block = block->next_block();
178 }
179 return block;
180 }
181
182 LayoutRawBlock* start() { return _start; }
183 void set_start(LayoutRawBlock* start) { _start = start; }
184 LayoutRawBlock* last_block() { return _last; }
185
186 LayoutRawBlock* first_field_block();
187 void add(GrowableArray<LayoutRawBlock*>* list, LayoutRawBlock* start = nullptr);
188 void add_field_at_offset(LayoutRawBlock* blocks, int offset, LayoutRawBlock* start = nullptr);
189 void add_contiguously(GrowableArray<LayoutRawBlock*>* list, LayoutRawBlock* start = nullptr);
190 LayoutRawBlock* insert_field_block(LayoutRawBlock* slot, LayoutRawBlock* block);
191 bool reconstruct_layout(const InstanceKlass* ik);
192 void fill_holes(const InstanceKlass* ik);
193 LayoutRawBlock* insert(LayoutRawBlock* slot, LayoutRawBlock* block);
194 void remove(LayoutRawBlock* block);
195 void print(outputStream* output, bool is_static, const InstanceKlass* super);
196 };
197
198
199 // FieldLayoutBuilder is the main entry point for layout computation.
200 // This class has three methods to generate layout: one for regular classes
201 // and two for classes with hard coded offsets (java,lang.ref.Reference
202 // and the boxing classes). The rationale for having multiple methods
203 // is that each kind of class has a different set goals regarding
204 // its layout, so instead of mixing several layout strategies into a
205 // single method, each kind has its own method (see comments below
206 // for more details about the allocation strategies).
207 //
208 // Computing the layout of a class always goes through 4 steps:
209 // 1 - Prologue: preparation of data structure and gathering of
210 // layout information inherited from super classes
211 // 2 - Field sorting: fields are sorted according to their
212 // kind (oop, primitive, inline class) and their contention
213 // annotation (if any)
214 // 3 - Layout is computed from the set of lists generated during
215 // step 2
216 // 4 - Epilogue: oopmaps are generated, layout information is
217 // prepared so other VM components can use it (instance size,
218 // static field size, non-static field size, etc.)
219 //
220 // Steps 1 and 4 are common to all layout computations. Step 2 and 3
221 // can vary with the allocation strategy.
222 //
223 class FieldLayoutBuilder : public ResourceObj {
224 private:
225
226 const Symbol* _classname;
227 const InstanceKlass* _super_klass;
228 ConstantPool* _constant_pool;
229 GrowableArray<FieldInfo>* _field_info;
230 FieldLayoutInfo* _info;
231 FieldGroup* _root_group;
232 GrowableArray<FieldGroup*> _contended_groups;
233 FieldGroup* _static_fields;
234 FieldLayout* _layout;
235 FieldLayout* _static_layout;
236 int _nonstatic_oopmap_count;
237 int _alignment;
238 bool _has_nonstatic_fields;
239 bool _is_contended; // is a contended class?
240
241 public:
242 FieldLayoutBuilder(const Symbol* classname, const InstanceKlass* super_klass, ConstantPool* constant_pool,
243 GrowableArray<FieldInfo>* field_info, bool is_contended, FieldLayoutInfo* info);
244
245 int get_alignment() {
246 assert(_alignment != -1, "Uninitialized");
247 return _alignment;
248 }
249
250 void build_layout();
251 void compute_regular_layout();
252 void insert_contended_padding(LayoutRawBlock* slot);
253
254 private:
255 void prologue();
256 void epilogue();
257 void regular_field_sorting();
258 FieldGroup* get_or_create_contended_group(int g);
259 };
260
261 #endif // SHARE_CLASSFILE_FIELDLAYOUTBUILDER_HPP
|
1 /*
2 * Copyright (c) 2020, 2025, 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_CLASSFILE_FIELDLAYOUTBUILDER_HPP
26 #define SHARE_CLASSFILE_FIELDLAYOUTBUILDER_HPP
27
28 #include "classfile/classFileParser.hpp"
29 #include "classfile/classLoaderData.hpp"
30 #include "memory/allocation.hpp"
31 #include "oops/fieldStreams.hpp"
32 #include "oops/inlineKlass.hpp"
33 #include "oops/instanceKlass.hpp"
34 #include "utilities/growableArray.hpp"
35
36 // Classes below are used to compute the field layout of classes.
37
38 // A LayoutRawBlock describes an element of a layout.
39 // Each field is represented by a LayoutRawBlock.
40 // LayoutRawBlocks can also represent elements injected by the JVM:
41 // padding, empty blocks, inherited fields, etc.
42 // All LayoutRawBlocks must have a size and an alignment. The size is the
43 // exact size of the field expressed in bytes. The alignment is
44 // the alignment constraint of the field (1 for byte, 2 for short,
45 // 4 for int, 8 for long, etc.)
46 //
47 // LayoutRawBlock are designed to be used in two data structures:
48 // - a linked list in a layout (using _next_block, _prev_block)
49 // - a GrowableArray in field group (the growable array contains pointers to LayoutRawBlocks)
50 //
51 // next/prev pointers are included in the LayoutRawBlock class to narrow
52 // the number of allocation required during the computation of a layout.
53 //
54
55 #define MAX_ATOMIC_OP_SIZE sizeof(uint64_t)
56
57 class LayoutRawBlock : public ResourceObj {
58 public:
59 // Some code relies on the order of values below.
60 enum Kind {
61 EMPTY, // empty slot, space is taken from this to allocate fields
62 RESERVED, // reserved for JVM usage (for instance object header)
63 PADDING, // padding (because of alignment constraints or @Contended)
64 REGULAR, // primitive or oop field (including not flat inline type fields)
65 FLAT, // flat field
66 INHERITED, // field(s) inherited from super classes
67 NULL_MARKER // stores the null marker for a flat field
68 };
69
70 private:
71 LayoutRawBlock* _next_block;
72 LayoutRawBlock* _prev_block;
73 InlineKlass* _inline_klass;
74 Kind _block_kind;
75 LayoutKind _layout_kind;
76 int _offset;
77 int _alignment;
78 int _size;
79 int _field_index;
80 int _null_marker_offset;
81
82 public:
83 LayoutRawBlock(Kind kind, int size);
84
85 LayoutRawBlock(int index, Kind kind, int size, int alignment);
86 LayoutRawBlock* next_block() const { return _next_block; }
87 void set_next_block(LayoutRawBlock* next) { _next_block = next; }
88 LayoutRawBlock* prev_block() const { return _prev_block; }
89 void set_prev_block(LayoutRawBlock* prev) { _prev_block = prev; }
90 Kind block_kind() const { return _block_kind; }
91 void set_block_kind(LayoutRawBlock::Kind kind) { _block_kind = kind; } // Dangerous operation, is only used by remove_null_marker();
92 int offset() const {
93 assert(_offset >= 0, "Must be initialized");
94 return _offset;
95 }
96 void set_offset(int offset) { _offset = offset; }
97 int alignment() const { return _alignment; }
98 int size() const { return _size; }
99 void set_size(int size) { _size = size; }
100 int field_index() const {
101 assert(_field_index != -1, "Must be initialized");
102 return _field_index;
103 }
104 void set_field_index(int field_index) {
105 assert(_field_index == -1, "Must not be initialized");
106 _field_index = field_index;
107 }
108 InlineKlass* inline_klass() const {
109 assert(_inline_klass != nullptr, "Must be initialized");
110 return _inline_klass;
111 }
112 void set_inline_klass(InlineKlass* inline_klass) { _inline_klass = inline_klass; }
113 void set_null_marker_offset(int offset) { _null_marker_offset = offset; }
114 int null_marker_offset() const { return _null_marker_offset; }
115
116 LayoutKind layout_kind() const { return _layout_kind; }
117 void set_layout_kind(LayoutKind kind) { _layout_kind = kind; }
118
119 bool fit(int size, int alignment);
120
121 static int compare_offset(LayoutRawBlock** x, LayoutRawBlock** y) { return (*x)->offset() - (*y)->offset(); }
122 // compare_size_inverted() returns the opposite of a regular compare method in order to
123 // sort fields in decreasing order.
124 // Note: with line types, the comparison should include alignment constraint if sizes are equals
125 static int compare_size_inverted(LayoutRawBlock** x, LayoutRawBlock** y) {
126 int diff = (*y)->size() - (*x)->size();
127 // qsort() may reverse the order of fields with the same size.
128 // The extension is to ensure stable sort.
129 if (diff == 0) {
130 diff = (*x)->field_index() - (*y)->field_index();
131 }
132 return diff;
133 }
134 };
135
136 // A Field group represents a set of fields that have to be allocated together,
137 // this is the way the @Contended annotation is supported.
138 // Inside a FieldGroup, fields are sorted based on their kind: primitive,
139 // oop, or flat.
140 //
141 class FieldGroup : public ResourceObj {
142
143 private:
144 FieldGroup* _next;
145
146 GrowableArray<LayoutRawBlock*>* _small_primitive_fields;
147 GrowableArray<LayoutRawBlock*>* _big_primitive_fields;
148 GrowableArray<LayoutRawBlock*>* _oop_fields;
149 int _contended_group;
150 int _oop_count;
151 static const int INITIAL_LIST_SIZE = 16;
152
153 public:
154 FieldGroup(int contended_group = -1);
155
156 FieldGroup* next() const { return _next; }
157 void set_next(FieldGroup* next) { _next = next; }
158 GrowableArray<LayoutRawBlock*>* small_primitive_fields() const { return _small_primitive_fields; }
159 GrowableArray<LayoutRawBlock*>* big_primitive_fields() const { return _big_primitive_fields; }
160 GrowableArray<LayoutRawBlock*>* oop_fields() const { return _oop_fields; }
161 int contended_group() const { return _contended_group; }
162 int oop_count() const { return _oop_count; }
163
164 void add_primitive_field(int idx, BasicType type);
165 void add_oop_field(int idx);
166 void add_flat_field(int idx, InlineKlass* vk, LayoutKind lk, int size, int alignment);
167 void add_block(LayoutRawBlock** list, LayoutRawBlock* block);
168 void sort_by_size();
169 private:
170 void add_to_small_primitive_list(LayoutRawBlock* block);
171 void add_to_big_primitive_list(LayoutRawBlock* block);
172 };
173
174 // The FieldLayout class represents a set of fields organized
175 // in a layout.
176 // An instance of FieldLayout can either represent the layout
177 // of non-static fields (used in an instance object) or the
178 // layout of static fields (to be included in the class mirror).
179 //
180 // _block is a pointer to a list of LayoutRawBlock ordered by increasing
181 // offsets.
182 // _start points to the LayoutRawBlock with the first offset that can
183 // be used to allocate fields of the current class
184 // _last points to the last LayoutRawBlock of the list. In order to
185 // simplify the code, the LayoutRawBlock list always ends with an
186 // EMPTY block (the kind of LayoutRawBlock from which space is taken
187 // to allocate fields) with a size big enough to satisfy all
188 // field allocations.
189 //
190 class FieldLayout : public ResourceObj {
191 private:
192 GrowableArray<FieldInfo>* _field_info;
193 Array<InlineLayoutInfo>* _inline_layout_info_array;
194 ConstantPool* _cp;
195 LayoutRawBlock* _blocks; // the layout being computed
196 LayoutRawBlock* _start; // points to the first block where a field can be inserted
197 LayoutRawBlock* _last; // points to the last block of the layout (big empty block)
198 int _super_first_field_offset;
199 int _super_alignment;
200 int _super_min_align_required;
201 int _default_value_offset; // offset of the default value in class mirror, only for static layout of inline classes
202 int _null_reset_value_offset; // offset of the reset value in class mirror, only for static layout of inline classes
203 bool _super_has_fields;
204 bool _has_inherited_fields;
205
206 public:
207 FieldLayout(GrowableArray<FieldInfo>* field_info, Array<InlineLayoutInfo>* inline_layout_info_array, ConstantPool* cp);
208 void initialize_static_layout();
209 void initialize_instance_layout(const InstanceKlass* ik);
210
211 LayoutRawBlock* first_empty_block() {
212 LayoutRawBlock* block = _start;
213 while (block->block_kind() != LayoutRawBlock::EMPTY) {
214 block = block->next_block();
215 }
216 return block;
217 }
218
219 LayoutRawBlock* blocks() const { return _blocks; }
220
221 LayoutRawBlock* start() const { return _start; }
222 void set_start(LayoutRawBlock* start) { _start = start; }
223 LayoutRawBlock* last_block() const { return _last; }
224 int super_first_field_offset() const { return _super_first_field_offset; }
225 int super_alignment() const { return _super_alignment; }
226 int super_min_align_required() const { return _super_min_align_required; }
227 int default_value_offset() const {
228 assert(_default_value_offset != -1, "Must have been set");
229 return _default_value_offset;
230 }
231 int null_reset_value_offset() const {
232 assert(_null_reset_value_offset != -1, "Must have been set");
233 return _null_reset_value_offset;
234 }
235 bool super_has_fields() const { return _super_has_fields; }
236 bool has_inherited_fields() const { return _has_inherited_fields; }
237
238 LayoutRawBlock* first_field_block();
239 void add(GrowableArray<LayoutRawBlock*>* list, LayoutRawBlock* start = nullptr);
240 void add_field_at_offset(LayoutRawBlock* blocks, int offset, LayoutRawBlock* start = nullptr);
241 void add_contiguously(GrowableArray<LayoutRawBlock*>* list, LayoutRawBlock* start = nullptr);
242 LayoutRawBlock* insert_field_block(LayoutRawBlock* slot, LayoutRawBlock* block);
243 bool reconstruct_layout(const InstanceKlass* ik);
244 void fill_holes(const InstanceKlass* ik);
245 LayoutRawBlock* insert(LayoutRawBlock* slot, LayoutRawBlock* block);
246 void remove(LayoutRawBlock* block);
247 void shift_fields(int shift);
248 LayoutRawBlock* find_null_marker();
249 void remove_null_marker();
250 void print(outputStream* output, bool is_static, const InstanceKlass* super, Array<InlineLayoutInfo>* inline_fields);
251 };
252
253
254 // FieldLayoutBuilder is the main entry point for layout computation.
255 // This class has two methods to generate layout: one for identity classes
256 // and one for inline classes. The rational for having two methods
257 // is that each kind of classes has a different set goals regarding
258 // its layout, so instead of mixing two layout strategies into a
259 // single method, each kind has its own method (see comments below
260 // for more details about the allocation strategies).
261 //
262 // Computing the layout of a class always goes through 4 steps:
263 // 1 - Prologue: preparation of data structure and gathering of
264 // layout information inherited from super classes
265 // 2 - Field sorting: fields are sorted according to their
266 // kind (oop, primitive, inline class) and their contention
267 // annotation (if any)
268 // 3 - Layout is computed from the set of lists generated during
269 // step 2
270 // 4 - Epilogue: oopmaps are generated, layout information is
271 // prepared so other VM components can use it (instance size,
272 // static field size, non-static field size, etc.)
273 //
274 // Steps 1 and 4 are common to all layout computations. Step 2 and 3
275 // differ for inline classes and identity classes.
276 //
277 class FieldLayoutBuilder : public ResourceObj {
278
279 private:
280 const Symbol* _classname;
281 ClassLoaderData* _loader_data;
282 const InstanceKlass* _super_klass;
283 ConstantPool* _constant_pool;
284 GrowableArray<FieldInfo>* _field_info;
285 FieldLayoutInfo* _info;
286 Array<InlineLayoutInfo>* _inline_layout_info_array;
287 FieldGroup* _root_group;
288 GrowableArray<FieldGroup*> _contended_groups;
289 FieldGroup* _static_fields;
290 FieldLayout* _layout;
291 FieldLayout* _static_layout;
292 int _nonstatic_oopmap_count;
293 int _payload_alignment;
294 int _first_field_offset;
295 int _null_marker_offset; // if any, -1 means no internal null marker
296 int _payload_size_in_bytes;
297 int _non_atomic_layout_size_in_bytes;
298 int _non_atomic_layout_alignment;
299 int _atomic_layout_size_in_bytes;
300 int _nullable_layout_size_in_bytes;
301 int _fields_size_sum;
302 int _declared_non_static_fields_count;
303 bool _has_non_naturally_atomic_fields;
304 bool _is_naturally_atomic;
305 bool _must_be_atomic;
306 bool _has_nonstatic_fields;
307 bool _has_inline_type_fields;
308 bool _is_contended;
309 bool _is_inline_type;
310 bool _is_abstract_value;
311 bool _has_flattening_information;
312 bool _is_empty_inline_class;
313
314 FieldGroup* get_or_create_contended_group(int g);
315
316 public:
317 FieldLayoutBuilder(const Symbol* classname, ClassLoaderData* loader_data, const InstanceKlass* super_klass, ConstantPool* constant_pool,
318 GrowableArray<FieldInfo>* field_info, bool is_contended, bool is_inline_type, bool is_abstract_value,
319 bool must_be_atomic, FieldLayoutInfo* info, Array<InlineLayoutInfo>* inline_layout_info_array);
320
321 int first_field_offset() const { assert(_first_field_offset != -1, "Uninitialized"); return _first_field_offset; }
322 int payload_layout_size_in_bytes() const { return _payload_size_in_bytes; }
323 int payload_layout_alignment() const { assert(_payload_alignment != -1, "Uninitialized"); return _payload_alignment; }
324 bool has_non_atomic_flat_layout() const { return _non_atomic_layout_size_in_bytes != -1; }
325 int non_atomic_layout_size_in_bytes() const { return _non_atomic_layout_size_in_bytes; }
326 int non_atomic_layout_alignment() const { return _non_atomic_layout_alignment; }
327 bool has_atomic_layout() const { return _atomic_layout_size_in_bytes != -1; }
328 int atomic_layout_size_in_bytes() const { return _atomic_layout_size_in_bytes; }
329 bool has_nullable_atomic_layout() const { return _nullable_layout_size_in_bytes != -1; }
330 int nullable_layout_size_in_bytes() const { return _nullable_layout_size_in_bytes; }
331 int null_marker_offset() const { return _null_marker_offset; }
332 bool is_empty_inline_class() const { return _is_empty_inline_class; }
333
334 void build_layout();
335 void compute_regular_layout();
336 void compute_inline_class_layout();
337 void insert_contended_padding(LayoutRawBlock* slot);
338
339 protected:
340 void prologue();
341 void epilogue();
342 void regular_field_sorting();
343 void inline_class_field_sorting();
344 void add_flat_field_oopmap(OopMapBlocksBuilder* nonstatic_oop_map, InlineKlass* vk, int offset);
345 void register_embedded_oops_from_list(OopMapBlocksBuilder* nonstatic_oop_maps, GrowableArray<LayoutRawBlock*>* list);
346 void register_embedded_oops(OopMapBlocksBuilder* nonstatic_oop_maps, FieldGroup* group);
347 };
348
349 #endif // SHARE_CLASSFILE_FIELDLAYOUTBUILDER_HPP
|