annotate src/share/vm/runtime/vmThread.cpp @ 2368:dde920245681

6896099: Integrate CMS heap ergo with default heap sizing ergo 6627787: CMS: JVM refuses to start up with -Xms16m -Xmx16m 7000125: CMS: Anti-monotone young gen sizing with respect to maximum whole heap size specification 7027529: CMS: retire CMSUseOldDefaults flag Summary: Simplify CMS heap sizing code, relying on ergonomic initial sizing consistent with other collectors for the most part, controlling only young gen sizing to rein in pause times. Make CMS young gen sizing default statically cpu-dependant. Remove inconsistencies wrt generation sizing and policy code, allowing for the fixing for 6627787 and 7000125. For 7027529, retire the flag CMSUseOldDefaults which had been introduced as a bridge from JDK 5 to JDK 6 a number of years ago. Reviewed-by: brutisso, poonam
author ysr
date Wed, 16 Mar 2011 10:37:08 -0700
parents f95d63e2154a
children 1f4413413144
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();
a61af66fc99e Initial load
duke
parents:
diff changeset
294 Universe::verify(true, true); // Silent verification to not polute normal output
a61af66fc99e Initial load
duke
parents:
diff changeset
295 }
a61af66fc99e Initial load
duke
parents:
diff changeset
296
a61af66fc99e Initial load
duke
parents:
diff changeset
297 CompileBroker::set_should_block();
a61af66fc99e Initial load
duke
parents:
diff changeset
298
a61af66fc99e Initial load
duke
parents:
diff changeset
299 // wait for threads (compiler threads or daemon threads) in the
a61af66fc99e Initial load
duke
parents:
diff changeset
300 // _thread_in_native state to block.
a61af66fc99e Initial load
duke
parents:
diff changeset
301 VM_Exit::wait_for_threads_in_native_to_block();
a61af66fc99e Initial load
duke
parents:
diff changeset
302
a61af66fc99e Initial load
duke
parents:
diff changeset
303 // signal other threads that VM process is gone
a61af66fc99e Initial load
duke
parents:
diff changeset
304 {
a61af66fc99e Initial load
duke
parents:
diff changeset
305 // Note: we must have the _no_safepoint_check_flag. Mutex::lock() allows
a61af66fc99e Initial load
duke
parents:
diff changeset
306 // VM thread to enter any lock at Safepoint as long as its _owner is NULL.
a61af66fc99e Initial load
duke
parents:
diff changeset
307 // If that happens after _terminate_lock->wait() has unset _owner
a61af66fc99e Initial load
duke
parents:
diff changeset
308 // but before it actually drops the lock and waits, the notification below
a61af66fc99e Initial load
duke
parents:
diff changeset
309 // may get lost and we will have a hang. To avoid this, we need to use
a61af66fc99e Initial load
duke
parents:
diff changeset
310 // Mutex::lock_without_safepoint_check().
a61af66fc99e Initial load
duke
parents:
diff changeset
311 MutexLockerEx ml(_terminate_lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
312 _terminated = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
313 _terminate_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
314 }
a61af66fc99e Initial load
duke
parents:
diff changeset
315
a61af66fc99e Initial load
duke
parents:
diff changeset
316 // Deletion must be done synchronously by the JNI DestroyJavaVM thread
a61af66fc99e Initial load
duke
parents:
diff changeset
317 // so that the VMThread deletion completes before the main thread frees
a61af66fc99e Initial load
duke
parents:
diff changeset
318 // up the CodeHeap.
a61af66fc99e Initial load
duke
parents:
diff changeset
319
a61af66fc99e Initial load
duke
parents:
diff changeset
320 }
a61af66fc99e Initial load
duke
parents:
diff changeset
321
a61af66fc99e Initial load
duke
parents:
diff changeset
322
a61af66fc99e Initial load
duke
parents:
diff changeset
323 // Notify the VMThread that the last non-daemon JavaThread has terminated,
a61af66fc99e Initial load
duke
parents:
diff changeset
324 // and wait until operation is performed.
a61af66fc99e Initial load
duke
parents:
diff changeset
325 void VMThread::wait_for_vm_thread_exit() {
a61af66fc99e Initial load
duke
parents:
diff changeset
326 { MutexLocker mu(VMOperationQueue_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
327 _should_terminate = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
328 VMOperationQueue_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
329 }
a61af66fc99e Initial load
duke
parents:
diff changeset
330
a61af66fc99e Initial load
duke
parents:
diff changeset
331 // Note: VM thread leaves at Safepoint. We are not stopped by Safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
332 // because this thread has been removed from the threads list. But anything
a61af66fc99e Initial load
duke
parents:
diff changeset
333 // that could get blocked by Safepoint should not be used after this point,
a61af66fc99e Initial load
duke
parents:
diff changeset
334 // otherwise we will hang, since there is no one can end the safepoint.
a61af66fc99e Initial load
duke
parents:
diff changeset
335
a61af66fc99e Initial load
duke
parents:
diff changeset
336 // Wait until VM thread is terminated
a61af66fc99e Initial load
duke
parents:
diff changeset
337 // Note: it should be OK to use Terminator_lock here. But this is called
a61af66fc99e Initial load
duke
parents:
diff changeset
338 // at a very delicate time (VM shutdown) and we are operating in non- VM
a61af66fc99e Initial load
duke
parents:
diff changeset
339 // thread at Safepoint. It's safer to not share lock with other threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
340 { MutexLockerEx ml(_terminate_lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
341 while(!VMThread::is_terminated()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
342 _terminate_lock->wait(Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
343 }
a61af66fc99e Initial load
duke
parents:
diff changeset
344 }
a61af66fc99e Initial load
duke
parents:
diff changeset
345 }
a61af66fc99e Initial load
duke
parents:
diff changeset
346
a61af66fc99e Initial load
duke
parents:
diff changeset
347 void VMThread::print_on(outputStream* st) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
348 st->print("\"%s\" ", name());
a61af66fc99e Initial load
duke
parents:
diff changeset
349 Thread::print_on(st);
a61af66fc99e Initial load
duke
parents:
diff changeset
350 st->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
351 }
a61af66fc99e Initial load
duke
parents:
diff changeset
352
a61af66fc99e Initial load
duke
parents:
diff changeset
353 void VMThread::evaluate_operation(VM_Operation* op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
354 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
355
a61af66fc99e Initial load
duke
parents:
diff changeset
356 {
a61af66fc99e Initial load
duke
parents:
diff changeset
357 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
358 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
359 op->evaluation_mode());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
360 op->evaluate();
1324
e392695de029 6935224: Adding new DTrace probes to work with Palantir
fparain
parents: 1260
diff changeset
361 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
362 op->evaluation_mode());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
363 }
a61af66fc99e Initial load
duke
parents:
diff changeset
364
a61af66fc99e Initial load
duke
parents:
diff changeset
365 // Last access of info in _cur_vm_operation!
a61af66fc99e Initial load
duke
parents:
diff changeset
366 bool c_heap_allocated = op->is_cheap_allocated();
a61af66fc99e Initial load
duke
parents:
diff changeset
367
a61af66fc99e Initial load
duke
parents:
diff changeset
368 // Mark as completed
a61af66fc99e Initial load
duke
parents:
diff changeset
369 if (!op->evaluate_concurrently()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
370 op->calling_thread()->increment_vm_operation_completed_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
371 }
a61af66fc99e Initial load
duke
parents:
diff changeset
372 // It is unsafe to access the _cur_vm_operation after the 'increment_vm_operation_completed_count' call,
a61af66fc99e Initial load
duke
parents:
diff changeset
373 // since if it is stack allocated the calling thread might have deallocated
a61af66fc99e Initial load
duke
parents:
diff changeset
374 if (c_heap_allocated) {
a61af66fc99e Initial load
duke
parents:
diff changeset
375 delete _cur_vm_operation;
a61af66fc99e Initial load
duke
parents:
diff changeset
376 }
a61af66fc99e Initial load
duke
parents:
diff changeset
377 }
a61af66fc99e Initial load
duke
parents:
diff changeset
378
a61af66fc99e Initial load
duke
parents:
diff changeset
379
a61af66fc99e Initial load
duke
parents:
diff changeset
380 void VMThread::loop() {
a61af66fc99e Initial load
duke
parents:
diff changeset
381 assert(_cur_vm_operation == NULL, "no current one should be executing");
a61af66fc99e Initial load
duke
parents:
diff changeset
382
a61af66fc99e Initial load
duke
parents:
diff changeset
383 while(true) {
a61af66fc99e Initial load
duke
parents:
diff changeset
384 VM_Operation* safepoint_ops = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
385 //
a61af66fc99e Initial load
duke
parents:
diff changeset
386 // Wait for VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
387 //
a61af66fc99e Initial load
duke
parents:
diff changeset
388 // use no_safepoint_check to get lock without attempting to "sneak"
a61af66fc99e Initial load
duke
parents:
diff changeset
389 { MutexLockerEx mu_queue(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
390 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
391
a61af66fc99e Initial load
duke
parents:
diff changeset
392 // Look for new operation
a61af66fc99e Initial load
duke
parents:
diff changeset
393 assert(_cur_vm_operation == NULL, "no current one should be executing");
a61af66fc99e Initial load
duke
parents:
diff changeset
394 _cur_vm_operation = _vm_queue->remove_next();
a61af66fc99e Initial load
duke
parents:
diff changeset
395
a61af66fc99e Initial load
duke
parents:
diff changeset
396 // Stall time tracking code
a61af66fc99e Initial load
duke
parents:
diff changeset
397 if (PrintVMQWaitTime && _cur_vm_operation != NULL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
398 !_cur_vm_operation->evaluate_concurrently()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
399 long stall = os::javaTimeMillis() - _cur_vm_operation->timestamp();
a61af66fc99e Initial load
duke
parents:
diff changeset
400 if (stall > 0)
a61af66fc99e Initial load
duke
parents:
diff changeset
401 tty->print_cr("%s stall: %Ld", _cur_vm_operation->name(), stall);
a61af66fc99e Initial load
duke
parents:
diff changeset
402 }
a61af66fc99e Initial load
duke
parents:
diff changeset
403
a61af66fc99e Initial load
duke
parents:
diff changeset
404 while (!should_terminate() && _cur_vm_operation == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
405 // wait with a timeout to guarantee safepoints at regular intervals
a61af66fc99e Initial load
duke
parents:
diff changeset
406 bool timedout =
a61af66fc99e Initial load
duke
parents:
diff changeset
407 VMOperationQueue_lock->wait(Mutex::_no_safepoint_check_flag,
a61af66fc99e Initial load
duke
parents:
diff changeset
408 GuaranteedSafepointInterval);
a61af66fc99e Initial load
duke
parents:
diff changeset
409
a61af66fc99e Initial load
duke
parents:
diff changeset
410 // Support for self destruction
a61af66fc99e Initial load
duke
parents:
diff changeset
411 if ((SelfDestructTimer != 0) && !is_error_reported() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
412 (os::elapsedTime() > SelfDestructTimer * 60)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
413 tty->print_cr("VM self-destructed");
a61af66fc99e Initial load
duke
parents:
diff changeset
414 exit(-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
415 }
a61af66fc99e Initial load
duke
parents:
diff changeset
416
a61af66fc99e Initial load
duke
parents:
diff changeset
417 if (timedout && (SafepointALot ||
a61af66fc99e Initial load
duke
parents:
diff changeset
418 SafepointSynchronize::is_cleanup_needed())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
419 MutexUnlockerEx mul(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
420 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
421 // Force a safepoint since we have not had one for at least
a61af66fc99e Initial load
duke
parents:
diff changeset
422 // 'GuaranteedSafepointInterval' milliseconds. This will run all
a61af66fc99e Initial load
duke
parents:
diff changeset
423 // the clean-up processing that needs to be done regularly at a
a61af66fc99e Initial load
duke
parents:
diff changeset
424 // safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
425 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
426 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
427 if (GCALotAtAllSafepoints) InterfaceSupport::check_gc_alot();
a61af66fc99e Initial load
duke
parents:
diff changeset
428 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
429 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
430 }
a61af66fc99e Initial load
duke
parents:
diff changeset
431 _cur_vm_operation = _vm_queue->remove_next();
a61af66fc99e Initial load
duke
parents:
diff changeset
432
a61af66fc99e Initial load
duke
parents:
diff changeset
433 // If we are at a safepoint we will evaluate all the operations that
a61af66fc99e Initial load
duke
parents:
diff changeset
434 // follow that also require a safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
435 if (_cur_vm_operation != NULL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
436 _cur_vm_operation->evaluate_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
437 safepoint_ops = _vm_queue->drain_at_safepoint_priority();
a61af66fc99e Initial load
duke
parents:
diff changeset
438 }
a61af66fc99e Initial load
duke
parents:
diff changeset
439 }
a61af66fc99e Initial load
duke
parents:
diff changeset
440
a61af66fc99e Initial load
duke
parents:
diff changeset
441 if (should_terminate()) break;
a61af66fc99e Initial load
duke
parents:
diff changeset
442 } // Release mu_queue_lock
a61af66fc99e Initial load
duke
parents:
diff changeset
443
a61af66fc99e Initial load
duke
parents:
diff changeset
444 //
a61af66fc99e Initial load
duke
parents:
diff changeset
445 // Execute VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
446 //
a61af66fc99e Initial load
duke
parents:
diff changeset
447 { HandleMark hm(VMThread::vm_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
448
a61af66fc99e Initial load
duke
parents:
diff changeset
449 EventMark em("Executing VM operation: %s", vm_operation()->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
450 assert(_cur_vm_operation != NULL, "we should have found an operation to execute");
a61af66fc99e Initial load
duke
parents:
diff changeset
451
a61af66fc99e Initial load
duke
parents:
diff changeset
452 // Give the VM thread an extra quantum. Jobs tend to be bursty and this
a61af66fc99e Initial load
duke
parents:
diff changeset
453 // helps the VM thread to finish up the job.
a61af66fc99e Initial load
duke
parents:
diff changeset
454 // FIXME: When this is enabled and there are many threads, this can degrade
a61af66fc99e Initial load
duke
parents:
diff changeset
455 // performance significantly.
a61af66fc99e Initial load
duke
parents:
diff changeset
456 if( VMThreadHintNoPreempt )
a61af66fc99e Initial load
duke
parents:
diff changeset
457 os::hint_no_preempt();
a61af66fc99e Initial load
duke
parents:
diff changeset
458
a61af66fc99e Initial load
duke
parents:
diff changeset
459 // If we are at a safepoint we will evaluate all the operations that
a61af66fc99e Initial load
duke
parents:
diff changeset
460 // follow that also require a safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
461 if (_cur_vm_operation->evaluate_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
462
a61af66fc99e Initial load
duke
parents:
diff changeset
463 _vm_queue->set_drain_list(safepoint_ops); // ensure ops can be scanned
a61af66fc99e Initial load
duke
parents:
diff changeset
464
a61af66fc99e Initial load
duke
parents:
diff changeset
465 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
466 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
467 // now process all queued safepoint ops, iteratively draining
a61af66fc99e Initial load
duke
parents:
diff changeset
468 // the queue until there are none left
a61af66fc99e Initial load
duke
parents:
diff changeset
469 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
470 _cur_vm_operation = safepoint_ops;
a61af66fc99e Initial load
duke
parents:
diff changeset
471 if (_cur_vm_operation != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
472 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
473 // evaluate_operation deletes the op object so we have
a61af66fc99e Initial load
duke
parents:
diff changeset
474 // to grab the next op now
a61af66fc99e Initial load
duke
parents:
diff changeset
475 VM_Operation* next = _cur_vm_operation->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
476 _vm_queue->set_drain_list(next);
a61af66fc99e Initial load
duke
parents:
diff changeset
477 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
478 _cur_vm_operation = next;
a61af66fc99e Initial load
duke
parents:
diff changeset
479 if (PrintSafepointStatistics) {
a61af66fc99e Initial load
duke
parents:
diff changeset
480 SafepointSynchronize::inc_vmop_coalesced_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
481 }
a61af66fc99e Initial load
duke
parents:
diff changeset
482 } while (_cur_vm_operation != NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
483 }
a61af66fc99e Initial load
duke
parents:
diff changeset
484 // There is a chance that a thread enqueued a safepoint op
a61af66fc99e Initial load
duke
parents:
diff changeset
485 // since we released the op-queue lock and initiated the safepoint.
a61af66fc99e Initial load
duke
parents:
diff changeset
486 // So we drain the queue again if there is anything there, as an
a61af66fc99e Initial load
duke
parents:
diff changeset
487 // optimization to try and reduce the number of safepoints.
a61af66fc99e Initial load
duke
parents:
diff changeset
488 // As the safepoint synchronizes us with JavaThreads we will see
a61af66fc99e Initial load
duke
parents:
diff changeset
489 // any enqueue made by a JavaThread, but the peek will not
a61af66fc99e Initial load
duke
parents:
diff changeset
490 // necessarily detect a concurrent enqueue by a GC thread, but
a61af66fc99e Initial load
duke
parents:
diff changeset
491 // that simply means the op will wait for the next major cycle of the
a61af66fc99e Initial load
duke
parents:
diff changeset
492 // VMThread - just as it would if the GC thread lost the race for
a61af66fc99e Initial load
duke
parents:
diff changeset
493 // the lock.
a61af66fc99e Initial load
duke
parents:
diff changeset
494 if (_vm_queue->peek_at_safepoint_priority()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
495 // must hold lock while draining queue
a61af66fc99e Initial load
duke
parents:
diff changeset
496 MutexLockerEx mu_queue(VMOperationQueue_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
497 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
498 safepoint_ops = _vm_queue->drain_at_safepoint_priority();
a61af66fc99e Initial load
duke
parents:
diff changeset
499 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
500 safepoint_ops = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
501 }
a61af66fc99e Initial load
duke
parents:
diff changeset
502 } while(safepoint_ops != NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
503
a61af66fc99e Initial load
duke
parents:
diff changeset
504 _vm_queue->set_drain_list(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
505
a61af66fc99e Initial load
duke
parents:
diff changeset
506 // Complete safepoint synchronization
a61af66fc99e Initial load
duke
parents:
diff changeset
507 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
508
a61af66fc99e Initial load
duke
parents:
diff changeset
509 } else { // not a safepoint operation
a61af66fc99e Initial load
duke
parents:
diff changeset
510 if (TraceLongCompiles) {
a61af66fc99e Initial load
duke
parents:
diff changeset
511 elapsedTimer t;
a61af66fc99e Initial load
duke
parents:
diff changeset
512 t.start();
a61af66fc99e Initial load
duke
parents:
diff changeset
513 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
514 t.stop();
a61af66fc99e Initial load
duke
parents:
diff changeset
515 double secs = t.seconds();
a61af66fc99e Initial load
duke
parents:
diff changeset
516 if (secs * 1e3 > LongCompileThreshold) {
a61af66fc99e Initial load
duke
parents:
diff changeset
517 // XXX - _cur_vm_operation should not be accessed after
a61af66fc99e Initial load
duke
parents:
diff changeset
518 // the completed count has been incremented; the waiting
a61af66fc99e Initial load
duke
parents:
diff changeset
519 // thread may have already freed this memory.
a61af66fc99e Initial load
duke
parents:
diff changeset
520 tty->print_cr("vm %s: %3.7f secs]", _cur_vm_operation->name(), secs);
a61af66fc99e Initial load
duke
parents:
diff changeset
521 }
a61af66fc99e Initial load
duke
parents:
diff changeset
522 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
523 evaluate_operation(_cur_vm_operation);
a61af66fc99e Initial load
duke
parents:
diff changeset
524 }
a61af66fc99e Initial load
duke
parents:
diff changeset
525
a61af66fc99e Initial load
duke
parents:
diff changeset
526 _cur_vm_operation = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
527 }
a61af66fc99e Initial load
duke
parents:
diff changeset
528 }
a61af66fc99e Initial load
duke
parents:
diff changeset
529
a61af66fc99e Initial load
duke
parents:
diff changeset
530 //
a61af66fc99e Initial load
duke
parents:
diff changeset
531 // Notify (potential) waiting Java thread(s) - lock without safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
532 // check so that sneaking is not possible
a61af66fc99e Initial load
duke
parents:
diff changeset
533 { MutexLockerEx mu(VMOperationRequest_lock,
a61af66fc99e Initial load
duke
parents:
diff changeset
534 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
535 VMOperationRequest_lock->notify_all();
a61af66fc99e Initial load
duke
parents:
diff changeset
536 }
a61af66fc99e Initial load
duke
parents:
diff changeset
537
a61af66fc99e Initial load
duke
parents:
diff changeset
538 //
a61af66fc99e Initial load
duke
parents:
diff changeset
539 // We want to make sure that we get to a safepoint regularly.
a61af66fc99e Initial load
duke
parents:
diff changeset
540 //
a61af66fc99e Initial load
duke
parents:
diff changeset
541 if (SafepointALot || SafepointSynchronize::is_cleanup_needed()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
542 long interval = SafepointSynchronize::last_non_safepoint_interval();
a61af66fc99e Initial load
duke
parents:
diff changeset
543 bool max_time_exceeded = GuaranteedSafepointInterval != 0 && (interval > GuaranteedSafepointInterval);
a61af66fc99e Initial load
duke
parents:
diff changeset
544 if (SafepointALot || max_time_exceeded) {
a61af66fc99e Initial load
duke
parents:
diff changeset
545 HandleMark hm(VMThread::vm_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
546 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
547 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
548 }
a61af66fc99e Initial load
duke
parents:
diff changeset
549 }
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 void VMThread::execute(VM_Operation* op) {
a61af66fc99e Initial load
duke
parents:
diff changeset
554 Thread* t = Thread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
555
a61af66fc99e Initial load
duke
parents:
diff changeset
556 if (!t->is_VM_thread()) {
806
821269eca479 6820167: GCALotAtAllSafepoints + FullGCALot(ScavengeALot) options crash JVM
ysr
parents: 0
diff changeset
557 SkipGCALot sgcalot(t); // avoid re-entrant attempts to gc-a-lot
0
a61af66fc99e Initial load
duke
parents:
diff changeset
558 // JavaThread or WatcherThread
a61af66fc99e Initial load
duke
parents:
diff changeset
559 t->check_for_valid_safepoint_state(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
560
a61af66fc99e Initial load
duke
parents:
diff changeset
561 // New request from Java thread, evaluate prologue
a61af66fc99e Initial load
duke
parents:
diff changeset
562 if (!op->doit_prologue()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
563 return; // op was cancelled
a61af66fc99e Initial load
duke
parents:
diff changeset
564 }
a61af66fc99e Initial load
duke
parents:
diff changeset
565
a61af66fc99e Initial load
duke
parents:
diff changeset
566 // Setup VM_operations for execution
a61af66fc99e Initial load
duke
parents:
diff changeset
567 op->set_calling_thread(t, Thread::get_priority(t));
a61af66fc99e Initial load
duke
parents:
diff changeset
568
a61af66fc99e Initial load
duke
parents:
diff changeset
569 // It does not make sense to execute the epilogue, if the VM operation object is getting
a61af66fc99e Initial load
duke
parents:
diff changeset
570 // deallocated by the VM thread.
a61af66fc99e Initial load
duke
parents:
diff changeset
571 bool concurrent = op->evaluate_concurrently();
a61af66fc99e Initial load
duke
parents:
diff changeset
572 bool execute_epilog = !op->is_cheap_allocated();
a61af66fc99e Initial load
duke
parents:
diff changeset
573 assert(!concurrent || op->is_cheap_allocated(), "concurrent => cheap_allocated");
a61af66fc99e Initial load
duke
parents:
diff changeset
574
a61af66fc99e Initial load
duke
parents:
diff changeset
575 // Get ticket number for non-concurrent VM operations
a61af66fc99e Initial load
duke
parents:
diff changeset
576 int ticket = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
577 if (!concurrent) {
a61af66fc99e Initial load
duke
parents:
diff changeset
578 ticket = t->vm_operation_ticket();
a61af66fc99e Initial load
duke
parents:
diff changeset
579 }
a61af66fc99e Initial load
duke
parents:
diff changeset
580
a61af66fc99e Initial load
duke
parents:
diff changeset
581 // Add VM operation to list of waiting threads. We are guaranteed not to block while holding the
a61af66fc99e Initial load
duke
parents:
diff changeset
582 // VMOperationQueue_lock, so we can block without a safepoint check. This allows vm operation requests
a61af66fc99e Initial load
duke
parents:
diff changeset
583 // to be queued up during a safepoint synchronization.
a61af66fc99e Initial load
duke
parents:
diff changeset
584 {
a61af66fc99e Initial load
duke
parents:
diff changeset
585 VMOperationQueue_lock->lock_without_safepoint_check();
a61af66fc99e Initial load
duke
parents:
diff changeset
586 bool ok = _vm_queue->add(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
587 op->set_timestamp(os::javaTimeMillis());
a61af66fc99e Initial load
duke
parents:
diff changeset
588 VMOperationQueue_lock->notify();
a61af66fc99e Initial load
duke
parents:
diff changeset
589 VMOperationQueue_lock->unlock();
a61af66fc99e Initial load
duke
parents:
diff changeset
590 // VM_Operation got skipped
a61af66fc99e Initial load
duke
parents:
diff changeset
591 if (!ok) {
a61af66fc99e Initial load
duke
parents:
diff changeset
592 assert(concurrent, "can only skip concurrent tasks");
a61af66fc99e Initial load
duke
parents:
diff changeset
593 if (op->is_cheap_allocated()) delete op;
a61af66fc99e Initial load
duke
parents:
diff changeset
594 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
595 }
a61af66fc99e Initial load
duke
parents:
diff changeset
596 }
a61af66fc99e Initial load
duke
parents:
diff changeset
597
a61af66fc99e Initial load
duke
parents:
diff changeset
598 if (!concurrent) {
a61af66fc99e Initial load
duke
parents:
diff changeset
599 // Wait for completion of request (non-concurrent)
a61af66fc99e Initial load
duke
parents:
diff changeset
600 // Note: only a JavaThread triggers the safepoint check when locking
a61af66fc99e Initial load
duke
parents:
diff changeset
601 MutexLocker mu(VMOperationRequest_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
602 while(t->vm_operation_completed_count() < ticket) {
a61af66fc99e Initial load
duke
parents:
diff changeset
603 VMOperationRequest_lock->wait(!t->is_Java_thread());
a61af66fc99e Initial load
duke
parents:
diff changeset
604 }
a61af66fc99e Initial load
duke
parents:
diff changeset
605 }
a61af66fc99e Initial load
duke
parents:
diff changeset
606
a61af66fc99e Initial load
duke
parents:
diff changeset
607 if (execute_epilog) {
a61af66fc99e Initial load
duke
parents:
diff changeset
608 op->doit_epilogue();
a61af66fc99e Initial load
duke
parents:
diff changeset
609 }
a61af66fc99e Initial load
duke
parents:
diff changeset
610 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
611 // invoked by VM thread; usually nested VM operation
a61af66fc99e Initial load
duke
parents:
diff changeset
612 assert(t->is_VM_thread(), "must be a VM thread");
a61af66fc99e Initial load
duke
parents:
diff changeset
613 VM_Operation* prev_vm_operation = vm_operation();
a61af66fc99e Initial load
duke
parents:
diff changeset
614 if (prev_vm_operation != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
615 // Check the VM operation allows nested VM operation. This normally not the case, e.g., the compiler
a61af66fc99e Initial load
duke
parents:
diff changeset
616 // does not allow nested scavenges or compiles.
a61af66fc99e Initial load
duke
parents:
diff changeset
617 if (!prev_vm_operation->allow_nested_vm_operations()) {
1490
f03d0a26bf83 6888954: argument formatting for assert() and friends
jcoomes
parents: 1489
diff changeset
618 fatal(err_msg("Nested VM operation %s requested by operation %s",
f03d0a26bf83 6888954: argument formatting for assert() and friends
jcoomes
parents: 1489
diff changeset
619 op->name(), vm_operation()->name()));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
620 }
a61af66fc99e Initial load
duke
parents:
diff changeset
621 op->set_calling_thread(prev_vm_operation->calling_thread(), prev_vm_operation->priority());
a61af66fc99e Initial load
duke
parents:
diff changeset
622 }
a61af66fc99e Initial load
duke
parents:
diff changeset
623
a61af66fc99e Initial load
duke
parents:
diff changeset
624 EventMark em("Executing %s VM operation: %s", prev_vm_operation ? "nested" : "", op->name());
a61af66fc99e Initial load
duke
parents:
diff changeset
625
a61af66fc99e Initial load
duke
parents:
diff changeset
626 // Release all internal handles after operation is evaluated
a61af66fc99e Initial load
duke
parents:
diff changeset
627 HandleMark hm(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
628 _cur_vm_operation = op;
a61af66fc99e Initial load
duke
parents:
diff changeset
629
a61af66fc99e Initial load
duke
parents:
diff changeset
630 if (op->evaluate_at_safepoint() && !SafepointSynchronize::is_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
631 SafepointSynchronize::begin();
a61af66fc99e Initial load
duke
parents:
diff changeset
632 op->evaluate();
a61af66fc99e Initial load
duke
parents:
diff changeset
633 SafepointSynchronize::end();
a61af66fc99e Initial load
duke
parents:
diff changeset
634 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
635 op->evaluate();
a61af66fc99e Initial load
duke
parents:
diff changeset
636 }
a61af66fc99e Initial load
duke
parents:
diff changeset
637
a61af66fc99e Initial load
duke
parents:
diff changeset
638 // Free memory if needed
a61af66fc99e Initial load
duke
parents:
diff changeset
639 if (op->is_cheap_allocated()) delete op;
a61af66fc99e Initial load
duke
parents:
diff changeset
640
a61af66fc99e Initial load
duke
parents:
diff changeset
641 _cur_vm_operation = prev_vm_operation;
a61af66fc99e Initial load
duke
parents:
diff changeset
642 }
a61af66fc99e Initial load
duke
parents:
diff changeset
643 }
a61af66fc99e Initial load
duke
parents:
diff changeset
644
a61af66fc99e Initial load
duke
parents:
diff changeset
645
989
148e5441d916 6863023: need non-perm oops in code cache for JSR 292
jrose
parents: 844
diff changeset
646 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
647 Thread::oops_do(f, cf);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
648 _vm_queue->oops_do(f);
a61af66fc99e Initial load
duke
parents:
diff changeset
649 }
a61af66fc99e Initial load
duke
parents:
diff changeset
650
a61af66fc99e Initial load
duke
parents:
diff changeset
651 //------------------------------------------------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
652 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
653
a61af66fc99e Initial load
duke
parents:
diff changeset
654 void VMOperationQueue::verify_queue(int prio) {
a61af66fc99e Initial load
duke
parents:
diff changeset
655 // Check that list is correctly linked
a61af66fc99e Initial load
duke
parents:
diff changeset
656 int length = _queue_length[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
657 VM_Operation *cur = _queue[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
658 int i;
a61af66fc99e Initial load
duke
parents:
diff changeset
659
a61af66fc99e Initial load
duke
parents:
diff changeset
660 // Check forward links
a61af66fc99e Initial load
duke
parents:
diff changeset
661 for(i = 0; i < length; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
662 cur = cur->next();
a61af66fc99e Initial load
duke
parents:
diff changeset
663 assert(cur != _queue[prio], "list to short (forward)");
a61af66fc99e Initial load
duke
parents:
diff changeset
664 }
a61af66fc99e Initial load
duke
parents:
diff changeset
665 assert(cur->next() == _queue[prio], "list to long (forward)");
a61af66fc99e Initial load
duke
parents:
diff changeset
666
a61af66fc99e Initial load
duke
parents:
diff changeset
667 // Check backwards links
a61af66fc99e Initial load
duke
parents:
diff changeset
668 cur = _queue[prio];
a61af66fc99e Initial load
duke
parents:
diff changeset
669 for(i = 0; i < length; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
670 cur = cur->prev();
a61af66fc99e Initial load
duke
parents:
diff changeset
671 assert(cur != _queue[prio], "list to short (backwards)");
a61af66fc99e Initial load
duke
parents:
diff changeset
672 }
a61af66fc99e Initial load
duke
parents:
diff changeset
673 assert(cur->prev() == _queue[prio], "list to long (backwards)");
a61af66fc99e Initial load
duke
parents:
diff changeset
674 }
a61af66fc99e Initial load
duke
parents:
diff changeset
675
a61af66fc99e Initial load
duke
parents:
diff changeset
676 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
677
a61af66fc99e Initial load
duke
parents:
diff changeset
678 void VMThread::verify() {
989
148e5441d916 6863023: need non-perm oops in code cache for JSR 292
jrose
parents: 844
diff changeset
679 oops_do(&VerifyOopClosure::verify_oop, NULL);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
680 }