< prev index next >

src/hotspot/share/oops/klass.hpp

Print this page
*** 1,7 ***
  /*
!  * Copyright (c) 1997, 2024, Oracle and/or its affiliates. All rights reserved.
   * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   *
   * This code is free software; you can redistribute it and/or modify it
   * under the terms of the GNU General Public License version 2 only, as
   * published by the Free Software Foundation.
--- 1,7 ---
  /*
!  * Copyright (c) 1997, 2025, Oracle and/or its affiliates. All rights reserved.
   * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   *
   * This code is free software; you can redistribute it and/or modify it
   * under the terms of the GNU General Public License version 2 only, as
   * published by the Free Software Foundation.

*** 70,15 ***
--- 70,17 ---
    friend class JVMCIVMStructs;
   public:
    // Klass Kinds for all subclasses of Klass
    enum KlassKind {
      InstanceKlassKind,
+     InlineKlassKind,
      InstanceRefKlassKind,
      InstanceMirrorKlassKind,
      InstanceClassLoaderKlassKind,
      InstanceStackChunkKlassKind,
      TypeArrayKlassKind,
+     FlatArrayKlassKind,
      ObjArrayKlassKind,
      UnknownKlassKind
    };
  
    static const uint KLASS_KIND_COUNT = ObjArrayKlassKind + 1;

*** 101,11 ***
    //
    // For arrays, layout helper is a negative number, containing four
    // distinct bytes, as follows:
    //    MSB:[tag, hsz, ebt, log2(esz)]:LSB
    // where:
!   //    tag is 0x80 if the elements are oops, 0xC0 if non-oops
    //    hsz is array header size in bytes (i.e., offset of first element)
    //    ebt is the BasicType of the elements
    //    esz is the element size in bytes
    // This packed word is arranged so as to be quickly unpacked by the
    // various fast paths that use the various subfields.
--- 103,11 ---
    //
    // For arrays, layout helper is a negative number, containing four
    // distinct bytes, as follows:
    //    MSB:[tag, hsz, ebt, log2(esz)]:LSB
    // where:
!   //    tag is 0x80 if the elements are oops, 0xC0 if non-oops, 0xA0 if value types
    //    hsz is array header size in bytes (i.e., offset of first element)
    //    ebt is the BasicType of the elements
    //    esz is the element size in bytes
    // This packed word is arranged so as to be quickly unpacked by the
    // various fast paths that use the various subfields.

*** 173,10 ***
--- 175,11 ---
                                  // Some flags created by the JVM, not in the class file itself,
                                  // are in _misc_flags below.
  
    JFR_ONLY(DEFINE_TRACE_ID_FIELD;)
  
+   markWord _prototype_header;  // inline type and inline array mark patterns
  private:
    // This is an index into FileMapHeader::_shared_path_table[], to
    // associate this class with the JAR file where it's loaded from during
    // dump time. If a class is not loaded from the shared archive, this field is
    // -1.

*** 446,15 ***
    static const int _lh_log2_element_size_mask  = BitsPerLong-1;
    static const int _lh_element_type_shift      = BitsPerByte*1;
    static const int _lh_element_type_mask       = right_n_bits(BitsPerByte);  // shifted mask
    static const int _lh_header_size_shift       = BitsPerByte*2;
    static const int _lh_header_size_mask        = right_n_bits(BitsPerByte);  // shifted mask
!   static const int _lh_array_tag_bits          = 2;
    static const int _lh_array_tag_shift         = BitsPerInt - _lh_array_tag_bits;
-   static const int _lh_array_tag_obj_value     = ~0x01;   // 0x80000000 >> 30
  
!   static const unsigned int _lh_array_tag_type_value = 0Xffffffff; // ~0x00,  // 0xC0000000 >> 30
  
    static int layout_helper_size_in_bytes(jint lh) {
      assert(lh > (jint)_lh_neutral_value, "must be instance");
      return (int) lh & ~_lh_instance_slow_path_bit;
    }
--- 449,22 ---
    static const int _lh_log2_element_size_mask  = BitsPerLong-1;
    static const int _lh_element_type_shift      = BitsPerByte*1;
    static const int _lh_element_type_mask       = right_n_bits(BitsPerByte);  // shifted mask
    static const int _lh_header_size_shift       = BitsPerByte*2;
    static const int _lh_header_size_mask        = right_n_bits(BitsPerByte);  // shifted mask
!   static const int _lh_array_tag_bits          = 3;
    static const int _lh_array_tag_shift         = BitsPerInt - _lh_array_tag_bits;
  
!   static const unsigned int _lh_array_tag_type_value = 0Xfffffffc;
+   static const unsigned int _lh_array_tag_vt_value   = 0Xfffffffd;
+   static const unsigned int _lh_array_tag_obj_value  = 0Xfffffffe;
+ 
+   // null-free array flag bit under the array tag bits, shift one more to get array tag value
+   static const int _lh_null_free_shift = _lh_array_tag_shift - 1;
+   static const int _lh_null_free_mask  = 1;
+ 
+   static const jint _lh_array_tag_flat_value_bit_inplace = (jint) (1 << _lh_array_tag_shift);
  
    static int layout_helper_size_in_bytes(jint lh) {
      assert(lh > (jint)_lh_neutral_value, "must be instance");
      return (int) lh & ~_lh_instance_slow_path_bit;
    }

*** 467,27 ***
    }
    static bool layout_helper_is_array(jint lh) {
      return (jint)lh < (jint)_lh_neutral_value;
    }
    static bool layout_helper_is_typeArray(jint lh) {
!     // _lh_array_tag_type_value == (lh >> _lh_array_tag_shift);
-     return (juint)lh >= (juint)(_lh_array_tag_type_value << _lh_array_tag_shift);
    }
    static bool layout_helper_is_objArray(jint lh) {
!     // _lh_array_tag_obj_value == (lh >> _lh_array_tag_shift);
!     return (jint)lh < (jint)(_lh_array_tag_type_value << _lh_array_tag_shift);
    }
    static int layout_helper_header_size(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int hsize = (lh >> _lh_header_size_shift) & _lh_header_size_mask;
      assert(hsize > 0 && hsize < (int)sizeof(oopDesc)*3, "sanity");
      return hsize;
    }
    static BasicType layout_helper_element_type(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int btvalue = (lh >> _lh_element_type_shift) & _lh_element_type_mask;
!     assert(btvalue >= T_BOOLEAN && btvalue <= T_OBJECT, "sanity");
      return (BasicType) btvalue;
    }
  
    // Want a pattern to quickly diff against layout header in register
    // find something less clever!
--- 477,37 ---
    }
    static bool layout_helper_is_array(jint lh) {
      return (jint)lh < (jint)_lh_neutral_value;
    }
    static bool layout_helper_is_typeArray(jint lh) {
!     return (juint) _lh_array_tag_type_value == (juint)(lh >> _lh_array_tag_shift);
    }
    static bool layout_helper_is_objArray(jint lh) {
!     return (juint)_lh_array_tag_obj_value == (juint)(lh >> _lh_array_tag_shift);
!   }
+   static bool layout_helper_is_flatArray(jint lh) {
+     return (juint)_lh_array_tag_vt_value == (juint)(lh >> _lh_array_tag_shift);
+   }
+   static bool layout_helper_is_null_free(jint lh) {
+     assert(layout_helper_is_flatArray(lh) || layout_helper_is_objArray(lh), "must be array of inline types");
+     return ((lh >> _lh_null_free_shift) & _lh_null_free_mask);
+   }
+   static jint layout_helper_set_null_free(jint lh) {
+     lh |= (_lh_null_free_mask << _lh_null_free_shift);
+     assert(layout_helper_is_null_free(lh), "Bad encoding");
+     return lh;
    }
    static int layout_helper_header_size(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int hsize = (lh >> _lh_header_size_shift) & _lh_header_size_mask;
      assert(hsize > 0 && hsize < (int)sizeof(oopDesc)*3, "sanity");
      return hsize;
    }
    static BasicType layout_helper_element_type(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int btvalue = (lh >> _lh_element_type_shift) & _lh_element_type_mask;
!     assert((btvalue >= T_BOOLEAN && btvalue <= T_OBJECT) || btvalue == T_FLAT_ELEMENT, "sanity");
      return (BasicType) btvalue;
    }
  
    // Want a pattern to quickly diff against layout header in register
    // find something less clever!

*** 504,16 ***
    }
  
    static int layout_helper_log2_element_size(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int l2esz = (lh >> _lh_log2_element_size_shift) & _lh_log2_element_size_mask;
!     assert(l2esz <= LogBytesPerLong,
             "sanity. l2esz: 0x%x for lh: 0x%x", (uint)l2esz, (uint)lh);
      return l2esz;
    }
!   static jint array_layout_helper(jint tag, int hsize, BasicType etype, int log2_esize) {
      return (tag        << _lh_array_tag_shift)
        |    (hsize      << _lh_header_size_shift)
        |    ((int)etype << _lh_element_type_shift)
        |    (log2_esize << _lh_log2_element_size_shift);
    }
    static jint instance_layout_helper(jint size, bool slow_path_flag) {
--- 524,17 ---
    }
  
    static int layout_helper_log2_element_size(jint lh) {
      assert(lh < (jint)_lh_neutral_value, "must be array");
      int l2esz = (lh >> _lh_log2_element_size_shift) & _lh_log2_element_size_mask;
!     assert(layout_helper_element_type(lh) == T_FLAT_ELEMENT || l2esz <= LogBytesPerLong,
             "sanity. l2esz: 0x%x for lh: 0x%x", (uint)l2esz, (uint)lh);
      return l2esz;
    }
!   static jint array_layout_helper(jint tag, bool null_free, int hsize, BasicType etype, int log2_esize) {
      return (tag        << _lh_array_tag_shift)
+       |    ((null_free ? 1 : 0) <<  _lh_null_free_shift)
        |    (hsize      << _lh_header_size_shift)
        |    ((int)etype << _lh_element_type_shift)
        |    (log2_esize << _lh_log2_element_size_shift);
    }
    static jint instance_layout_helper(jint size, bool slow_path_flag) {

*** 651,11 ***
--- 672,14 ---
   protected:
    virtual bool is_instance_klass_slow()     const { return false; }
    virtual bool is_array_klass_slow()        const { return false; }
    virtual bool is_objArray_klass_slow()     const { return false; }
    virtual bool is_typeArray_klass_slow()    const { return false; }
+   virtual bool is_flatArray_klass_slow()    const { return false; }
  #endif // ASSERT
+   // current implementation uses this method even in non debug builds
+   virtual bool is_inline_klass_slow()       const { return false; }
   public:
  
    // Fast non-virtual versions
    #ifndef ASSERT
    #define assert_same_query(xval, xcheck) xval

*** 667,31 ***
    }
   public:
    #endif
  
    bool is_instance_klass()              const { return assert_same_query(_kind <= InstanceStackChunkKlassKind, is_instance_klass_slow()); }
!   // Other is anything that is not one of the more specialized kinds of InstanceKlass.
-   bool is_other_instance_klass()        const { return _kind == InstanceKlassKind; }
    bool is_reference_instance_klass()    const { return _kind == InstanceRefKlassKind; }
    bool is_mirror_instance_klass()       const { return _kind == InstanceMirrorKlassKind; }
    bool is_class_loader_instance_klass() const { return _kind == InstanceClassLoaderKlassKind; }
    bool is_array_klass()                 const { return assert_same_query( _kind >= TypeArrayKlassKind, is_array_klass_slow()); }
    bool is_stack_chunk_instance_klass()  const { return _kind == InstanceStackChunkKlassKind; }
    bool is_objArray_klass()              const { return assert_same_query( _kind == ObjArrayKlassKind,  is_objArray_klass_slow()); }
    bool is_typeArray_klass()             const { return assert_same_query( _kind == TypeArrayKlassKind, is_typeArray_klass_slow()); }
    #undef assert_same_query
  
    // Access flags
    AccessFlags access_flags() const         { return _access_flags;  }
    void set_access_flags(AccessFlags flags) { _access_flags = flags; }
  
    bool is_public() const                { return _access_flags.is_public(); }
    bool is_final() const                 { return _access_flags.is_final(); }
    bool is_interface() const             { return _access_flags.is_interface(); }
    bool is_abstract() const              { return _access_flags.is_abstract(); }
-   bool is_super() const                 { return _access_flags.is_super(); }
    bool is_synthetic() const             { return _access_flags.is_synthetic(); }
    void set_is_synthetic()               { _access_flags.set_is_synthetic(); }
    bool has_finalizer() const            { return _misc_flags.has_finalizer(); }
    void set_has_finalizer()              { _misc_flags.set_has_finalizer(true); }
    bool is_hidden() const                { return _misc_flags.is_hidden_class(); }
    void set_is_hidden()                  { _misc_flags.set_is_hidden_class(true); }
--- 691,33 ---
    }
   public:
    #endif
  
    bool is_instance_klass()              const { return assert_same_query(_kind <= InstanceStackChunkKlassKind, is_instance_klass_slow()); }
!   bool is_inline_klass()                const { return assert_same_query(_kind == InlineKlassKind, is_inline_klass_slow()); }
    bool is_reference_instance_klass()    const { return _kind == InstanceRefKlassKind; }
    bool is_mirror_instance_klass()       const { return _kind == InstanceMirrorKlassKind; }
    bool is_class_loader_instance_klass() const { return _kind == InstanceClassLoaderKlassKind; }
    bool is_array_klass()                 const { return assert_same_query( _kind >= TypeArrayKlassKind, is_array_klass_slow()); }
    bool is_stack_chunk_instance_klass()  const { return _kind == InstanceStackChunkKlassKind; }
+   bool is_flatArray_klass()             const { return assert_same_query( _kind == FlatArrayKlassKind, is_flatArray_klass_slow()); }
    bool is_objArray_klass()              const { return assert_same_query( _kind == ObjArrayKlassKind,  is_objArray_klass_slow()); }
    bool is_typeArray_klass()             const { return assert_same_query( _kind == TypeArrayKlassKind, is_typeArray_klass_slow()); }
    #undef assert_same_query
  
+   inline bool is_null_free_array_klass()      const { return layout_helper_is_null_free(layout_helper()); }
+ 
    // Access flags
    AccessFlags access_flags() const         { return _access_flags;  }
    void set_access_flags(AccessFlags flags) { _access_flags = flags; }
  
    bool is_public() const                { return _access_flags.is_public(); }
    bool is_final() const                 { return _access_flags.is_final(); }
    bool is_interface() const             { return _access_flags.is_interface(); }
    bool is_abstract() const              { return _access_flags.is_abstract(); }
    bool is_synthetic() const             { return _access_flags.is_synthetic(); }
+   bool is_identity_class() const        { return _access_flags.is_identity_class(); }
    void set_is_synthetic()               { _access_flags.set_is_synthetic(); }
    bool has_finalizer() const            { return _misc_flags.has_finalizer(); }
    void set_has_finalizer()              { _misc_flags.set_has_finalizer(true); }
    bool is_hidden() const                { return _misc_flags.is_hidden_class(); }
    void set_is_hidden()                  { _misc_flags.set_is_hidden_class(true); }

*** 703,10 ***
--- 729,21 ---
    inline bool is_non_strong_hidden() const;
  
    bool is_cloneable() const;
    void set_is_cloneable();
  
+   // inline types and inline type array patterns
+   markWord prototype_header() const {
+     return _prototype_header;
+   }
+   static inline markWord default_prototype_header(Klass* k) {
+     return (k == nullptr) ? markWord::prototype() : k->prototype_header();
+   }
+ 
+   inline void set_prototype_header(markWord header);
+   static ByteSize prototype_header_offset() { return in_ByteSize(offset_of(Klass, _prototype_header)); }
+ 
    JFR_ONLY(DEFINE_TRACE_ID_METHODS;)
  
    virtual void metaspace_pointers_do(MetaspaceClosure* iter);
    virtual MetaspaceObj::Type type() const { return ClassType; }
  
< prev index next >