annotate src/share/vm/runtime/vmThread.cpp @ 3979:4dfb2df418f2

6484982: G1: process references during evacuation pauses Summary: G1 now uses two reference processors - one is used by concurrent marking and the other is used by STW GCs (both full and incremental evacuation pauses). In an evacuation pause, the reference processor is embedded into the closures used to scan objects. Doing so causes causes reference objects to be 'discovered' by the reference processor. At the end of the evacuation pause, these discovered reference objects are processed - preserving (and copying) referent objects (and their reachable graphs) as appropriate. Reviewed-by: ysr, jwilhelm, brutisso, stefank, tonyp
author johnc
date Thu, 22 Sep 2011 10:57:37 -0700
parents 1f4413413144
children f08d439fab8c
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1490
diff changeset
2 * Copyright (c) 1998, 2010, Oracle and/or its affiliates. All rights reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1490
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1490
diff changeset
20 * or visit www.oracle.com if you need additional information or have any
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1490
diff changeset
21 * questions.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
1972
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
25 #include "precompiled.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
26 #include "compiler/compileBroker.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
27 #include "gc_interface/collectedHeap.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
28 #include "memory/resourceArea.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
29 #include "oops/methodOop.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
30 #include "oops/oop.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
31 #include "runtime/interfaceSupport.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
32 #include "runtime/mutexLocker.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
33 #include "runtime/os.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
34 #include "runtime/vmThread.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
35 #include "runtime/vm_operations.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
36 #include "services/runtimeService.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
37 #include "utilities/dtrace.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
38 #include "utilities/events.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
39 #include "utilities/xmlstream.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
40 #ifdef TARGET_OS_FAMILY_linux
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
41 # include "thread_linux.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
42 #endif
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
43 #ifdef TARGET_OS_FAMILY_solaris
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
44 # include "thread_solaris.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
45 #endif
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
46 #ifdef TARGET_OS_FAMILY_windows
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
47 # include "thread_windows.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1552
diff changeset
48 #endif
0
a61af66fc99e Initial load
duke
parents:
diff changeset
49
1324
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
50 HS_DTRACE_PROBE_DECL3(hotspot, vmops__request, char *, uintptr_t, int);
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
51 HS_DTRACE_PROBE_DECL3(hotspot, vmops__begin, char *, uintptr_t, int);
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
52 HS_DTRACE_PROBE_DECL3(hotspot, vmops__end, char *, uintptr_t, int);
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
53
0
a61af66fc99e Initial load
duke
parents:
diff changeset
54 // Dummy VM operation to act as first element in our circular double-linked list
a61af66fc99e Initial load
duke
parents:
diff changeset
55 class VM_Dummy: public VM_Operation {
a61af66fc99e Initial load
duke
parents:
diff changeset
56 VMOp_Type type() const { return VMOp_Dummy; }
a61af66fc99e Initial load
duke
parents:
diff changeset
57 void doit() {};
a61af66fc99e Initial load
duke
parents:
diff changeset
58 };
a61af66fc99e Initial load
duke
parents:
diff changeset
59
a61af66fc99e Initial load
duke
parents:
diff changeset
60 VMOperationQueue::VMOperationQueue() {
a61af66fc99e Initial load
duke
parents:
diff changeset
61 // The queue is a circular doubled-linked list, which always contains
a61af66fc99e Initial load
duke
parents:
diff changeset
62 // one element (i.e., one element means empty).
a61af66fc99e Initial load
duke
parents:
diff changeset
63 for(int i = 0; i < nof_priorities; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
64 _queue_length[i] = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
65 _queue_counter = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
66 _queue[i] = new VM_Dummy();
a61af66fc99e Initial load
duke
parents:
diff changeset
67 _queue[i]->set_next(_queue[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
68 _queue[i]->set_prev(_queue[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
69 }
a61af66fc99e Initial load
duke
parents:
diff changeset
70 _drain_list = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
71 }
a61af66fc99e Initial load
duke
parents:
diff changeset
72
a61af66fc99e Initial load
duke
parents:
diff changeset
73
a61af66fc99e Initial load
duke
parents:
diff changeset
74 bool VMOperationQueue::queue_empty(int prio) {
a61af66fc99e Initial load
duke
parents:
diff changeset
75 // It is empty if there is exactly one element
a61af66fc99e Initial load
duke
parents:
diff changeset
76 bool empty = (_queue[prio] == _queue[prio]->next());
a61af66fc99e Initial load
duke
parents:
diff changeset
77 assert( (_queue_length[prio] == 0 && empty) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
78 (_queue_length[prio] > 0 && !empty), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
79 return _queue_length[prio] == 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
80 }
a61af66fc99e Initial load
duke
parents:
diff changeset
81
a61af66fc99e Initial load
duke
parents:
diff changeset
82 // Inserts an element to the right of the q element
a61af66fc99e Initial load
duke
parents:
diff changeset
83 void VMOperationQueue::insert(VM_Operation* q, VM_Operation* n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
84 assert(q->next()->prev() == q && q->prev()->next() == q, "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
85 n->set_prev(q);
a61af66fc99e Initial load
duke
parents:
diff changeset
86 n->set_next(q->next());
a61af66fc99e Initial load
duke
parents:
diff changeset
87 q->next()->set_prev(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
88 q->set_next(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
89 }
a61af66fc99e Initial load
duke
parents:
diff changeset
90
a61af66fc99e Initial load
duke
parents:
diff changeset
91 void VMOperationQueue::queue_add_front(int prio, VM_Operation *op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
92 _queue_length[prio]++;
a61af66fc99e Initial load
duke
parents:
diff changeset
93 insert(_queue[prio]->next(), op);
a61af66fc99e Initial load
duke
parents:
diff changeset
94 }
a61af66fc99e Initial load
duke
parents:
diff changeset
95
a61af66fc99e Initial load
duke
parents:
diff changeset
96 void VMOperationQueue::queue_add_back(int prio, VM_Operation *op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
97 _queue_length[prio]++;
a61af66fc99e Initial load
duke
parents:
diff changeset
98 insert(_queue[prio]->prev(), op);
a61af66fc99e Initial load
duke
parents:
diff changeset
99 }
a61af66fc99e Initial load
duke
parents:
diff changeset
100
a61af66fc99e Initial load
duke
parents:
diff changeset
101
a61af66fc99e Initial load
duke
parents:
diff changeset
102 void VMOperationQueue::unlink(VM_Operation* q) {
a61af66fc99e Initial load
duke
parents:
diff changeset
103 assert(q->next()->prev() == q && q->prev()->next() == q, "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
104 q->prev()->set_next(q->next());
a61af66fc99e Initial load
duke
parents:
diff changeset
105 q->next()->set_prev(q->prev());
a61af66fc99e Initial load
duke
parents:
diff changeset
106 }
a61af66fc99e Initial load
duke
parents:
diff changeset
107
a61af66fc99e Initial load
duke
parents:
diff changeset
108 VM_Operation* VMOperationQueue::queue_remove_front(int prio) {
a61af66fc99e Initial load
duke
parents:
diff changeset
109 if (queue_empty(prio)) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
110 assert(_queue_length[prio] >= 0, "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
111 _queue_length[prio]--;
a61af66fc99e Initial load
duke
parents:
diff changeset
112 VM_Operation* r = _queue[prio]->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
113 assert(r != _queue[prio], "cannot remove base element");
a61af66fc99e Initial load
duke
parents:
diff changeset
114 unlink(r);
a61af66fc99e Initial load
duke
parents:
diff changeset
115 return r;
a61af66fc99e Initial load
duke
parents:
diff changeset
116 }
a61af66fc99e Initial load
duke
parents:
diff changeset
117
a61af66fc99e Initial load
duke
parents:
diff changeset
118 VM_Operation* VMOperationQueue::queue_drain(int prio) {
a61af66fc99e Initial load
duke
parents:
diff changeset
119 if (queue_empty(prio)) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
120 DEBUG_ONLY(int length = _queue_length[prio];);
a61af66fc99e Initial load
duke
parents:
diff changeset
121 assert(length >= 0, "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
122 _queue_length[prio] = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
123 VM_Operation* r = _queue[prio]->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
124 assert(r != _queue[prio], "cannot remove base element");
a61af66fc99e Initial load
duke
parents:
diff changeset
125 // remove links to base element from head and tail
a61af66fc99e Initial load
duke
parents:
diff changeset
126 r->set_prev(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
127 _queue[prio]->prev()->set_next(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
128 // restore queue to empty state
a61af66fc99e Initial load
duke
parents:
diff changeset
129 _queue[prio]->set_next(_queue[prio]);
a61af66fc99e Initial load
duke
parents:
diff changeset
130 _queue[prio]->set_prev(_queue[prio]);
1489
cff162798819 6888953: some calls to function-like macros are missing semicolons
jcoomes
parents: 1324
diff changeset
131 assert(queue_empty(prio), "drain corrupted queue");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
132 #ifdef DEBUG
a61af66fc99e Initial load
duke
parents:
diff changeset
133 int len = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
134 VM_Operation* cur;
a61af66fc99e Initial load
duke
parents:
diff changeset
135 for(cur = r; cur != NULL; cur=cur->next()) len++;
a61af66fc99e Initial load
duke
parents:
diff changeset
136 assert(len == length, "drain lost some ops");
a61af66fc99e Initial load
duke
parents:
diff changeset
137 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
138 return r;
a61af66fc99e Initial load
duke
parents:
diff changeset
139 }
a61af66fc99e Initial load
duke
parents:
diff changeset
140
a61af66fc99e Initial load
duke
parents:
diff changeset
141 void VMOperationQueue::queue_oops_do(int queue, OopClosure* f) {
a61af66fc99e Initial load
duke
parents:
diff changeset
142 VM_Operation* cur = _queue[queue];
a61af66fc99e Initial load
duke
parents:
diff changeset
143 cur = cur->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
144 while (cur != _queue[queue]) {
a61af66fc99e Initial load
duke
parents:
diff changeset
145 cur->oops_do(f);
a61af66fc99e Initial load
duke
parents:
diff changeset
146 cur = cur->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
147 }
a61af66fc99e Initial load
duke
parents:
diff changeset
148 }
a61af66fc99e Initial load
duke
parents:
diff changeset
149
a61af66fc99e Initial load
duke
parents:
diff changeset
150 void VMOperationQueue::drain_list_oops_do(OopClosure* f) {
a61af66fc99e Initial load
duke
parents:
diff changeset
151 VM_Operation* cur = _drain_list;
a61af66fc99e Initial load
duke
parents:
diff changeset
152 while (cur != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
153 cur->oops_do(f);
a61af66fc99e Initial load
duke
parents:
diff changeset
154 cur = cur->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
155 }
a61af66fc99e Initial load
duke
parents:
diff changeset
156 }
a61af66fc99e Initial load
duke
parents:
diff changeset
157
a61af66fc99e Initial load
duke
parents:
diff changeset
158 //-----------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
159 // High-level interface
a61af66fc99e Initial load
duke
parents:
diff changeset
160 bool VMOperationQueue::add(VM_Operation *op) {
1324
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
161
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
162 HS_DTRACE_PROBE3(hotspot, vmops__request, op->name(), strlen(op->name()),
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
163 op->evaluation_mode());
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
164
0
a61af66fc99e Initial load
duke
parents:
diff changeset
165 // Encapsulates VM queue policy. Currently, that
a61af66fc99e Initial load
duke
parents:
diff changeset
166 // only involves putting them on the right list
a61af66fc99e Initial load
duke
parents:
diff changeset
167 if (op->evaluate_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
168 queue_add_back(SafepointPriority, op);
a61af66fc99e Initial load
duke
parents:
diff changeset
169 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
170 }
a61af66fc99e Initial load
duke
parents:
diff changeset
171
a61af66fc99e Initial load
duke
parents:
diff changeset
172 queue_add_back(MediumPriority, op);
a61af66fc99e Initial load
duke
parents:
diff changeset
173 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
174 }
a61af66fc99e Initial load
duke
parents:
diff changeset
175
a61af66fc99e Initial load
duke
parents:
diff changeset
176 VM_Operation* VMOperationQueue::remove_next() {
a61af66fc99e Initial load
duke
parents:
diff changeset
177 // Assuming VMOperation queue is two-level priority queue. If there are
a61af66fc99e Initial load
duke
parents:
diff changeset
178 // more than two priorities, we need a different scheduling algorithm.
a61af66fc99e Initial load
duke
parents:
diff changeset
179 assert(SafepointPriority == 0 && MediumPriority == 1 && nof_priorities == 2,
a61af66fc99e Initial load
duke
parents:
diff changeset
180 "current algorithm does not work");
a61af66fc99e Initial load
duke
parents:
diff changeset
181
a61af66fc99e Initial load
duke
parents:
diff changeset
182 // simple counter based scheduling to prevent starvation of lower priority
a61af66fc99e Initial load
duke
parents:
diff changeset
183 // queue. -- see 4390175
a61af66fc99e Initial load
duke
parents:
diff changeset
184 int high_prio, low_prio;
a61af66fc99e Initial load
duke
parents:
diff changeset
185 if (_queue_counter++ < 10) {
a61af66fc99e Initial load
duke
parents:
diff changeset
186 high_prio = SafepointPriority;
a61af66fc99e Initial load
duke
parents:
diff changeset
187 low_prio = MediumPriority;
a61af66fc99e Initial load
duke
parents:
diff changeset
188 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
189 _queue_counter = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
190 high_prio = MediumPriority;
a61af66fc99e Initial load
duke
parents:
diff changeset
191 low_prio = SafepointPriority;
a61af66fc99e Initial load
duke
parents:
diff changeset
192 }
a61af66fc99e Initial load
duke
parents:
diff changeset
193
a61af66fc99e Initial load
duke
parents:
diff changeset
194 return queue_remove_front(queue_empty(high_prio) ? low_prio : high_prio);
a61af66fc99e Initial load
duke
parents:
diff changeset
195 }
a61af66fc99e Initial load
duke
parents:
diff changeset
196
a61af66fc99e Initial load
duke
parents:
diff changeset
197 void VMOperationQueue::oops_do(OopClosure* f) {
a61af66fc99e Initial load
duke
parents:
diff changeset
198 for(int i = 0; i < nof_priorities; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
199 queue_oops_do(i, f);
a61af66fc99e Initial load
duke
parents:
diff changeset
200 }
a61af66fc99e Initial load
duke
parents:
diff changeset
201 drain_list_oops_do(f);
a61af66fc99e Initial load
duke
parents:
diff changeset
202 }
a61af66fc99e Initial load
duke
parents:
diff changeset
203
a61af66fc99e Initial load
duke
parents:
diff changeset
204
a61af66fc99e Initial load
duke
parents:
diff changeset
205 //------------------------------------------------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
206 // Implementation of VMThread stuff
a61af66fc99e Initial load
duke
parents:
diff changeset
207
a61af66fc99e Initial load
duke
parents:
diff changeset
208 bool VMThread::_should_terminate = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
209 bool VMThread::_terminated = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
210 Monitor* VMThread::_terminate_lock = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
211 VMThread* VMThread::_vm_thread = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
212 VM_Operation* VMThread::_cur_vm_operation = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
213 VMOperationQueue* VMThread::_vm_queue = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
214 PerfCounter* VMThread::_perf_accumulated_vm_operation_time = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
215
a61af66fc99e Initial load
duke
parents:
diff changeset
216
a61af66fc99e Initial load
duke
parents:
diff changeset
217 void VMThread::create() {
a61af66fc99e Initial load
duke
parents:
diff changeset
218 assert(vm_thread() == NULL, "we can only allocate one VMThread");
a61af66fc99e Initial load
duke
parents:
diff changeset
219 _vm_thread = new VMThread();
a61af66fc99e Initial load
duke
parents:
diff changeset
220
a61af66fc99e Initial load
duke
parents:
diff changeset
221 // Create VM operation queue
a61af66fc99e Initial load
duke
parents:
diff changeset
222 _vm_queue = new VMOperationQueue();
a61af66fc99e Initial load
duke
parents:
diff changeset
223 guarantee(_vm_queue != NULL, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
224
a61af66fc99e Initial load
duke
parents:
diff changeset
225 _terminate_lock = new Monitor(Mutex::safepoint, "VMThread::_terminate_lock", true);
a61af66fc99e Initial load
duke
parents:
diff changeset
226
a61af66fc99e Initial load
duke
parents:
diff changeset
227 if (UsePerfData) {
a61af66fc99e Initial load
duke
parents:
diff changeset
228 // jvmstat performance counters
a61af66fc99e Initial load
duke
parents:
diff changeset
229 Thread* THREAD = Thread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
230 _perf_accumulated_vm_operation_time =
a61af66fc99e Initial load
duke
parents:
diff changeset
231 PerfDataManager::create_counter(SUN_THREADS, "vmOperationTime",
a61af66fc99e Initial load
duke
parents:
diff changeset
232 PerfData::U_Ticks, CHECK);
a61af66fc99e Initial load
duke
parents:
diff changeset
233 }
a61af66fc99e Initial load
duke
parents:
diff changeset
234 }
a61af66fc99e Initial load
duke
parents:
diff changeset
235
a61af66fc99e Initial load
duke
parents:
diff changeset
236
1119
547f81740344 6361589: Print out stack trace for target thread of GC crash
minqi
parents: 989
diff changeset
237 VMThread::VMThread() : NamedThread() {
547f81740344 6361589: Print out stack trace for target thread of GC crash
minqi
parents: 989
diff changeset
238 set_name("VM Thread");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
239 }
a61af66fc99e Initial load
duke
parents:
diff changeset
240
a61af66fc99e Initial load
duke
parents:
diff changeset
241 void VMThread::destroy() {
a61af66fc99e Initial load
duke
parents:
diff changeset
242 if (_vm_thread != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
243 delete _vm_thread;
a61af66fc99e Initial load
duke
parents:
diff changeset
244 _vm_thread = NULL; // VM thread is gone
a61af66fc99e Initial load
duke
parents:
diff changeset
245 }
a61af66fc99e Initial load
duke
parents:
diff changeset
246 }
a61af66fc99e Initial load
duke
parents:
diff changeset
247
a61af66fc99e Initial load
duke
parents:
diff changeset
248 void VMThread::run() {
a61af66fc99e Initial load
duke
parents:
diff changeset
249 assert(this == vm_thread(), "check");
a61af66fc99e Initial load
duke
parents:
diff changeset
250
a61af66fc99e Initial load
duke
parents:
diff changeset
251 this->initialize_thread_local_storage();
a61af66fc99e Initial load
duke
parents:
diff changeset
252 this->record_stack_base_and_size();
a61af66fc99e Initial load
duke
parents:
diff changeset
253 // Notify_lock wait checks on active_handles() to rewait in
a61af66fc99e Initial load
duke
parents:
diff changeset
254 // case of spurious wakeup, it should wait on the last
a61af66fc99e Initial load
duke
parents:
diff changeset
255 // value set prior to the notify
a61af66fc99e Initial load
duke
parents:
diff changeset
256 this->set_active_handles(JNIHandleBlock::allocate_block());
a61af66fc99e Initial load
duke
parents:
diff changeset
257
a61af66fc99e Initial load
duke
parents:
diff changeset
258 {
a61af66fc99e Initial load
duke
parents:
diff changeset
259 MutexLocker ml(Notify_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
260 Notify_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
261 }
a61af66fc99e Initial load
duke
parents:
diff changeset
262 // Notify_lock is destroyed by Threads::create_vm()
a61af66fc99e Initial load
duke
parents:
diff changeset
263
a61af66fc99e Initial load
duke
parents:
diff changeset
264 int prio = (VMThreadPriority == -1)
a61af66fc99e Initial load
duke
parents:
diff changeset
265 ? os::java_to_os_priority[NearMaxPriority]
a61af66fc99e Initial load
duke
parents:
diff changeset
266 : VMThreadPriority;
a61af66fc99e Initial load
duke
parents:
diff changeset
267 // Note that I cannot call os::set_priority because it expects Java
a61af66fc99e Initial load
duke
parents:
diff changeset
268 // priorities and I am *explicitly* using OS priorities so that it's
a61af66fc99e Initial load
duke
parents:
diff changeset
269 // possible to set the VM thread priority higher than any Java thread.
a61af66fc99e Initial load
duke
parents:
diff changeset
270 os::set_native_priority( this, prio );
a61af66fc99e Initial load
duke
parents:
diff changeset
271
a61af66fc99e Initial load
duke
parents:
diff changeset
272 // Wait for VM_Operations until termination
a61af66fc99e Initial load
duke
parents:
diff changeset
273 this->loop();
a61af66fc99e Initial load
duke
parents:
diff changeset
274
a61af66fc99e Initial load
duke
parents:
diff changeset
275 // Note the intention to exit before safepointing.
a61af66fc99e Initial load
duke
parents:
diff changeset
276 // 6295565 This has the effect of waiting for any large tty
a61af66fc99e Initial load
duke
parents:
diff changeset
277 // outputs to finish.
a61af66fc99e Initial load
duke
parents:
diff changeset
278 if (xtty != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
279 ttyLocker ttyl;
a61af66fc99e Initial load
duke
parents:
diff changeset
280 xtty->begin_elem("destroy_vm");
a61af66fc99e Initial load
duke
parents:
diff changeset
281 xtty->stamp();
a61af66fc99e Initial load
duke
parents:
diff changeset
282 xtty->end_elem();
a61af66fc99e Initial load
duke
parents:
diff changeset
283 assert(should_terminate(), "termination flag must be set");
a61af66fc99e Initial load
duke
parents:
diff changeset
284 }
a61af66fc99e Initial load
duke
parents:
diff changeset
285
a61af66fc99e Initial load
duke
parents:
diff changeset
286 // 4526887 let VM thread exit at Safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
287 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
288
a61af66fc99e Initial load
duke
parents:
diff changeset
289 if (VerifyBeforeExit) {
a61af66fc99e Initial load
duke
parents:
diff changeset
290 HandleMark hm(VMThread::vm_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
291 // Among other things, this ensures that Eden top is correct.
a61af66fc99e Initial load
duke
parents:
diff changeset
292 Universe::heap()->prepare_for_verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
293 os::check_heap();
3293
1f4413413144 7039089: G1: changeset for 7037276 broke heap verification, and related cleanups
ysr
parents: 1972
diff changeset
294 // Silent verification so as not to pollute normal output,
1f4413413144 7039089: G1: changeset for 7037276 broke heap verification, and related cleanups
ysr
parents: 1972
diff changeset
295 // unless we really asked for it.
1f4413413144 7039089: G1: changeset for 7037276 broke heap verification, and related cleanups
ysr
parents: 1972
diff changeset
296 Universe::verify(true, !(PrintGCDetails || Verbose));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
297 }
a61af66fc99e Initial load
duke
parents:
diff changeset
298
a61af66fc99e Initial load
duke
parents:
diff changeset
299 CompileBroker::set_should_block();
a61af66fc99e Initial load
duke
parents:
diff changeset
300
a61af66fc99e Initial load
duke
parents:
diff changeset
301 // wait for threads (compiler threads or daemon threads) in the
a61af66fc99e Initial load
duke
parents:
diff changeset
302 // _thread_in_native state to block.
a61af66fc99e Initial load
duke
parents:
diff changeset
303 VM_Exit::wait_for_threads_in_native_to_block();
a61af66fc99e Initial load
duke
parents:
diff changeset
304
a61af66fc99e Initial load
duke
parents:
diff changeset
305 // signal other threads that VM process is gone
a61af66fc99e Initial load
duke
parents:
diff changeset
306 {
a61af66fc99e Initial load
duke
parents:
diff changeset
307 // Note: we must have the _no_safepoint_check_flag. Mutex::lock() allows
a61af66fc99e Initial load
duke
parents:
diff changeset
308 // VM thread to enter any lock at Safepoint as long as its _owner is NULL.
a61af66fc99e Initial load
duke
parents:
diff changeset
309 // If that happens after _terminate_lock->wait() has unset _owner
a61af66fc99e Initial load
duke
parents:
diff changeset
310 // but before it actually drops the lock and waits, the notification below
a61af66fc99e Initial load
duke
parents:
diff changeset
311 // may get lost and we will have a hang. To avoid this, we need to use
a61af66fc99e Initial load
duke
parents:
diff changeset
312 // Mutex::lock_without_safepoint_check().
a61af66fc99e Initial load
duke
parents:
diff changeset
313 MutexLockerEx ml(_terminate_lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
314 _terminated = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
315 _terminate_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
316 }
a61af66fc99e Initial load
duke
parents:
diff changeset
317
a61af66fc99e Initial load
duke
parents:
diff changeset
318 // Deletion must be done synchronously by the JNI DestroyJavaVM thread
a61af66fc99e Initial load
duke
parents:
diff changeset
319 // so that the VMThread deletion completes before the main thread frees
a61af66fc99e Initial load
duke
parents:
diff changeset
320 // up the CodeHeap.
a61af66fc99e Initial load
duke
parents:
diff changeset
321
a61af66fc99e Initial load
duke
parents:
diff changeset
322 }
a61af66fc99e Initial load
duke
parents:
diff changeset
323
a61af66fc99e Initial load
duke
parents:
diff changeset
324
a61af66fc99e Initial load
duke
parents:
diff changeset
325 // Notify the VMThread that the last non-daemon JavaThread has terminated,
a61af66fc99e Initial load
duke
parents:
diff changeset
326 // and wait until operation is performed.
a61af66fc99e Initial load
duke
parents:
diff changeset
327 void VMThread::wait_for_vm_thread_exit() {
a61af66fc99e Initial load
duke
parents:
diff changeset
328 { MutexLocker mu(VMOperationQueue_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
329 _should_terminate = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
330 VMOperationQueue_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
331 }
a61af66fc99e Initial load
duke
parents:
diff changeset
332
a61af66fc99e Initial load
duke
parents:
diff changeset
333 // Note: VM thread leaves at Safepoint. We are not stopped by Safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
334 // because this thread has been removed from the threads list. But anything
a61af66fc99e Initial load
duke
parents:
diff changeset
335 // that could get blocked by Safepoint should not be used after this point,
a61af66fc99e Initial load
duke
parents:
diff changeset
336 // otherwise we will hang, since there is no one can end the safepoint.
a61af66fc99e Initial load
duke
parents:
diff changeset
337
a61af66fc99e Initial load
duke
parents:
diff changeset
338 // Wait until VM thread is terminated
a61af66fc99e Initial load
duke
parents:
diff changeset
339 // Note: it should be OK to use Terminator_lock here. But this is called
a61af66fc99e Initial load
duke
parents:
diff changeset
340 // at a very delicate time (VM shutdown) and we are operating in non- VM
a61af66fc99e Initial load
duke
parents:
diff changeset
341 // thread at Safepoint. It's safer to not share lock with other threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
342 { MutexLockerEx ml(_terminate_lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
343 while(!VMThread::is_terminated()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
344 _terminate_lock->wait(Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
345 }
a61af66fc99e Initial load
duke
parents:
diff changeset
346 }
a61af66fc99e Initial load
duke
parents:
diff changeset
347 }
a61af66fc99e Initial load
duke
parents:
diff changeset
348
a61af66fc99e Initial load
duke
parents:
diff changeset
349 void VMThread::print_on(outputStream* st) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
350 st->print("\"%s\" ", name());
a61af66fc99e Initial load
duke
parents:
diff changeset
351 Thread::print_on(st);
a61af66fc99e Initial load
duke
parents:
diff changeset
352 st->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
353 }
a61af66fc99e Initial load
duke
parents:
diff changeset
354
a61af66fc99e Initial load
duke
parents:
diff changeset
355 void VMThread::evaluate_operation(VM_Operation* op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
356 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
357
a61af66fc99e Initial load
duke
parents:
diff changeset
358 {
a61af66fc99e Initial load
duke
parents:
diff changeset
359 PerfTraceTime vm_op_timer(perf_accumulated_vm_operation_time());
1324
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
360 HS_DTRACE_PROBE3(hotspot, vmops__begin, op->name(), strlen(op->name()),
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
361 op->evaluation_mode());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
362 op->evaluate();
1324
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
363 HS_DTRACE_PROBE3(hotspot, vmops__end, op->name(), strlen(op->name()),
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
364 op->evaluation_mode());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
365 }
a61af66fc99e Initial load
duke
parents:
diff changeset
366
a61af66fc99e Initial load
duke
parents:
diff changeset
367 // Last access of info in _cur_vm_operation!
a61af66fc99e Initial load
duke
parents:
diff changeset
368 bool c_heap_allocated = op->is_cheap_allocated();
a61af66fc99e Initial load
duke
parents:
diff changeset
369
a61af66fc99e Initial load
duke
parents:
diff changeset
370 // Mark as completed
a61af66fc99e Initial load
duke
parents:
diff changeset
371 if (!op->evaluate_concurrently()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
372 op->calling_thread()->increment_vm_operation_completed_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
373 }
a61af66fc99e Initial load
duke
parents:
diff changeset
374 // It is unsafe to access the _cur_vm_operation after the 'increment_vm_operation_completed_count' call,
a61af66fc99e Initial load
duke
parents:
diff changeset
375 // since if it is stack allocated the calling thread might have deallocated
a61af66fc99e Initial load
duke
parents:
diff changeset
376 if (c_heap_allocated) {
a61af66fc99e Initial load
duke
parents:
diff changeset
377 delete _cur_vm_operation;
a61af66fc99e Initial load
duke
parents:
diff changeset
378 }
a61af66fc99e Initial load
duke
parents:
diff changeset
379 }
a61af66fc99e Initial load
duke
parents:
diff changeset
380
a61af66fc99e Initial load
duke
parents:
diff changeset
381
a61af66fc99e Initial load
duke
parents:
diff changeset
382 void VMThread::loop() {
a61af66fc99e Initial load
duke
parents:
diff changeset
383 assert(_cur_vm_operation == NULL, "no current one should be executing");
a61af66fc99e Initial load
duke
parents:
diff changeset
384
a61af66fc99e Initial load
duke
parents:
diff changeset
385 while(true) {
a61af66fc99e Initial load
duke
parents:
diff changeset
386 VM_Operation* safepoint_ops = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
387 //
a61af66fc99e Initial load
duke
parents:
diff changeset
388 // Wait for VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
389 //
a61af66fc99e Initial load
duke
parents:
diff changeset
390 // use no_safepoint_check to get lock without attempting to "sneak"
a61af66fc99e Initial load
duke
parents:
diff changeset
391 { MutexLockerEx mu_queue(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
392 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
393
a61af66fc99e Initial load
duke
parents:
diff changeset
394 // Look for new operation
a61af66fc99e Initial load
duke
parents:
diff changeset
395 assert(_cur_vm_operation == NULL, "no current one should be executing");
a61af66fc99e Initial load
duke
parents:
diff changeset
396 _cur_vm_operation = _vm_queue->remove_next();
a61af66fc99e Initial load
duke
parents:
diff changeset
397
a61af66fc99e Initial load
duke
parents:
diff changeset
398 // Stall time tracking code
a61af66fc99e Initial load
duke
parents:
diff changeset
399 if (PrintVMQWaitTime && _cur_vm_operation != NULL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
400 !_cur_vm_operation->evaluate_concurrently()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
401 long stall = os::javaTimeMillis() - _cur_vm_operation->timestamp();
a61af66fc99e Initial load
duke
parents:
diff changeset
402 if (stall > 0)
a61af66fc99e Initial load
duke
parents:
diff changeset
403 tty->print_cr("%s stall: %Ld", _cur_vm_operation->name(), stall);
a61af66fc99e Initial load
duke
parents:
diff changeset
404 }
a61af66fc99e Initial load
duke
parents:
diff changeset
405
a61af66fc99e Initial load
duke
parents:
diff changeset
406 while (!should_terminate() && _cur_vm_operation == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
407 // wait with a timeout to guarantee safepoints at regular intervals
a61af66fc99e Initial load
duke
parents:
diff changeset
408 bool timedout =
a61af66fc99e Initial load
duke
parents:
diff changeset
409 VMOperationQueue_lock->wait(Mutex::_no_safepoint_check_flag,
a61af66fc99e Initial load
duke
parents:
diff changeset
410 GuaranteedSafepointInterval);
a61af66fc99e Initial load
duke
parents:
diff changeset
411
a61af66fc99e Initial load
duke
parents:
diff changeset
412 // Support for self destruction
a61af66fc99e Initial load
duke
parents:
diff changeset
413 if ((SelfDestructTimer != 0) && !is_error_reported() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
414 (os::elapsedTime() > SelfDestructTimer * 60)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
415 tty->print_cr("VM self-destructed");
a61af66fc99e Initial load
duke
parents:
diff changeset
416 exit(-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
417 }
a61af66fc99e Initial load
duke
parents:
diff changeset
418
a61af66fc99e Initial load
duke
parents:
diff changeset
419 if (timedout && (SafepointALot ||
a61af66fc99e Initial load
duke
parents:
diff changeset
420 SafepointSynchronize::is_cleanup_needed())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
421 MutexUnlockerEx mul(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
422 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
423 // Force a safepoint since we have not had one for at least
a61af66fc99e Initial load
duke
parents:
diff changeset
424 // 'GuaranteedSafepointInterval' milliseconds. This will run all
a61af66fc99e Initial load
duke
parents:
diff changeset
425 // the clean-up processing that needs to be done regularly at a
a61af66fc99e Initial load
duke
parents:
diff changeset
426 // safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
427 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
428 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
429 if (GCALotAtAllSafepoints) InterfaceSupport::check_gc_alot();
a61af66fc99e Initial load
duke
parents:
diff changeset
430 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
431 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
432 }
a61af66fc99e Initial load
duke
parents:
diff changeset
433 _cur_vm_operation = _vm_queue->remove_next();
a61af66fc99e Initial load
duke
parents:
diff changeset
434
a61af66fc99e Initial load
duke
parents:
diff changeset
435 // If we are at a safepoint we will evaluate all the operations that
a61af66fc99e Initial load
duke
parents:
diff changeset
436 // follow that also require a safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
437 if (_cur_vm_operation != NULL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
438 _cur_vm_operation->evaluate_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
439 safepoint_ops = _vm_queue->drain_at_safepoint_priority();
a61af66fc99e Initial load
duke
parents:
diff changeset
440 }
a61af66fc99e Initial load
duke
parents:
diff changeset
441 }
a61af66fc99e Initial load
duke
parents:
diff changeset
442
a61af66fc99e Initial load
duke
parents:
diff changeset
443 if (should_terminate()) break;
a61af66fc99e Initial load
duke
parents:
diff changeset
444 } // Release mu_queue_lock
a61af66fc99e Initial load
duke
parents:
diff changeset
445
a61af66fc99e Initial load
duke
parents:
diff changeset
446 //
a61af66fc99e Initial load
duke
parents:
diff changeset
447 // Execute VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
448 //
a61af66fc99e Initial load
duke
parents:
diff changeset
449 { HandleMark hm(VMThread::vm_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
450
a61af66fc99e Initial load
duke
parents:
diff changeset
451 EventMark em("Executing VM operation: %s", vm_operation()->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
452 assert(_cur_vm_operation != NULL, "we should have found an operation to execute");
a61af66fc99e Initial load
duke
parents:
diff changeset
453
a61af66fc99e Initial load
duke
parents:
diff changeset
454 // Give the VM thread an extra quantum. Jobs tend to be bursty and this
a61af66fc99e Initial load
duke
parents:
diff changeset
455 // helps the VM thread to finish up the job.
a61af66fc99e Initial load
duke
parents:
diff changeset
456 // FIXME: When this is enabled and there are many threads, this can degrade
a61af66fc99e Initial load
duke
parents:
diff changeset
457 // performance significantly.
a61af66fc99e Initial load
duke
parents:
diff changeset
458 if( VMThreadHintNoPreempt )
a61af66fc99e Initial load
duke
parents:
diff changeset
459 os::hint_no_preempt();
a61af66fc99e Initial load
duke
parents:
diff changeset
460
a61af66fc99e Initial load
duke
parents:
diff changeset
461 // If we are at a safepoint we will evaluate all the operations that
a61af66fc99e Initial load
duke
parents:
diff changeset
462 // follow that also require a safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
463 if (_cur_vm_operation->evaluate_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
464
a61af66fc99e Initial load
duke
parents:
diff changeset
465 _vm_queue->set_drain_list(safepoint_ops); // ensure ops can be scanned
a61af66fc99e Initial load
duke
parents:
diff changeset
466
a61af66fc99e Initial load
duke
parents:
diff changeset
467 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
468 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
469 // now process all queued safepoint ops, iteratively draining
a61af66fc99e Initial load
duke
parents:
diff changeset
470 // the queue until there are none left
a61af66fc99e Initial load
duke
parents:
diff changeset
471 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
472 _cur_vm_operation = safepoint_ops;
a61af66fc99e Initial load
duke
parents:
diff changeset
473 if (_cur_vm_operation != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
474 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
475 // evaluate_operation deletes the op object so we have
a61af66fc99e Initial load
duke
parents:
diff changeset
476 // to grab the next op now
a61af66fc99e Initial load
duke
parents:
diff changeset
477 VM_Operation* next = _cur_vm_operation->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
478 _vm_queue->set_drain_list(next);
a61af66fc99e Initial load
duke
parents:
diff changeset
479 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
480 _cur_vm_operation = next;
a61af66fc99e Initial load
duke
parents:
diff changeset
481 if (PrintSafepointStatistics) {
a61af66fc99e Initial load
duke
parents:
diff changeset
482 SafepointSynchronize::inc_vmop_coalesced_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
483 }
a61af66fc99e Initial load
duke
parents:
diff changeset
484 } while (_cur_vm_operation != NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
485 }
a61af66fc99e Initial load
duke
parents:
diff changeset
486 // There is a chance that a thread enqueued a safepoint op
a61af66fc99e Initial load
duke
parents:
diff changeset
487 // since we released the op-queue lock and initiated the safepoint.
a61af66fc99e Initial load
duke
parents:
diff changeset
488 // So we drain the queue again if there is anything there, as an
a61af66fc99e Initial load
duke
parents:
diff changeset
489 // optimization to try and reduce the number of safepoints.
a61af66fc99e Initial load
duke
parents:
diff changeset
490 // As the safepoint synchronizes us with JavaThreads we will see
a61af66fc99e Initial load
duke
parents:
diff changeset
491 // any enqueue made by a JavaThread, but the peek will not
a61af66fc99e Initial load
duke
parents:
diff changeset
492 // necessarily detect a concurrent enqueue by a GC thread, but
a61af66fc99e Initial load
duke
parents:
diff changeset
493 // that simply means the op will wait for the next major cycle of the
a61af66fc99e Initial load
duke
parents:
diff changeset
494 // VMThread - just as it would if the GC thread lost the race for
a61af66fc99e Initial load
duke
parents:
diff changeset
495 // the lock.
a61af66fc99e Initial load
duke
parents:
diff changeset
496 if (_vm_queue->peek_at_safepoint_priority()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
497 // must hold lock while draining queue
a61af66fc99e Initial load
duke
parents:
diff changeset
498 MutexLockerEx mu_queue(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
499 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
500 safepoint_ops = _vm_queue->drain_at_safepoint_priority();
a61af66fc99e Initial load
duke
parents:
diff changeset
501 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
502 safepoint_ops = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
503 }
a61af66fc99e Initial load
duke
parents:
diff changeset
504 } while(safepoint_ops != NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
505
a61af66fc99e Initial load
duke
parents:
diff changeset
506 _vm_queue->set_drain_list(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
507
a61af66fc99e Initial load
duke
parents:
diff changeset
508 // Complete safepoint synchronization
a61af66fc99e Initial load
duke
parents:
diff changeset
509 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
510
a61af66fc99e Initial load
duke
parents:
diff changeset
511 } else { // not a safepoint operation
a61af66fc99e Initial load
duke
parents:
diff changeset
512 if (TraceLongCompiles) {
a61af66fc99e Initial load
duke
parents:
diff changeset
513 elapsedTimer t;
a61af66fc99e Initial load
duke
parents:
diff changeset
514 t.start();
a61af66fc99e Initial load
duke
parents:
diff changeset
515 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
516 t.stop();
a61af66fc99e Initial load
duke
parents:
diff changeset
517 double secs = t.seconds();
a61af66fc99e Initial load
duke
parents:
diff changeset
518 if (secs * 1e3 > LongCompileThreshold) {
a61af66fc99e Initial load
duke
parents:
diff changeset
519 // XXX - _cur_vm_operation should not be accessed after
a61af66fc99e Initial load
duke
parents:
diff changeset
520 // the completed count has been incremented; the waiting
a61af66fc99e Initial load
duke
parents:
diff changeset
521 // thread may have already freed this memory.
a61af66fc99e Initial load
duke
parents:
diff changeset
522 tty->print_cr("vm %s: %3.7f secs]", _cur_vm_operation->name(), secs);
a61af66fc99e Initial load
duke
parents:
diff changeset
523 }
a61af66fc99e Initial load
duke
parents:
diff changeset
524 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
525 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
526 }
a61af66fc99e Initial load
duke
parents:
diff changeset
527
a61af66fc99e Initial load
duke
parents:
diff changeset
528 _cur_vm_operation = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
529 }
a61af66fc99e Initial load
duke
parents:
diff changeset
530 }
a61af66fc99e Initial load
duke
parents:
diff changeset
531
a61af66fc99e Initial load
duke
parents:
diff changeset
532 //
a61af66fc99e Initial load
duke
parents:
diff changeset
533 // Notify (potential) waiting Java thread(s) - lock without safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
534 // check so that sneaking is not possible
a61af66fc99e Initial load
duke
parents:
diff changeset
535 { MutexLockerEx mu(VMOperationRequest_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
536 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
537 VMOperationRequest_lock->notify_all();
a61af66fc99e Initial load
duke
parents:
diff changeset
538 }
a61af66fc99e Initial load
duke
parents:
diff changeset
539
a61af66fc99e Initial load
duke
parents:
diff changeset
540 //
a61af66fc99e Initial load
duke
parents:
diff changeset
541 // We want to make sure that we get to a safepoint regularly.
a61af66fc99e Initial load
duke
parents:
diff changeset
542 //
a61af66fc99e Initial load
duke
parents:
diff changeset
543 if (SafepointALot || SafepointSynchronize::is_cleanup_needed()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
544 long interval = SafepointSynchronize::last_non_safepoint_interval();
a61af66fc99e Initial load
duke
parents:
diff changeset
545 bool max_time_exceeded = GuaranteedSafepointInterval != 0 && (interval > GuaranteedSafepointInterval);
a61af66fc99e Initial load
duke
parents:
diff changeset
546 if (SafepointALot || max_time_exceeded) {
a61af66fc99e Initial load
duke
parents:
diff changeset
547 HandleMark hm(VMThread::vm_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
548 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
549 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
550 }
a61af66fc99e Initial load
duke
parents:
diff changeset
551 }
a61af66fc99e Initial load
duke
parents:
diff changeset
552 }
a61af66fc99e Initial load
duke
parents:
diff changeset
553 }
a61af66fc99e Initial load
duke
parents:
diff changeset
554
a61af66fc99e Initial load
duke
parents:
diff changeset
555 void VMThread::execute(VM_Operation* op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
556 Thread* t = Thread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
557
a61af66fc99e Initial load
duke
parents:
diff changeset
558 if (!t->is_VM_thread()) {
806
821269eca479 6820167: GCALotAtAllSafepoints + FullGCALot(ScavengeALot) options crash JVM
ysr
parents: 0
diff changeset
559 SkipGCALot sgcalot(t); // avoid re-entrant attempts to gc-a-lot
0
a61af66fc99e Initial load
duke
parents:
diff changeset
560 // JavaThread or WatcherThread
a61af66fc99e Initial load
duke
parents:
diff changeset
561 t->check_for_valid_safepoint_state(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
562
a61af66fc99e Initial load
duke
parents:
diff changeset
563 // New request from Java thread, evaluate prologue
a61af66fc99e Initial load
duke
parents:
diff changeset
564 if (!op->doit_prologue()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
565 return; // op was cancelled
a61af66fc99e Initial load
duke
parents:
diff changeset
566 }
a61af66fc99e Initial load
duke
parents:
diff changeset
567
a61af66fc99e Initial load
duke
parents:
diff changeset
568 // Setup VM_operations for execution
a61af66fc99e Initial load
duke
parents:
diff changeset
569 op->set_calling_thread(t, Thread::get_priority(t));
a61af66fc99e Initial load
duke
parents:
diff changeset
570
a61af66fc99e Initial load
duke
parents:
diff changeset
571 // It does not make sense to execute the epilogue, if the VM operation object is getting
a61af66fc99e Initial load
duke
parents:
diff changeset
572 // deallocated by the VM thread.
a61af66fc99e Initial load
duke
parents:
diff changeset
573 bool concurrent = op->evaluate_concurrently();
a61af66fc99e Initial load
duke
parents:
diff changeset
574 bool execute_epilog = !op->is_cheap_allocated();
a61af66fc99e Initial load
duke
parents:
diff changeset
575 assert(!concurrent || op->is_cheap_allocated(), "concurrent => cheap_allocated");
a61af66fc99e Initial load
duke
parents:
diff changeset
576
a61af66fc99e Initial load
duke
parents:
diff changeset
577 // Get ticket number for non-concurrent VM operations
a61af66fc99e Initial load
duke
parents:
diff changeset
578 int ticket = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
579 if (!concurrent) {
a61af66fc99e Initial load
duke
parents:
diff changeset
580 ticket = t->vm_operation_ticket();
a61af66fc99e Initial load
duke
parents:
diff changeset
581 }
a61af66fc99e Initial load
duke
parents:
diff changeset
582
a61af66fc99e Initial load
duke
parents:
diff changeset
583 // Add VM operation to list of waiting threads. We are guaranteed not to block while holding the
a61af66fc99e Initial load
duke
parents:
diff changeset
584 // VMOperationQueue_lock, so we can block without a safepoint check. This allows vm operation requests
a61af66fc99e Initial load
duke
parents:
diff changeset
585 // to be queued up during a safepoint synchronization.
a61af66fc99e Initial load
duke
parents:
diff changeset
586 {
a61af66fc99e Initial load
duke
parents:
diff changeset
587 VMOperationQueue_lock->lock_without_safepoint_check();
a61af66fc99e Initial load
duke
parents:
diff changeset
588 bool ok = _vm_queue->add(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
589 op->set_timestamp(os::javaTimeMillis());
a61af66fc99e Initial load
duke
parents:
diff changeset
590 VMOperationQueue_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
591 VMOperationQueue_lock->unlock();
a61af66fc99e Initial load
duke
parents:
diff changeset
592 // VM_Operation got skipped
a61af66fc99e Initial load
duke
parents:
diff changeset
593 if (!ok) {
a61af66fc99e Initial load
duke
parents:
diff changeset
594 assert(concurrent, "can only skip concurrent tasks");
a61af66fc99e Initial load
duke
parents:
diff changeset
595 if (op->is_cheap_allocated()) delete op;
a61af66fc99e Initial load
duke
parents:
diff changeset
596 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
597 }
a61af66fc99e Initial load
duke
parents:
diff changeset
598 }
a61af66fc99e Initial load
duke
parents:
diff changeset
599
a61af66fc99e Initial load
duke
parents:
diff changeset
600 if (!concurrent) {
a61af66fc99e Initial load
duke
parents:
diff changeset
601 // Wait for completion of request (non-concurrent)
a61af66fc99e Initial load
duke
parents:
diff changeset
602 // Note: only a JavaThread triggers the safepoint check when locking
a61af66fc99e Initial load
duke
parents:
diff changeset
603 MutexLocker mu(VMOperationRequest_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
604 while(t->vm_operation_completed_count() < ticket) {
a61af66fc99e Initial load
duke
parents:
diff changeset
605 VMOperationRequest_lock->wait(!t->is_Java_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
606 }
a61af66fc99e Initial load
duke
parents:
diff changeset
607 }
a61af66fc99e Initial load
duke
parents:
diff changeset
608
a61af66fc99e Initial load
duke
parents:
diff changeset
609 if (execute_epilog) {
a61af66fc99e Initial load
duke
parents:
diff changeset
610 op->doit_epilogue();
a61af66fc99e Initial load
duke
parents:
diff changeset
611 }
a61af66fc99e Initial load
duke
parents:
diff changeset
612 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
613 // invoked by VM thread; usually nested VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
614 assert(t->is_VM_thread(), "must be a VM thread");
a61af66fc99e Initial load
duke
parents:
diff changeset
615 VM_Operation* prev_vm_operation = vm_operation();
a61af66fc99e Initial load
duke
parents:
diff changeset
616 if (prev_vm_operation != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
617 // Check the VM operation allows nested VM operation. This normally not the case, e.g., the compiler
a61af66fc99e Initial load
duke
parents:
diff changeset
618 // does not allow nested scavenges or compiles.
a61af66fc99e Initial load
duke
parents:
diff changeset
619 if (!prev_vm_operation->allow_nested_vm_operations()) {
1490
f03d0a26bf83 6888954: argument formatting for assert() and friends
jcoomes
parents: 1489
diff changeset
620 fatal(err_msg("Nested VM operation %s requested by operation %s",
f03d0a26bf83 6888954: argument formatting for assert() and friends
jcoomes
parents: 1489
diff changeset
621 op->name(), vm_operation()->name()));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
622 }
a61af66fc99e Initial load
duke
parents:
diff changeset
623 op->set_calling_thread(prev_vm_operation->calling_thread(), prev_vm_operation->priority());
a61af66fc99e Initial load
duke
parents:
diff changeset
624 }
a61af66fc99e Initial load
duke
parents:
diff changeset
625
a61af66fc99e Initial load
duke
parents:
diff changeset
626 EventMark em("Executing %s VM operation: %s", prev_vm_operation ? "nested" : "", op->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
627
a61af66fc99e Initial load
duke
parents:
diff changeset
628 // Release all internal handles after operation is evaluated
a61af66fc99e Initial load
duke
parents:
diff changeset
629 HandleMark hm(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
630 _cur_vm_operation = op;
a61af66fc99e Initial load
duke
parents:
diff changeset
631
a61af66fc99e Initial load
duke
parents:
diff changeset
632 if (op->evaluate_at_safepoint() && !SafepointSynchronize::is_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
633 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
634 op->evaluate();
a61af66fc99e Initial load
duke
parents:
diff changeset
635 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
636 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
637 op->evaluate();
a61af66fc99e Initial load
duke
parents:
diff changeset
638 }
a61af66fc99e Initial load
duke
parents:
diff changeset
639
a61af66fc99e Initial load
duke
parents:
diff changeset
640 // Free memory if needed
a61af66fc99e Initial load
duke
parents:
diff changeset
641 if (op->is_cheap_allocated()) delete op;
a61af66fc99e Initial load
duke
parents:
diff changeset
642
a61af66fc99e Initial load
duke
parents:
diff changeset
643 _cur_vm_operation = prev_vm_operation;
a61af66fc99e Initial load
duke
parents:
diff changeset
644 }
a61af66fc99e Initial load
duke
parents:
diff changeset
645 }
a61af66fc99e Initial load
duke
parents:
diff changeset
646
a61af66fc99e Initial load
duke
parents:
diff changeset
647
989
148e5441d916 6863023: need non-perm oops in code cache for JSR 292
jrose
parents: 844
diff changeset
648 void VMThread::oops_do(OopClosure* f, CodeBlobClosure* cf) {
148e5441d916 6863023: need non-perm oops in code cache for JSR 292
jrose
parents: 844
diff changeset
649 Thread::oops_do(f, cf);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
650 _vm_queue->oops_do(f);
a61af66fc99e Initial load
duke
parents:
diff changeset
651 }
a61af66fc99e Initial load
duke
parents:
diff changeset
652
a61af66fc99e Initial load
duke
parents:
diff changeset
653 //------------------------------------------------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
654 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
655
a61af66fc99e Initial load
duke
parents:
diff changeset
656 void VMOperationQueue::verify_queue(int prio) {
a61af66fc99e Initial load
duke
parents:
diff changeset
657 // Check that list is correctly linked
a61af66fc99e Initial load
duke
parents:
diff changeset
658 int length = _queue_length[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
659 VM_Operation *cur = _queue[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
660 int i;
a61af66fc99e Initial load
duke
parents:
diff changeset
661
a61af66fc99e Initial load
duke
parents:
diff changeset
662 // Check forward links
a61af66fc99e Initial load
duke
parents:
diff changeset
663 for(i = 0; i < length; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
664 cur = cur->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
665 assert(cur != _queue[prio], "list to short (forward)");
a61af66fc99e Initial load
duke
parents:
diff changeset
666 }
a61af66fc99e Initial load
duke
parents:
diff changeset
667 assert(cur->next() == _queue[prio], "list to long (forward)");
a61af66fc99e Initial load
duke
parents:
diff changeset
668
a61af66fc99e Initial load
duke
parents:
diff changeset
669 // Check backwards links
a61af66fc99e Initial load
duke
parents:
diff changeset
670 cur = _queue[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
671 for(i = 0; i < length; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
672 cur = cur->prev();
a61af66fc99e Initial load
duke
parents:
diff changeset
673 assert(cur != _queue[prio], "list to short (backwards)");
a61af66fc99e Initial load
duke
parents:
diff changeset
674 }
a61af66fc99e Initial load
duke
parents:
diff changeset
675 assert(cur->prev() == _queue[prio], "list to long (backwards)");
a61af66fc99e Initial load
duke
parents:
diff changeset
676 }
a61af66fc99e Initial load
duke
parents:
diff changeset
677
a61af66fc99e Initial load
duke
parents:
diff changeset
678 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
679
a61af66fc99e Initial load
duke
parents:
diff changeset
680 void VMThread::verify() {
989
148e5441d916 6863023: need non-perm oops in code cache for JSR 292
jrose
parents: 844
diff changeset
681 oops_do(&VerifyOopClosure::verify_oop, NULL);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
682 }