1/*
2 * Copyright (c) 2011, 2017, 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 */
23package jdk.vm.ci.hotspot;
24
25import static java.util.Objects.requireNonNull;
26import static jdk.vm.ci.hotspot.CompilerToVM.compilerToVM;
27import static jdk.vm.ci.hotspot.HotSpotConstantPool.isSignaturePolymorphicHolder;
28import static jdk.vm.ci.hotspot.HotSpotJVMCIRuntime.runtime;
29import static jdk.vm.ci.hotspot.HotSpotModifiers.jvmClassModifiers;
30import static jdk.vm.ci.hotspot.HotSpotVMConfig.config;
31import static jdk.vm.ci.hotspot.UnsafeAccess.UNSAFE;
32
33import java.lang.annotation.Annotation;
34import java.lang.reflect.Array;
35import java.lang.reflect.Constructor;
36import java.lang.reflect.Method;
37import java.lang.reflect.Modifier;
38import java.nio.ByteOrder;
39import java.util.HashMap;
40
41import jdk.vm.ci.common.JVMCIError;
42import jdk.vm.ci.meta.Assumptions.AssumptionResult;
43import jdk.vm.ci.meta.Assumptions.ConcreteMethod;
44import jdk.vm.ci.meta.Assumptions.ConcreteSubtype;
45import jdk.vm.ci.meta.Assumptions.LeafType;
46import jdk.vm.ci.meta.Assumptions.NoFinalizableSubclass;
47import jdk.vm.ci.meta.Constant;
48import jdk.vm.ci.meta.JavaConstant;
49import jdk.vm.ci.meta.JavaKind;
50import jdk.vm.ci.meta.JavaType;
51import jdk.vm.ci.meta.ResolvedJavaField;
52import jdk.vm.ci.meta.ResolvedJavaMethod;
53import jdk.vm.ci.meta.ResolvedJavaType;
54
55/**
56 * Implementation of {@link JavaType} for resolved non-primitive HotSpot classes.
57 */
58final class HotSpotResolvedObjectTypeImpl extends HotSpotResolvedJavaType implements HotSpotResolvedObjectType, MetaspaceWrapperObject {
59
60    private static final HotSpotResolvedJavaField[] NO_FIELDS = new HotSpotResolvedJavaField[0];
61    private static final int METHOD_CACHE_ARRAY_CAPACITY = 8;
62
63    /**
64     * The Java class this type represents.
65     */
66    private final Class<?> javaClass;
67    private HotSpotResolvedJavaMethodImpl[] methodCacheArray;
68    private HashMap<Long, HotSpotResolvedJavaMethodImpl> methodCacheHashMap;
69    private HotSpotResolvedJavaField[] instanceFields;
70    private HotSpotResolvedObjectTypeImpl[] interfaces;
71    private HotSpotConstantPool constantPool;
72    final HotSpotJVMCIMetaAccessContext context;
73    private HotSpotResolvedObjectType arrayOfType;
74
75    /**
76     * Gets the JVMCI mirror for a {@link Class} object.
77     *
78     * @return the {@link HotSpotResolvedJavaType} corresponding to {@code javaClass}
79     */
80    static HotSpotResolvedObjectTypeImpl fromObjectClass(Class<?> javaClass) {
81        return (HotSpotResolvedObjectTypeImpl) runtime().fromClass(javaClass);
82    }
83
84    /**
85     * Gets the JVMCI mirror from a HotSpot type. Since {@link Class} is already a proxy for the
86     * underlying Klass*, it is used instead of the raw Klass*.
87     *
88     * Called from the VM.
89     *
90     * @param javaClass a {@link Class} object
91     * @return the {@link ResolvedJavaType} corresponding to {@code javaClass}
92     */
93    @SuppressWarnings("unused")
94    private static HotSpotResolvedObjectTypeImpl fromMetaspace(Class<?> javaClass) {
95        return fromObjectClass(javaClass);
96    }
97
98    /**
99     * Creates the JVMCI mirror for a {@link Class} object.
100     *
101     * <p>
102     * <b>NOTE</b>: Creating an instance of this class does not install the mirror for the
103     * {@link Class} type. Use {@link #fromObjectClass(Class)} or {@link #fromMetaspace(Class)}
104     * instead.
105     * </p>
106     *
107     * @param javaClass the Class to create the mirror for
108     * @param context
109     */
110    HotSpotResolvedObjectTypeImpl(Class<?> javaClass, HotSpotJVMCIMetaAccessContext context) {
111        super(getSignatureName(javaClass));
112        this.javaClass = javaClass;
113        this.context = context;
114        assert getName().charAt(0) != '[' || isArray() : getName();
115    }
116
117    /**
118     * Returns the name of this type as it would appear in a signature.
119     */
120    private static String getSignatureName(Class<?> javaClass) {
121        if (javaClass.isArray()) {
122            return javaClass.getName().replace('.', '/');
123        }
124        return "L" + javaClass.getName().replace('.', '/') + ";";
125    }
126
127    /**
128     * Gets the metaspace Klass for this type.
129     */
130    long getMetaspaceKlass() {
131        if (HotSpotJVMCIRuntime.getHostWordKind() == JavaKind.Long) {
132            return UNSAFE.getLong(javaClass, config().klassOffset);
133        }
134        return UNSAFE.getInt(javaClass, config().klassOffset) & 0xFFFFFFFFL;
135    }
136
137    @Override
138    public long getMetaspacePointer() {
139        return getMetaspaceKlass();
140    }
141
142    /**
143     * The Klass* for this object is kept alive by the direct reference to {@link #javaClass} so no
144     * extra work is required.
145     */
146    @Override
147    public boolean isRegistered() {
148        return true;
149    }
150
151    @Override
152    public int getModifiers() {
153        if (isArray()) {
154            return (getElementalType().getModifiers() & (Modifier.PUBLIC | Modifier.PRIVATE | Modifier.PROTECTED)) | Modifier.FINAL | Modifier.ABSTRACT;
155        } else {
156            return getAccessFlags() & jvmClassModifiers();
157        }
158    }
159
160    public int getAccessFlags() {
161        HotSpotVMConfig config = config();
162        return UNSAFE.getInt(getMetaspaceKlass() + config.klassAccessFlagsOffset);
163    }
164
165    @Override
166    public HotSpotResolvedObjectType getArrayClass() {
167        if (arrayOfType == null) {
168            arrayOfType = fromObjectClass(Array.newInstance(mirror(), 0).getClass());
169        }
170        return arrayOfType;
171    }
172
173    @Override
174    public ResolvedJavaType getComponentType() {
175        Class<?> javaComponentType = mirror().getComponentType();
176        return javaComponentType == null ? null : runtime().fromClass(javaComponentType);
177    }
178
179    @Override
180    public AssumptionResult<ResolvedJavaType> findLeafConcreteSubtype() {
181        if (isLeaf()) {
182            // No assumptions are required.
183            return new AssumptionResult<>(this);
184        }
185        HotSpotVMConfig config = config();
186        if (isArray()) {
187            ResolvedJavaType elementalType = getElementalType();
188            AssumptionResult<ResolvedJavaType> elementType = elementalType.findLeafConcreteSubtype();
189            if (elementType != null && elementType.getResult().equals(elementalType)) {
190                /*
191                 * If the elementType is leaf then the array is leaf under the same assumptions but
192                 * only if the element type is exactly the leaf type. The element type can be
193                 * abstract even if there is only one implementor of the abstract type.
194                 */
195                AssumptionResult<ResolvedJavaType> result = new AssumptionResult<>(this);
196                result.add(elementType);
197                return result;
198            }
199            return null;
200        } else if (isInterface()) {
201            HotSpotResolvedObjectTypeImpl implementor = getSingleImplementor();
202            /*
203             * If the implementor field contains itself that indicates that the interface has more
204             * than one implementors (see: InstanceKlass::add_implementor).
205             */
206            if (implementor == null || implementor.equals(this)) {
207                return null;
208            }
209
210            assert !implementor.isInterface();
211            if (implementor.isAbstract() || !implementor.isLeafClass()) {
212                AssumptionResult<ResolvedJavaType> leafConcreteSubtype = implementor.findLeafConcreteSubtype();
213                if (leafConcreteSubtype != null) {
214                    assert !leafConcreteSubtype.getResult().equals(implementor);
215                    AssumptionResult<ResolvedJavaType> newResult = new AssumptionResult<>(leafConcreteSubtype.getResult(), new ConcreteSubtype(this, implementor));
216                    // Accumulate leaf assumptions and return the combined result.
217                    newResult.add(leafConcreteSubtype);
218                    return newResult;
219                }
220                return null;
221            }
222            return concreteSubtype(implementor);
223        } else {
224            HotSpotResolvedObjectTypeImpl type = this;
225            while (type.isAbstract()) {
226                HotSpotResolvedObjectTypeImpl subklass = type.getSubklass();
227                if (subklass == null || UNSAFE.getAddress(subklass.getMetaspaceKlass() + config.nextSiblingOffset) != 0) {
228                    return null;
229                }
230                type = subklass;
231            }
232            if (type.isAbstract() || type.isInterface() || !type.isLeafClass()) {
233                return null;
234            }
235            if (this.isAbstract()) {
236                return concreteSubtype(type);
237            } else {
238                assert this.equals(type);
239                return new AssumptionResult<>(type, new LeafType(type));
240            }
241        }
242    }
243
244    private AssumptionResult<ResolvedJavaType> concreteSubtype(HotSpotResolvedObjectTypeImpl type) {
245        if (type.isLeaf()) {
246            return new AssumptionResult<>(type, new ConcreteSubtype(this, type));
247        } else {
248            return new AssumptionResult<>(type, new LeafType(type), new ConcreteSubtype(this, type));
249        }
250    }
251
252    /**
253     * Returns if type {@code type} is a leaf class. This is the case if the
254     * {@code Klass::_subklass} field of the underlying class is zero.
255     *
256     * @return true if the type is a leaf class
257     */
258    private boolean isLeafClass() {
259        return UNSAFE.getLong(this.getMetaspaceKlass() + config().subklassOffset) == 0;
260    }
261
262    /**
263     * Returns the {@code Klass::_subklass} field of the underlying metaspace klass for the given
264     * type {@code type}.
265     *
266     * @return value of the subklass field as metaspace klass pointer
267     */
268    private HotSpotResolvedObjectTypeImpl getSubklass() {
269        return compilerToVM().getResolvedJavaType(this, config().subklassOffset, false);
270    }
271
272    @Override
273    public HotSpotResolvedObjectTypeImpl getSuperclass() {
274        Class<?> javaSuperclass = mirror().getSuperclass();
275        return javaSuperclass == null ? null : fromObjectClass(javaSuperclass);
276    }
277
278    @Override
279    public HotSpotResolvedObjectTypeImpl[] getInterfaces() {
280        if (interfaces == null) {
281            Class<?>[] javaInterfaces = mirror().getInterfaces();
282            HotSpotResolvedObjectTypeImpl[] result = new HotSpotResolvedObjectTypeImpl[javaInterfaces.length];
283            for (int i = 0; i < javaInterfaces.length; i++) {
284                result[i] = fromObjectClass(javaInterfaces[i]);
285            }
286            interfaces = result;
287        }
288        return interfaces;
289    }
290
291    @Override
292    public HotSpotResolvedObjectTypeImpl getSingleImplementor() {
293        if (!isInterface()) {
294            throw new JVMCIError("Cannot call getSingleImplementor() on a non-interface type: %s", this);
295        }
296        return compilerToVM().getImplementor(this);
297    }
298
299    public HotSpotResolvedObjectTypeImpl getSupertype() {
300        if (isArray()) {
301            ResolvedJavaType componentType = getComponentType();
302            if (mirror() == Object[].class || componentType.isPrimitive()) {
303                return fromObjectClass(Object.class);
304            }
305            return (HotSpotResolvedObjectTypeImpl) ((HotSpotResolvedObjectTypeImpl) componentType).getSupertype().getArrayClass();
306        }
307        if (isInterface()) {
308            return fromObjectClass(Object.class);
309        }
310        return getSuperclass();
311    }
312
313    @Override
314    public HotSpotResolvedObjectType findLeastCommonAncestor(ResolvedJavaType otherType) {
315        if (otherType.isPrimitive()) {
316            return null;
317        } else {
318            HotSpotResolvedObjectTypeImpl t1 = this;
319            HotSpotResolvedObjectTypeImpl t2 = (HotSpotResolvedObjectTypeImpl) otherType;
320            while (true) {
321                if (t1.isAssignableFrom(t2)) {
322                    return t1;
323                }
324                if (t2.isAssignableFrom(t1)) {
325                    return t2;
326                }
327                t1 = t1.getSupertype();
328                t2 = t2.getSupertype();
329            }
330        }
331    }
332
333    @Override
334    public AssumptionResult<Boolean> hasFinalizableSubclass() {
335        assert !isArray();
336        if (!compilerToVM().hasFinalizableSubclass(this)) {
337            return new AssumptionResult<>(false, new NoFinalizableSubclass(this));
338        }
339        return new AssumptionResult<>(true);
340    }
341
342    @Override
343    public boolean hasFinalizer() {
344        return (getAccessFlags() & config().jvmAccHasFinalizer) != 0;
345    }
346
347    @Override
348    public boolean isPrimitive() {
349        return false;
350    }
351
352    @Override
353    public boolean isArray() {
354        return mirror().isArray();
355    }
356
357    @Override
358    public boolean isInitialized() {
359        return isArray() ? true : getInitState() == config().instanceKlassStateFullyInitialized;
360    }
361
362    @Override
363    public boolean isLinked() {
364        return isArray() ? true : getInitState() >= config().instanceKlassStateLinked;
365    }
366
367    /**
368     * Returns the value of the state field {@code InstanceKlass::_init_state} of the metaspace
369     * klass.
370     *
371     * @return state field value of this type
372     */
373    private int getInitState() {
374        assert !isArray() : "_init_state only exists in InstanceKlass";
375        return UNSAFE.getByte(getMetaspaceKlass() + config().instanceKlassInitStateOffset) & 0xFF;
376    }
377
378    @Override
379    public void initialize() {
380        if (!isInitialized()) {
381            UNSAFE.ensureClassInitialized(mirror());
382            assert isInitialized();
383        }
384    }
385
386    @Override
387    public boolean isInstance(JavaConstant obj) {
388        if (obj.getJavaKind() == JavaKind.Object && !obj.isNull()) {
389            return mirror().isInstance(((HotSpotObjectConstantImpl) obj).object());
390        }
391        return false;
392    }
393
394    @Override
395    public boolean isInstanceClass() {
396        return !isArray() && !isInterface();
397    }
398
399    @Override
400    public boolean isInterface() {
401        return mirror().isInterface();
402    }
403
404    @Override
405    public boolean isAssignableFrom(ResolvedJavaType other) {
406        assert other != null;
407        if (other instanceof HotSpotResolvedObjectTypeImpl) {
408            HotSpotResolvedObjectTypeImpl otherType = (HotSpotResolvedObjectTypeImpl) other;
409            return mirror().isAssignableFrom(otherType.mirror());
410        }
411        return false;
412    }
413
414    @Override
415    public ResolvedJavaType getHostClass() {
416        if (isArray()) {
417            return null;
418        }
419        return compilerToVM().getHostClass(this);
420    }
421
422    @Override
423    public boolean isJavaLangObject() {
424        return javaClass.equals(Object.class);
425    }
426
427    @Override
428    public JavaKind getJavaKind() {
429        return JavaKind.Object;
430    }
431
432    @Override
433    public ResolvedJavaMethod resolveMethod(ResolvedJavaMethod method, ResolvedJavaType callerType) {
434        assert !callerType.isArray();
435        if (isInterface()) {
436            // Methods can only be resolved against concrete types
437            return null;
438        }
439        if (method.isConcrete() && method.getDeclaringClass().equals(this) && method.isPublic() && !isSignaturePolymorphicHolder(method.getDeclaringClass())) {
440            return method;
441        }
442        if (!method.getDeclaringClass().isAssignableFrom(this)) {
443            return null;
444        }
445        HotSpotResolvedJavaMethodImpl hotSpotMethod = (HotSpotResolvedJavaMethodImpl) method;
446        HotSpotResolvedObjectTypeImpl hotSpotCallerType = (HotSpotResolvedObjectTypeImpl) callerType;
447        return compilerToVM().resolveMethod(this, hotSpotMethod, hotSpotCallerType);
448    }
449
450    public HotSpotConstantPool getConstantPool() {
451        if (constantPool == null || !isArray() && UNSAFE.getAddress(getMetaspaceKlass() + config().instanceKlassConstantsOffset) != constantPool.getMetaspaceConstantPool()) {
452            /*
453             * If the pointer to the ConstantPool has changed since this was last read refresh the
454             * HotSpotConstantPool wrapper object. This ensures that uses of the constant pool are
455             * operating on the latest one and that HotSpotResolvedJavaMethodImpls will be able to
456             * use the shared copy instead of creating their own instance.
457             */
458            constantPool = compilerToVM().getConstantPool(this);
459        }
460        return constantPool;
461    }
462
463    /**
464     * Gets the instance size of this type. If an instance of this type cannot be fast path
465     * allocated, then the returned value is negative (its absolute value gives the size). Must not
466     * be called if this is an array or interface type.
467     */
468    public int instanceSize() {
469        assert !isArray();
470        assert !isInterface();
471
472        HotSpotVMConfig config = config();
473        final int layoutHelper = layoutHelper();
474        assert layoutHelper > config.klassLayoutHelperNeutralValue : "must be instance";
475
476        // See: Klass::layout_helper_size_in_bytes
477        int size = layoutHelper & ~config.klassLayoutHelperInstanceSlowPathBit;
478
479        // See: Klass::layout_helper_needs_slow_path
480        boolean needsSlowPath = (layoutHelper & config.klassLayoutHelperInstanceSlowPathBit) != 0;
481
482        return needsSlowPath ? -size : size;
483    }
484
485    public int layoutHelper() {
486        HotSpotVMConfig config = config();
487        return UNSAFE.getInt(getMetaspaceKlass() + config.klassLayoutHelperOffset);
488    }
489
490    @Override
491    public long getFingerprint() {
492        return compilerToVM().getFingerprint(getMetaspaceKlass());
493    }
494
495    synchronized HotSpotResolvedJavaMethod createMethod(long metaspaceMethod) {
496        // Maintain cache as array.
497        if (methodCacheArray == null) {
498            methodCacheArray = new HotSpotResolvedJavaMethodImpl[METHOD_CACHE_ARRAY_CAPACITY];
499        }
500
501        int i = 0;
502        for (; i < methodCacheArray.length; ++i) {
503            HotSpotResolvedJavaMethodImpl curMethod = methodCacheArray[i];
504            if (curMethod == null) {
505                HotSpotResolvedJavaMethodImpl newMethod = new HotSpotResolvedJavaMethodImpl(this, metaspaceMethod);
506                methodCacheArray[i] = newMethod;
507                context.add(newMethod);
508                return newMethod;
509            } else if (curMethod.getMetaspacePointer() == metaspaceMethod) {
510                return curMethod;
511            }
512        }
513
514        // Fall-back to hash table.
515        if (methodCacheHashMap == null) {
516            methodCacheHashMap = new HashMap<>();
517        }
518
519        HotSpotResolvedJavaMethodImpl lookupResult = methodCacheHashMap.get(metaspaceMethod);
520        if (lookupResult == null) {
521            HotSpotResolvedJavaMethodImpl newMethod = new HotSpotResolvedJavaMethodImpl(this, metaspaceMethod);
522            methodCacheHashMap.put(metaspaceMethod, newMethod);
523            context.add(lookupResult);
524            return newMethod;
525        } else {
526            return lookupResult;
527        }
528    }
529
530    public int getVtableLength() {
531        HotSpotVMConfig config = config();
532        if (isInterface() || isArray()) {
533            /* Everything has the core vtable of java.lang.Object */
534            return config.baseVtableLength();
535        }
536        int result = UNSAFE.getInt(getMetaspaceKlass() + config.klassVtableLengthOffset) / (config.vtableEntrySize / config.heapWordSize);
537        assert result >= config.baseVtableLength() : UNSAFE.getInt(getMetaspaceKlass() + config.klassVtableLengthOffset) + " " + config.vtableEntrySize;
538        return result;
539    }
540
541    synchronized HotSpotResolvedJavaField createField(JavaType type, long offset, int rawFlags, int index) {
542        return new HotSpotResolvedJavaFieldImpl(this, type, offset, rawFlags, index);
543    }
544
545    @Override
546    public AssumptionResult<ResolvedJavaMethod> findUniqueConcreteMethod(ResolvedJavaMethod method) {
547        HotSpotResolvedJavaMethod hmethod = (HotSpotResolvedJavaMethod) method;
548        HotSpotResolvedObjectType declaredHolder = hmethod.getDeclaringClass();
549        /*
550         * Sometimes the receiver type in the graph hasn't stabilized to a subtype of declared
551         * holder, usually because of phis, so make sure that the type is related to the declared
552         * type before using it for lookup. Unlinked types should also be ignored because we can't
553         * resolve the proper method to invoke. Generally unlinked types in invokes should result in
554         * a deopt instead since they can't really be used if they aren't linked yet.
555         */
556        if (!declaredHolder.isAssignableFrom(this) || this.isArray() || this.equals(declaredHolder) || !isLinked() || isInterface()) {
557            ResolvedJavaMethod result = hmethod.uniqueConcreteMethod(declaredHolder);
558            if (result != null) {
559                return new AssumptionResult<>(result, new ConcreteMethod(method, declaredHolder, result));
560            }
561            return null;
562        }
563        /*
564         * The holder may be a subtype of the declaredHolder so make sure to resolve the method to
565         * the correct method for the subtype.
566         */
567        HotSpotResolvedJavaMethod resolvedMethod = (HotSpotResolvedJavaMethod) resolveMethod(hmethod, this);
568        if (resolvedMethod == null) {
569            // The type isn't known to implement the method.
570            return null;
571        }
572
573        ResolvedJavaMethod result = resolvedMethod.uniqueConcreteMethod(this);
574        if (result != null) {
575            return new AssumptionResult<>(result, new ConcreteMethod(method, this, result));
576        }
577        return null;
578    }
579
580    FieldInfo createFieldInfo(int index) {
581        return new FieldInfo(index);
582    }
583
584    /**
585     * This class represents the field information for one field contained in the fields array of an
586     * {@code InstanceKlass}. The implementation is similar to the native {@code FieldInfo} class.
587     */
588    class FieldInfo {
589        /**
590         * Native pointer into the array of Java shorts.
591         */
592        private final long metaspaceData;
593
594        /**
595         * Creates a field info for the field in the fields array at index {@code index}.
596         *
597         * @param index index to the fields array
598         */
599        FieldInfo(int index) {
600            HotSpotVMConfig config = config();
601            // Get Klass::_fields
602            final long metaspaceFields = UNSAFE.getAddress(getMetaspaceKlass() + config.instanceKlassFieldsOffset);
603            assert config.fieldInfoFieldSlots == 6 : "revisit the field parsing code";
604            int offset = config.fieldInfoFieldSlots * Short.BYTES * index;
605            metaspaceData = metaspaceFields + config.arrayU2DataOffset + offset;
606        }
607
608        private int getAccessFlags() {
609            return readFieldSlot(config().fieldInfoAccessFlagsOffset);
610        }
611
612        private int getNameIndex() {
613            return readFieldSlot(config().fieldInfoNameIndexOffset);
614        }
615
616        private int getSignatureIndex() {
617            return readFieldSlot(config().fieldInfoSignatureIndexOffset);
618        }
619
620        public int getOffset() {
621            HotSpotVMConfig config = config();
622            final int lowPacked = readFieldSlot(config.fieldInfoLowPackedOffset);
623            final int highPacked = readFieldSlot(config.fieldInfoHighPackedOffset);
624            final int offset = ((highPacked << Short.SIZE) | lowPacked) >> config.fieldInfoTagSize;
625            return offset;
626        }
627
628        /**
629         * Helper method to read an entry (slot) from the field array. Currently field info is laid
630         * on top an array of Java shorts.
631         */
632        private int readFieldSlot(int index) {
633            int offset = Short.BYTES * index;
634            return UNSAFE.getChar(metaspaceData + offset);
635        }
636
637        /**
638         * Returns the name of this field as a {@link String}. If the field is an internal field the
639         * name index is pointing into the vmSymbols table.
640         */
641        public String getName() {
642            final int nameIndex = getNameIndex();
643            return isInternal() ? config().symbolAt(nameIndex) : getConstantPool().lookupUtf8(nameIndex);
644        }
645
646        /**
647         * Returns the signature of this field as {@link String}. If the field is an internal field
648         * the signature index is pointing into the vmSymbols table.
649         */
650        public String getSignature() {
651            final int signatureIndex = getSignatureIndex();
652            return isInternal() ? config().symbolAt(signatureIndex) : getConstantPool().lookupUtf8(signatureIndex);
653        }
654
655        public JavaType getType() {
656            String signature = getSignature();
657            return runtime().lookupType(signature, HotSpotResolvedObjectTypeImpl.this, false);
658        }
659
660        private boolean isInternal() {
661            return (getAccessFlags() & config().jvmAccFieldInternal) != 0;
662        }
663
664        public boolean isStatic() {
665            return Modifier.isStatic(getAccessFlags());
666        }
667
668        public boolean hasGenericSignature() {
669            return (getAccessFlags() & config().jvmAccFieldHasGenericSignature) != 0;
670        }
671    }
672
673    @Override
674    public ResolvedJavaField[] getInstanceFields(boolean includeSuperclasses) {
675        if (instanceFields == null) {
676            if (isArray() || isInterface()) {
677                instanceFields = NO_FIELDS;
678            } else {
679                HotSpotResolvedJavaField[] prepend = NO_FIELDS;
680                if (getSuperclass() != null) {
681                    prepend = (HotSpotResolvedJavaField[]) getSuperclass().getInstanceFields(true);
682                }
683                instanceFields = getFields(false, prepend);
684            }
685        }
686        if (!includeSuperclasses && getSuperclass() != null) {
687            int superClassFieldCount = getSuperclass().getInstanceFields(true).length;
688            if (superClassFieldCount == instanceFields.length) {
689                // This class does not have any instance fields of its own.
690                return NO_FIELDS;
691            } else if (superClassFieldCount != 0) {
692                HotSpotResolvedJavaField[] result = new HotSpotResolvedJavaField[instanceFields.length - superClassFieldCount];
693                System.arraycopy(instanceFields, superClassFieldCount, result, 0, result.length);
694                return result;
695            } else {
696                // The super classes of this class do not have any instance fields.
697            }
698        }
699        return instanceFields;
700    }
701
702    @Override
703    public ResolvedJavaField[] getStaticFields() {
704        if (isArray()) {
705            return new HotSpotResolvedJavaField[0];
706        } else {
707            return getFields(true, NO_FIELDS);
708        }
709    }
710
711    /**
712     * Gets the instance or static fields of this class.
713     *
714     * @param retrieveStaticFields specifies whether to return instance or static fields
715     * @param prepend an array to be prepended to the returned result
716     */
717    private HotSpotResolvedJavaField[] getFields(boolean retrieveStaticFields, HotSpotResolvedJavaField[] prepend) {
718        HotSpotVMConfig config = config();
719        final long metaspaceFields = UNSAFE.getAddress(getMetaspaceKlass() + config.instanceKlassFieldsOffset);
720        int metaspaceFieldsLength = UNSAFE.getInt(metaspaceFields + config.arrayU1LengthOffset);
721        int resultCount = 0;
722        int index = 0;
723        for (int i = 0; i < metaspaceFieldsLength; i += config.fieldInfoFieldSlots, index++) {
724            FieldInfo field = new FieldInfo(index);
725            if (field.hasGenericSignature()) {
726                metaspaceFieldsLength--;
727            }
728
729            if (field.isStatic() == retrieveStaticFields) {
730                resultCount++;
731            }
732        }
733
734        if (resultCount == 0) {
735            return prepend;
736        }
737
738        int prependLength = prepend.length;
739        resultCount += prependLength;
740
741        HotSpotResolvedJavaField[] result = new HotSpotResolvedJavaField[resultCount];
742        if (prependLength != 0) {
743            System.arraycopy(prepend, 0, result, 0, prependLength);
744        }
745
746        int resultIndex = prependLength;
747        for (int i = 0; i < index; ++i) {
748            FieldInfo field = new FieldInfo(i);
749            if (field.isStatic() == retrieveStaticFields) {
750                int offset = field.getOffset();
751                HotSpotResolvedJavaField resolvedJavaField = createField(field.getType(), offset, field.getAccessFlags(), i);
752
753                // Make sure the result is sorted by offset.
754                int j;
755                for (j = resultIndex - 1; j >= prependLength && result[j].offset() > offset; j--) {
756                    result[j + 1] = result[j];
757                }
758                result[j + 1] = resolvedJavaField;
759                resultIndex++;
760            }
761        }
762
763        return result;
764    }
765
766    @Override
767    public Class<?> mirror() {
768        return javaClass;
769    }
770
771    @Override
772    public String getSourceFileName() {
773        HotSpotVMConfig config = config();
774        final int sourceFileNameIndex = UNSAFE.getChar(getMetaspaceKlass() + config.instanceKlassSourceFileNameIndexOffset);
775        if (sourceFileNameIndex == 0) {
776            return null;
777        }
778        return getConstantPool().lookupUtf8(sourceFileNameIndex);
779    }
780
781    @Override
782    public Annotation[] getAnnotations() {
783        return mirror().getAnnotations();
784    }
785
786    @Override
787    public Annotation[] getDeclaredAnnotations() {
788        return mirror().getDeclaredAnnotations();
789    }
790
791    @Override
792    public <T extends Annotation> T getAnnotation(Class<T> annotationClass) {
793        return mirror().getAnnotation(annotationClass);
794    }
795
796    /**
797     * Performs a fast-path check that this type is resolved in the context of a given accessing
798     * class. A negative result does not mean this type is not resolved with respect to
799     * {@code accessingClass}. That can only be determined by
800     * {@linkplain HotSpotJVMCIRuntime#lookupType(String, HotSpotResolvedObjectType, boolean)
801     * re-resolving} the type.
802     */
803    public boolean isDefinitelyResolvedWithRespectTo(ResolvedJavaType accessingClass) {
804        assert accessingClass != null;
805        ResolvedJavaType elementType = getElementalType();
806        if (elementType.isPrimitive()) {
807            // Primitive type resolution is context free.
808            return true;
809        }
810        if (elementType.getName().startsWith("Ljava/")) {
811            // Classes in a java.* package can only be defined by the
812            // boot or platform class loader.
813            return true;
814        }
815        ClassLoader thisCl = mirror().getClassLoader();
816        ClassLoader accessingClassCl = ((HotSpotResolvedObjectTypeImpl) accessingClass).mirror().getClassLoader();
817        return thisCl == accessingClassCl;
818    }
819
820    @Override
821    public ResolvedJavaType resolve(ResolvedJavaType accessingClass) {
822        if (isDefinitelyResolvedWithRespectTo(requireNonNull(accessingClass))) {
823            return this;
824        }
825        HotSpotResolvedObjectTypeImpl accessingType = (HotSpotResolvedObjectTypeImpl) accessingClass;
826        return (ResolvedJavaType) runtime().lookupType(getName(), accessingType, true);
827    }
828
829    /**
830     * Gets the metaspace Klass boxed in a {@link JavaConstant}.
831     */
832    public Constant klass() {
833        return HotSpotMetaspaceConstantImpl.forMetaspaceObject(this, false);
834    }
835
836    public boolean isPrimaryType() {
837        return config().secondarySuperCacheOffset != superCheckOffset();
838    }
839
840    public int superCheckOffset() {
841        HotSpotVMConfig config = config();
842        return UNSAFE.getInt(getMetaspaceKlass() + config.superCheckOffsetOffset);
843    }
844
845    public long prototypeMarkWord() {
846        HotSpotVMConfig config = config();
847        if (isArray()) {
848            return config.arrayPrototypeMarkWord();
849        } else {
850            return UNSAFE.getAddress(getMetaspaceKlass() + config.prototypeMarkWordOffset);
851        }
852    }
853
854    @Override
855    public ResolvedJavaField findInstanceFieldWithOffset(long offset, JavaKind expectedEntryKind) {
856        ResolvedJavaField[] declaredFields = getInstanceFields(true);
857        return findFieldWithOffset(offset, expectedEntryKind, declaredFields);
858    }
859
860    public ResolvedJavaField findStaticFieldWithOffset(long offset, JavaKind expectedEntryKind) {
861        ResolvedJavaField[] declaredFields = getStaticFields();
862        return findFieldWithOffset(offset, expectedEntryKind, declaredFields);
863    }
864
865    private static ResolvedJavaField findFieldWithOffset(long offset, JavaKind expectedEntryKind, ResolvedJavaField[] declaredFields) {
866        for (ResolvedJavaField field : declaredFields) {
867            HotSpotResolvedJavaField resolvedField = (HotSpotResolvedJavaField) field;
868            long resolvedFieldOffset = resolvedField.offset();
869            // @formatter:off
870            if (ByteOrder.nativeOrder() == ByteOrder.BIG_ENDIAN  &&
871                            expectedEntryKind.isPrimitive() &&
872                            !expectedEntryKind.equals(JavaKind.Void) &&
873                            resolvedField.getJavaKind().isPrimitive()) {
874                resolvedFieldOffset +=
875                                resolvedField.getJavaKind().getByteCount() -
876                                Math.min(resolvedField.getJavaKind().getByteCount(), 4 + expectedEntryKind.getByteCount());
877            }
878            if (resolvedFieldOffset == offset) {
879                return field;
880            }
881            // @formatter:on
882        }
883        return null;
884    }
885
886    @Override
887    public boolean isLocal() {
888        return mirror().isLocalClass();
889    }
890
891    @Override
892    public boolean isMember() {
893        return mirror().isMemberClass();
894    }
895
896    @Override
897    public HotSpotResolvedObjectTypeImpl getEnclosingType() {
898        final Class<?> encl = mirror().getEnclosingClass();
899        return encl == null ? null : fromObjectClass(encl);
900    }
901
902    @Override
903    public ResolvedJavaMethod[] getDeclaredConstructors() {
904        Constructor<?>[] constructors = mirror().getDeclaredConstructors();
905        ResolvedJavaMethod[] result = new ResolvedJavaMethod[constructors.length];
906        for (int i = 0; i < constructors.length; i++) {
907            result[i] = runtime().getHostJVMCIBackend().getMetaAccess().lookupJavaMethod(constructors[i]);
908            assert result[i].isConstructor();
909        }
910        return result;
911    }
912
913    @Override
914    public ResolvedJavaMethod[] getDeclaredMethods() {
915        Method[] methods = mirror().getDeclaredMethods();
916        ResolvedJavaMethod[] result = new ResolvedJavaMethod[methods.length];
917        for (int i = 0; i < methods.length; i++) {
918            result[i] = runtime().getHostJVMCIBackend().getMetaAccess().lookupJavaMethod(methods[i]);
919            assert !result[i].isConstructor();
920        }
921        return result;
922    }
923
924    public ResolvedJavaMethod getClassInitializer() {
925        return compilerToVM().getClassInitializer(this);
926    }
927
928    @Override
929    public String toString() {
930        return "HotSpotType<" + getName() + ", resolved>";
931    }
932
933    @Override
934    public boolean isCloneableWithAllocation() {
935        return (getAccessFlags() & config().jvmAccIsCloneableFast) != 0;
936    }
937}
938