1 /*
2 * Copyright (c) 1998, 2023, 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_RUNTIME_HANDLES_INLINE_HPP
26 #define SHARE_RUNTIME_HANDLES_INLINE_HPP
27
28 #include "runtime/handles.hpp"
29
30 #include "runtime/javaThread.hpp"
31 #include "oops/metadata.hpp"
32 #include "oops/oop.hpp"
33
34 // these inline functions are in a separate file to break an include cycle
35 // between Thread and Handle
36
37 inline Handle::Handle(Thread* thread, oop obj) {
38 assert(thread == Thread::current(), "sanity check");
39 if (obj == nullptr) {
40 _handle = nullptr;
41 } else {
42 _handle = thread->handle_area()->allocate_handle(obj);
43 }
44 }
45
46 inline void Handle::replace(oop obj) {
47 // Unlike in OopHandle::replace, we shouldn't use a barrier here.
48 // OopHandle has its storage in OopStorage, which is walked concurrently and uses barriers.
49 // Handle is thread private, and iterated by Thread::oops_do, which is why it shouldn't have any barriers at all.
50 assert(_handle != nullptr, "should not use replace");
51 *_handle = obj;
52 }
53
54 // Inline constructors for Specific Handles for different oop types
55 #define DEF_HANDLE_CONSTR(type, is_a) \
56 inline type##Handle::type##Handle (Thread* thread, type##Oop obj) : Handle(thread, (oop)obj) { \
57 assert(is_null() || ((oop)obj)->is_a(), "illegal type"); \
58 }
59
60 DEF_HANDLE_CONSTR(instance , is_instance_noinline )
61 DEF_HANDLE_CONSTR(array , is_array_noinline )
62 DEF_HANDLE_CONSTR(objArray , is_objArray_noinline )
63 DEF_HANDLE_CONSTR(typeArray, is_typeArray_noinline)
64 DEF_HANDLE_CONSTR(flatArray, is_flatArray_noinline)
65
66 // Constructor for metadata handles
67 #define DEF_METADATA_HANDLE_FN(name, type) \
68 inline name##Handle::name##Handle(Thread* thread, type* obj) : _value(obj), _thread(thread) { \
69 if (obj != nullptr) { \
70 assert(((Metadata*)obj)->is_valid(), "obj is valid"); \
71 assert(_thread == Thread::current(), "thread must be current"); \
72 assert(_thread->is_in_live_stack((address)this), "not on stack?"); \
73 _thread->metadata_handles()->push((Metadata*)obj); \
74 } \
75 } \
76
77 DEF_METADATA_HANDLE_FN(method, Method)
78 DEF_METADATA_HANDLE_FN(constantPool, ConstantPool)
79
80 inline void HandleMark::push() {
81 // This is intentionally a NOP. pop_and_restore will reset
82 // values to the HandleMark further down the stack, typically
83 // in JavaCalls::call_helper.
84 debug_only(_area->_handle_mark_nesting++);
85 }
86
87 inline void HandleMark::pop_and_restore() {
88 // Delete later chunks
89 if(_chunk->next() != nullptr) {
90 assert(_area->size_in_bytes() > size_in_bytes(), "Sanity check");
91 chop_later_chunks();
92 } else {
93 assert(_area->size_in_bytes() == size_in_bytes(), "Sanity check");
94 }
95 // Roll back arena to saved top markers
96 _area->_chunk = _chunk;
97 _area->_hwm = _hwm;
98 _area->_max = _max;
99 debug_only(_area->_handle_mark_nesting--);
100 }
101
102 inline HandleMarkCleaner::HandleMarkCleaner(Thread* thread) {
103 _thread = thread;
104 _thread->last_handle_mark()->push();
105 }
106
107 inline HandleMarkCleaner::~HandleMarkCleaner() {
108 _thread->last_handle_mark()->pop_and_restore();
109 }
110
111 #endif // SHARE_RUNTIME_HANDLES_INLINE_HPP
--- EOF ---