Path: blob/master/src/hotspot/share/oops/fieldStreams.hpp
40951 views
/*1* Copyright (c) 2011, 2020, Oracle and/or its affiliates. All rights reserved.2* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.3*4* This code is free software; you can redistribute it and/or modify it5* under the terms of the GNU General Public License version 2 only, as6* published by the Free Software Foundation.7*8* This code is distributed in the hope that it will be useful, but WITHOUT9* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or10* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License11* version 2 for more details (a copy is included in the LICENSE file that12* accompanied this code).13*14* You should have received a copy of the GNU General Public License version15* 2 along with this work; if not, write to the Free Software Foundation,16* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.17*18* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA19* or visit www.oracle.com if you need additional information or have any20* questions.21*22*/2324#ifndef SHARE_OOPS_FIELDSTREAMS_HPP25#define SHARE_OOPS_FIELDSTREAMS_HPP2627#include "oops/instanceKlass.hpp"28#include "oops/fieldInfo.hpp"29#include "runtime/fieldDescriptor.hpp"3031// The is the base class for iteration over the fields array32// describing the declared fields in the class. Several subclasses33// are provided depending on the kind of iteration required. The34// JavaFieldStream is for iterating over regular Java fields and it35// generally the preferred iterator. InternalFieldStream only36// iterates over fields that have been injected by the JVM.37// AllFieldStream exposes all fields and should only be used in rare38// cases.39class FieldStreamBase : public StackObj {40protected:41Array<u2>* _fields;42constantPoolHandle _constants;43int _index;44int _limit;45int _generic_signature_slot;46fieldDescriptor _fd_buf;4748FieldInfo* field() const { return FieldInfo::from_field_array(_fields, _index); }4950int init_generic_signature_start_slot() {51int length = _fields->length();52int num_fields = _index;53int skipped_generic_signature_slots = 0;54FieldInfo* fi;55AccessFlags flags;56/* Scan from 0 to the current _index. Count the number of generic57signature slots for field[0] to field[_index - 1]. */58for (int i = 0; i < _index; i++) {59fi = FieldInfo::from_field_array(_fields, i);60flags.set_flags(fi->access_flags());61if (flags.field_has_generic_signature()) {62length --;63skipped_generic_signature_slots ++;64}65}66/* Scan from the current _index. */67for (int i = _index; i*FieldInfo::field_slots < length; i++) {68fi = FieldInfo::from_field_array(_fields, i);69flags.set_flags(fi->access_flags());70if (flags.field_has_generic_signature()) {71length --;72}73num_fields ++;74}75_generic_signature_slot = length + skipped_generic_signature_slots;76assert(_generic_signature_slot <= _fields->length(), "");77return num_fields;78}7980inline FieldStreamBase(Array<u2>* fields, ConstantPool* constants, int start, int limit);8182inline FieldStreamBase(Array<u2>* fields, ConstantPool* constants);83public:84inline FieldStreamBase(InstanceKlass* klass);8586// accessors87int index() const { return _index; }88InstanceKlass* field_holder() const { return _constants->pool_holder(); }8990void next() {91if (access_flags().field_has_generic_signature()) {92_generic_signature_slot ++;93assert(_generic_signature_slot <= _fields->length(), "");94}95_index += 1;96}97bool done() const { return _index >= _limit; }9899// Accessors for current field100AccessFlags access_flags() const {101AccessFlags flags;102flags.set_flags(field()->access_flags());103return flags;104}105106void set_access_flags(u2 flags) const {107field()->set_access_flags(flags);108}109110void set_access_flags(AccessFlags flags) const {111set_access_flags(flags.as_short());112}113114Symbol* name() const {115return field()->name(_constants());116}117118Symbol* signature() const {119return field()->signature(_constants());120}121122Symbol* generic_signature() const {123if (access_flags().field_has_generic_signature()) {124assert(_generic_signature_slot < _fields->length(), "out of bounds");125int index = _fields->at(_generic_signature_slot);126return _constants->symbol_at(index);127} else {128return NULL;129}130}131132int offset() const {133return field()->offset();134}135136void set_offset(int offset) {137field()->set_offset(offset);138}139140bool is_offset_set() const {141return field()->is_offset_set();142}143144bool is_contended() const {145return field()->is_contended();146}147148int contended_group() const {149return field()->contended_group();150}151152// bridge to a heavier API:153fieldDescriptor& field_descriptor() const {154fieldDescriptor& field = const_cast<fieldDescriptor&>(_fd_buf);155field.reinitialize(field_holder(), _index);156return field;157}158};159160// Iterate over only the internal fields161class JavaFieldStream : public FieldStreamBase {162public:163JavaFieldStream(const InstanceKlass* k): FieldStreamBase(k->fields(), k->constants(), 0, k->java_fields_count()) {}164165int name_index() const {166assert(!field()->is_internal(), "regular only");167return field()->name_index();168}169void set_name_index(int index) {170assert(!field()->is_internal(), "regular only");171field()->set_name_index(index);172}173int signature_index() const {174assert(!field()->is_internal(), "regular only");175return field()->signature_index();176}177void set_signature_index(int index) {178assert(!field()->is_internal(), "regular only");179field()->set_signature_index(index);180}181int generic_signature_index() const {182assert(!field()->is_internal(), "regular only");183if (access_flags().field_has_generic_signature()) {184assert(_generic_signature_slot < _fields->length(), "out of bounds");185return _fields->at(_generic_signature_slot);186} else {187return 0;188}189}190void set_generic_signature_index(int index) {191assert(!field()->is_internal(), "regular only");192if (access_flags().field_has_generic_signature()) {193assert(_generic_signature_slot < _fields->length(), "out of bounds");194_fields->at_put(_generic_signature_slot, index);195}196}197int initval_index() const {198assert(!field()->is_internal(), "regular only");199return field()->initval_index();200}201void set_initval_index(int index) {202assert(!field()->is_internal(), "regular only");203return field()->set_initval_index(index);204}205};206207208// Iterate over only the internal fields209class InternalFieldStream : public FieldStreamBase {210public:211InternalFieldStream(InstanceKlass* k): FieldStreamBase(k->fields(), k->constants(), k->java_fields_count(), 0) {}212};213214215class AllFieldStream : public FieldStreamBase {216public:217AllFieldStream(Array<u2>* fields, ConstantPool* constants): FieldStreamBase(fields, constants) {}218AllFieldStream(InstanceKlass* k): FieldStreamBase(k->fields(), k->constants()) {}219};220221#endif // SHARE_OOPS_FIELDSTREAMS_HPP222223224