src/share/vm/oops/fieldStreams.hpp

Tue, 26 Nov 2013 14:35:38 +0100

author
sjohanss
date
Tue, 26 Nov 2013 14:35:38 +0100
changeset 6148
55a0da3d420b
parent 5732
b2e698d2276c
child 6876
710a3c8b516e
permissions
-rw-r--r--

8027675: Full collections with Serial slower in JDK 8 compared to 7u40
Summary: Reduced the number of calls to follow_class_loader and instead marked and pushed the klass holder directly. Also removed unneeded calls to adjust_klass.
Reviewed-by: coleenp, jmasa, mgerdin, tschatzl

never@3137 1 /*
drchase@5732 2 * Copyright (c) 2011, 2013, Oracle and/or its affiliates. All rights reserved.
never@3137 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
never@3137 4 *
never@3137 5 * This code is free software; you can redistribute it and/or modify it
never@3137 6 * under the terms of the GNU General Public License version 2 only, as
never@3137 7 * published by the Free Software Foundation.
never@3137 8 *
never@3137 9 * This code is distributed in the hope that it will be useful, but WITHOUT
never@3137 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
never@3137 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
never@3137 12 * version 2 for more details (a copy is included in the LICENSE file that
never@3137 13 * accompanied this code).
never@3137 14 *
never@3137 15 * You should have received a copy of the GNU General Public License version
never@3137 16 * 2 along with this work; if not, write to the Free Software Foundation,
never@3137 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
never@3137 18 *
never@3137 19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
never@3137 20 * or visit www.oracle.com if you need additional information or have any
never@3137 21 * questions.
never@3137 22 *
never@3137 23 */
never@3137 24
never@3137 25 #ifndef SHARE_VM_OOPS_FIELDSTREAMS_HPP
never@3137 26 #define SHARE_VM_OOPS_FIELDSTREAMS_HPP
never@3137 27
never@3137 28 #include "oops/instanceKlass.hpp"
never@3137 29 #include "oops/fieldInfo.hpp"
drchase@5732 30 #include "runtime/fieldDescriptor.hpp"
never@3137 31
never@3137 32 // The is the base class for iteration over the fields array
never@3137 33 // describing the declared fields in the class. Several subclasses
never@3137 34 // are provided depending on the kind of iteration required. The
never@3137 35 // JavaFieldStream is for iterating over regular Java fields and it
never@3137 36 // generally the preferred iterator. InternalFieldStream only
never@3137 37 // iterates over fields that have been injected by the JVM.
never@3137 38 // AllFieldStream exposes all fields and should only be used in rare
never@3137 39 // cases.
never@3137 40 class FieldStreamBase : public StackObj {
never@3137 41 protected:
coleenp@4037 42 Array<u2>* _fields;
never@3137 43 constantPoolHandle _constants;
never@3137 44 int _index;
never@3137 45 int _limit;
jiangli@3803 46 int _generic_signature_slot;
drchase@5732 47 fieldDescriptor _fd_buf;
never@3137 48
coleenp@4037 49 FieldInfo* field() const { return FieldInfo::from_field_array(_fields, _index); }
drchase@5732 50 InstanceKlass* field_holder() const { return _constants->pool_holder(); }
never@3137 51
jiangli@3803 52 int init_generic_signature_start_slot() {
jiangli@3803 53 int length = _fields->length();
jiangli@3803 54 int num_fields = 0;
jiangli@3803 55 int skipped_generic_signature_slots = 0;
jiangli@3803 56 FieldInfo* fi;
jiangli@3803 57 AccessFlags flags;
jiangli@3803 58 /* Scan from 0 to the current _index. Count the number of generic
jiangli@3803 59 signature slots for field[0] to field[_index - 1]. */
jiangli@3803 60 for (int i = 0; i < _index; i++) {
coleenp@4037 61 fi = FieldInfo::from_field_array(_fields, i);
jiangli@3803 62 flags.set_flags(fi->access_flags());
jiangli@3803 63 if (flags.field_has_generic_signature()) {
jiangli@3803 64 length --;
jiangli@3803 65 skipped_generic_signature_slots ++;
jiangli@3803 66 }
jiangli@3803 67 }
jiangli@3803 68 /* Scan from the current _index. */
jiangli@3803 69 for (int i = _index; i*FieldInfo::field_slots < length; i++) {
coleenp@4037 70 fi = FieldInfo::from_field_array(_fields, i);
jiangli@3803 71 flags.set_flags(fi->access_flags());
jiangli@3803 72 if (flags.field_has_generic_signature()) {
jiangli@3803 73 length --;
jiangli@3803 74 }
jiangli@3803 75 num_fields ++;
jiangli@3803 76 }
jiangli@3803 77 _generic_signature_slot = length + skipped_generic_signature_slots;
jiangli@3803 78 assert(_generic_signature_slot <= _fields->length(), "");
jiangli@3803 79 return num_fields;
jiangli@3803 80 }
jiangli@3803 81
coleenp@4037 82 FieldStreamBase(Array<u2>* fields, constantPoolHandle constants, int start, int limit) {
never@3137 83 _fields = fields;
never@3137 84 _constants = constants;
never@3137 85 _index = start;
jiangli@3803 86 int num_fields = init_generic_signature_start_slot();
jiangli@3803 87 if (limit < start) {
jiangli@3803 88 _limit = num_fields;
jiangli@3803 89 } else {
jiangli@3803 90 _limit = limit;
jiangli@3803 91 }
never@3137 92 }
never@3137 93
coleenp@4037 94 FieldStreamBase(Array<u2>* fields, constantPoolHandle constants) {
never@3137 95 _fields = fields;
never@3137 96 _constants = constants;
never@3137 97 _index = 0;
jiangli@3803 98 _limit = init_generic_signature_start_slot();
never@3137 99 }
never@3137 100
never@3137 101 public:
coleenp@4037 102 FieldStreamBase(InstanceKlass* klass) {
never@3137 103 _fields = klass->fields();
never@3137 104 _constants = klass->constants();
never@3137 105 _index = 0;
never@3137 106 _limit = klass->java_fields_count();
jiangli@3803 107 init_generic_signature_start_slot();
drchase@5732 108 assert(klass == field_holder(), "");
never@3137 109 }
never@3137 110 FieldStreamBase(instanceKlassHandle klass) {
never@3137 111 _fields = klass->fields();
never@3137 112 _constants = klass->constants();
never@3137 113 _index = 0;
never@3137 114 _limit = klass->java_fields_count();
jiangli@3803 115 init_generic_signature_start_slot();
drchase@5732 116 assert(klass == field_holder(), "");
never@3137 117 }
never@3137 118
never@3137 119 // accessors
never@3137 120 int index() const { return _index; }
never@3137 121
jiangli@3803 122 void next() {
jiangli@3803 123 if (access_flags().field_has_generic_signature()) {
jiangli@3803 124 _generic_signature_slot ++;
jiangli@3803 125 assert(_generic_signature_slot <= _fields->length(), "");
jiangli@3803 126 }
jiangli@3803 127 _index += 1;
jiangli@3803 128 }
never@3137 129 bool done() const { return _index >= _limit; }
never@3137 130
never@3137 131 // Accessors for current field
never@3137 132 AccessFlags access_flags() const {
never@3137 133 AccessFlags flags;
never@3137 134 flags.set_flags(field()->access_flags());
never@3137 135 return flags;
never@3137 136 }
never@3137 137
never@3137 138 void set_access_flags(u2 flags) const {
never@3137 139 field()->set_access_flags(flags);
never@3137 140 }
never@3137 141
never@3137 142 void set_access_flags(AccessFlags flags) const {
never@3137 143 set_access_flags(flags.as_short());
never@3137 144 }
never@3137 145
never@3137 146 Symbol* name() const {
never@3137 147 return field()->name(_constants);
never@3137 148 }
never@3137 149
never@3137 150 Symbol* signature() const {
never@3137 151 return field()->signature(_constants);
never@3137 152 }
never@3137 153
never@3137 154 Symbol* generic_signature() const {
jiangli@3803 155 if (access_flags().field_has_generic_signature()) {
jiangli@3803 156 assert(_generic_signature_slot < _fields->length(), "out of bounds");
coleenp@4037 157 int index = _fields->at(_generic_signature_slot);
jiangli@3803 158 return _constants->symbol_at(index);
jiangli@3803 159 } else {
jiangli@3803 160 return NULL;
jiangli@3803 161 }
never@3137 162 }
never@3137 163
never@3137 164 int offset() const {
never@3137 165 return field()->offset();
never@3137 166 }
never@3137 167
jwilhelm@4430 168 int allocation_type() const {
jwilhelm@4430 169 return field()->allocation_type();
jwilhelm@4430 170 }
jwilhelm@4430 171
never@3137 172 void set_offset(int offset) {
never@3137 173 field()->set_offset(offset);
never@3137 174 }
jwilhelm@4430 175
jwilhelm@4430 176 bool is_offset_set() const {
jwilhelm@4430 177 return field()->is_offset_set();
jwilhelm@4430 178 }
jwilhelm@4430 179
jwilhelm@4430 180 bool is_contended() const {
jwilhelm@4430 181 return field()->is_contended();
jwilhelm@4430 182 }
jwilhelm@4430 183
jwilhelm@4430 184 int contended_group() const {
jwilhelm@4430 185 return field()->contended_group();
jwilhelm@4430 186 }
jwilhelm@4430 187
drchase@5732 188 // bridge to a heavier API:
drchase@5732 189 fieldDescriptor& field_descriptor() const {
drchase@5732 190 fieldDescriptor& field = const_cast<fieldDescriptor&>(_fd_buf);
drchase@5732 191 field.reinitialize(field_holder(), _index);
drchase@5732 192 return field;
drchase@5732 193 }
never@3137 194 };
never@3137 195
never@3137 196 // Iterate over only the internal fields
never@3137 197 class JavaFieldStream : public FieldStreamBase {
never@3137 198 public:
never@3137 199 JavaFieldStream(instanceKlassHandle k): FieldStreamBase(k->fields(), k->constants(), 0, k->java_fields_count()) {}
never@3137 200
never@3137 201 int name_index() const {
never@3137 202 assert(!field()->is_internal(), "regular only");
never@3137 203 return field()->name_index();
never@3137 204 }
never@3137 205 void set_name_index(int index) {
never@3137 206 assert(!field()->is_internal(), "regular only");
never@3137 207 field()->set_name_index(index);
never@3137 208 }
never@3137 209 int signature_index() const {
never@3137 210 assert(!field()->is_internal(), "regular only");
never@3137 211 return field()->signature_index();
never@3137 212 }
never@3137 213 void set_signature_index(int index) {
never@3137 214 assert(!field()->is_internal(), "regular only");
never@3137 215 field()->set_signature_index(index);
never@3137 216 }
never@3137 217 int generic_signature_index() const {
never@3137 218 assert(!field()->is_internal(), "regular only");
jiangli@3803 219 if (access_flags().field_has_generic_signature()) {
jiangli@3803 220 assert(_generic_signature_slot < _fields->length(), "out of bounds");
coleenp@4037 221 return _fields->at(_generic_signature_slot);
jiangli@3803 222 } else {
jiangli@3803 223 return 0;
jiangli@3803 224 }
never@3137 225 }
never@3137 226 void set_generic_signature_index(int index) {
never@3137 227 assert(!field()->is_internal(), "regular only");
jiangli@3803 228 if (access_flags().field_has_generic_signature()) {
jiangli@3803 229 assert(_generic_signature_slot < _fields->length(), "out of bounds");
coleenp@4037 230 _fields->at_put(_generic_signature_slot, index);
jiangli@3803 231 }
never@3137 232 }
never@3137 233 int initval_index() const {
never@3137 234 assert(!field()->is_internal(), "regular only");
never@3137 235 return field()->initval_index();
never@3137 236 }
never@3137 237 void set_initval_index(int index) {
never@3137 238 assert(!field()->is_internal(), "regular only");
never@3137 239 return field()->set_initval_index(index);
never@3137 240 }
never@3137 241 };
never@3137 242
never@3137 243
never@3137 244 // Iterate over only the internal fields
never@3137 245 class InternalFieldStream : public FieldStreamBase {
never@3137 246 public:
coleenp@4037 247 InternalFieldStream(InstanceKlass* k): FieldStreamBase(k->fields(), k->constants(), k->java_fields_count(), 0) {}
jiangli@3803 248 InternalFieldStream(instanceKlassHandle k): FieldStreamBase(k->fields(), k->constants(), k->java_fields_count(), 0) {}
never@3137 249 };
never@3137 250
never@3137 251
never@3137 252 class AllFieldStream : public FieldStreamBase {
never@3137 253 public:
coleenp@4037 254 AllFieldStream(Array<u2>* fields, constantPoolHandle constants): FieldStreamBase(fields, constants) {}
coleenp@4037 255 AllFieldStream(InstanceKlass* k): FieldStreamBase(k->fields(), k->constants()) {}
never@3137 256 AllFieldStream(instanceKlassHandle k): FieldStreamBase(k->fields(), k->constants()) {}
never@3137 257 };
never@3137 258
never@3137 259 #endif // SHARE_VM_OOPS_FIELDSTREAMS_HPP

mercurial