annotate src/os/windows/vm/attachListener_windows.cpp @ 1552:c18cbe5936b8

6941466: Oracle rebranding changes for Hotspot repositories Summary: Change all the Sun copyrights to Oracle copyright Reviewed-by: ohair
author trims
date Thu, 27 May 2010 19:08:38 -0700
parents a61af66fc99e
children f95d63e2154a
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: 0
diff changeset
2 * Copyright (c) 2005, 2006, 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: 0
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 0
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: 0
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
a61af66fc99e Initial load
duke
parents:
diff changeset
25 # include "incls/_precompiled.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
26 # include "incls/_attachListener_windows.cpp.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
27
a61af66fc99e Initial load
duke
parents:
diff changeset
28 #include <windows.h>
a61af66fc99e Initial load
duke
parents:
diff changeset
29 #include <signal.h> // SIGBREAK
a61af66fc99e Initial load
duke
parents:
diff changeset
30
a61af66fc99e Initial load
duke
parents:
diff changeset
31 // The AttachListener thread services a queue of operations. It blocks in the dequeue
a61af66fc99e Initial load
duke
parents:
diff changeset
32 // function until an operation is enqueued. A client enqueues an operation by creating
a61af66fc99e Initial load
duke
parents:
diff changeset
33 // a thread in this process using the Win32 CreateRemoteThread function. That thread
a61af66fc99e Initial load
duke
parents:
diff changeset
34 // executes a small stub generated by the client. The stub invokes the
a61af66fc99e Initial load
duke
parents:
diff changeset
35 // JVM_EnqueueOperation function which checks the operation parameters and enqueues
a61af66fc99e Initial load
duke
parents:
diff changeset
36 // the operation to the queue serviced by the attach listener. The thread created by
a61af66fc99e Initial load
duke
parents:
diff changeset
37 // the client is a native thread and is restricted to a single page of stack. To keep
a61af66fc99e Initial load
duke
parents:
diff changeset
38 // it simple operations are pre-allocated at initialization time. An enqueue thus
a61af66fc99e Initial load
duke
parents:
diff changeset
39 // takes a preallocated operation, populates the operation parameters, adds it to
a61af66fc99e Initial load
duke
parents:
diff changeset
40 // queue and wakes up the attach listener.
a61af66fc99e Initial load
duke
parents:
diff changeset
41 //
a61af66fc99e Initial load
duke
parents:
diff changeset
42 // When an operation has completed the attach listener is required to send the
a61af66fc99e Initial load
duke
parents:
diff changeset
43 // operation result and any result data to the client. In this implementation the
a61af66fc99e Initial load
duke
parents:
diff changeset
44 // client is a pipe server. In the enqueue operation it provides the name of pipe
a61af66fc99e Initial load
duke
parents:
diff changeset
45 // to this process. When the operation is completed this process opens the pipe and
a61af66fc99e Initial load
duke
parents:
diff changeset
46 // sends the result and output back to the client. Note that writing to the pipe
a61af66fc99e Initial load
duke
parents:
diff changeset
47 // (and flushing the output) is a blocking operation. This means that a non-responsive
a61af66fc99e Initial load
duke
parents:
diff changeset
48 // client could potentially hang the attach listener thread indefinitely. In that
a61af66fc99e Initial load
duke
parents:
diff changeset
49 // case no new operations would be executed but the VM would continue as normal.
a61af66fc99e Initial load
duke
parents:
diff changeset
50 // As only suitably privileged processes can open this process we concluded that
a61af66fc99e Initial load
duke
parents:
diff changeset
51 // this wasn't worth worrying about.
a61af66fc99e Initial load
duke
parents:
diff changeset
52
a61af66fc99e Initial load
duke
parents:
diff changeset
53
a61af66fc99e Initial load
duke
parents:
diff changeset
54 // forward reference
a61af66fc99e Initial load
duke
parents:
diff changeset
55 class Win32AttachOperation;
a61af66fc99e Initial load
duke
parents:
diff changeset
56
a61af66fc99e Initial load
duke
parents:
diff changeset
57
a61af66fc99e Initial load
duke
parents:
diff changeset
58 class Win32AttachListener: AllStatic {
a61af66fc99e Initial load
duke
parents:
diff changeset
59 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
60 enum {
a61af66fc99e Initial load
duke
parents:
diff changeset
61 preallocate_count = 4 // number of preallocated operations
a61af66fc99e Initial load
duke
parents:
diff changeset
62 };
a61af66fc99e Initial load
duke
parents:
diff changeset
63
a61af66fc99e Initial load
duke
parents:
diff changeset
64 // protects the preallocated list and the operation list
a61af66fc99e Initial load
duke
parents:
diff changeset
65 static HANDLE _mutex;
a61af66fc99e Initial load
duke
parents:
diff changeset
66
a61af66fc99e Initial load
duke
parents:
diff changeset
67 // head of preallocated operations list
a61af66fc99e Initial load
duke
parents:
diff changeset
68 static Win32AttachOperation* _avail;
a61af66fc99e Initial load
duke
parents:
diff changeset
69
a61af66fc99e Initial load
duke
parents:
diff changeset
70 // head and tail of enqueue operations list
a61af66fc99e Initial load
duke
parents:
diff changeset
71 static Win32AttachOperation* _head;
a61af66fc99e Initial load
duke
parents:
diff changeset
72 static Win32AttachOperation* _tail;
a61af66fc99e Initial load
duke
parents:
diff changeset
73
a61af66fc99e Initial load
duke
parents:
diff changeset
74
a61af66fc99e Initial load
duke
parents:
diff changeset
75 static Win32AttachOperation* head() { return _head; }
a61af66fc99e Initial load
duke
parents:
diff changeset
76 static void set_head(Win32AttachOperation* head) { _head = head; }
a61af66fc99e Initial load
duke
parents:
diff changeset
77
a61af66fc99e Initial load
duke
parents:
diff changeset
78 static Win32AttachOperation* tail() { return _tail; }
a61af66fc99e Initial load
duke
parents:
diff changeset
79 static void set_tail(Win32AttachOperation* tail) { _tail = tail; }
a61af66fc99e Initial load
duke
parents:
diff changeset
80
a61af66fc99e Initial load
duke
parents:
diff changeset
81
a61af66fc99e Initial load
duke
parents:
diff changeset
82 // used to wakeup the listener
a61af66fc99e Initial load
duke
parents:
diff changeset
83 static HANDLE _wakeup;
a61af66fc99e Initial load
duke
parents:
diff changeset
84 static HANDLE wakeup() { return _wakeup; }
a61af66fc99e Initial load
duke
parents:
diff changeset
85
a61af66fc99e Initial load
duke
parents:
diff changeset
86 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
87 enum {
a61af66fc99e Initial load
duke
parents:
diff changeset
88 ATTACH_ERROR_DISABLED = 100, // error codes
a61af66fc99e Initial load
duke
parents:
diff changeset
89 ATTACH_ERROR_RESOURCE = 101,
a61af66fc99e Initial load
duke
parents:
diff changeset
90 ATTACH_ERROR_ILLEGALARG = 102,
a61af66fc99e Initial load
duke
parents:
diff changeset
91 ATTACH_ERROR_INTERNAL = 103
a61af66fc99e Initial load
duke
parents:
diff changeset
92 };
a61af66fc99e Initial load
duke
parents:
diff changeset
93
a61af66fc99e Initial load
duke
parents:
diff changeset
94 static int init();
a61af66fc99e Initial load
duke
parents:
diff changeset
95 static HANDLE mutex() { return _mutex; }
a61af66fc99e Initial load
duke
parents:
diff changeset
96
a61af66fc99e Initial load
duke
parents:
diff changeset
97 static Win32AttachOperation* available() { return _avail; }
a61af66fc99e Initial load
duke
parents:
diff changeset
98 static void set_available(Win32AttachOperation* avail) { _avail = avail; }
a61af66fc99e Initial load
duke
parents:
diff changeset
99
a61af66fc99e Initial load
duke
parents:
diff changeset
100 // enqueue an operation to the end of the list
a61af66fc99e Initial load
duke
parents:
diff changeset
101 static int enqueue(char* cmd, char* arg1, char* arg2, char* arg3, char* pipename);
a61af66fc99e Initial load
duke
parents:
diff changeset
102
a61af66fc99e Initial load
duke
parents:
diff changeset
103 // dequeue an operation from from head of the list
a61af66fc99e Initial load
duke
parents:
diff changeset
104 static Win32AttachOperation* dequeue();
a61af66fc99e Initial load
duke
parents:
diff changeset
105 };
a61af66fc99e Initial load
duke
parents:
diff changeset
106
a61af66fc99e Initial load
duke
parents:
diff changeset
107 // statics
a61af66fc99e Initial load
duke
parents:
diff changeset
108 HANDLE Win32AttachListener::_mutex;
a61af66fc99e Initial load
duke
parents:
diff changeset
109 HANDLE Win32AttachListener::_wakeup;
a61af66fc99e Initial load
duke
parents:
diff changeset
110 Win32AttachOperation* Win32AttachListener::_avail;
a61af66fc99e Initial load
duke
parents:
diff changeset
111 Win32AttachOperation* Win32AttachListener::_head;
a61af66fc99e Initial load
duke
parents:
diff changeset
112 Win32AttachOperation* Win32AttachListener::_tail;
a61af66fc99e Initial load
duke
parents:
diff changeset
113
a61af66fc99e Initial load
duke
parents:
diff changeset
114
a61af66fc99e Initial load
duke
parents:
diff changeset
115 // Win32AttachOperation is an AttachOperation that additionally encapsulates the name
a61af66fc99e Initial load
duke
parents:
diff changeset
116 // of a pipe which is used to send the operation reply/output to the client.
a61af66fc99e Initial load
duke
parents:
diff changeset
117 // Win32AttachOperation can also be linked in a list.
a61af66fc99e Initial load
duke
parents:
diff changeset
118
a61af66fc99e Initial load
duke
parents:
diff changeset
119 class Win32AttachOperation: public AttachOperation {
a61af66fc99e Initial load
duke
parents:
diff changeset
120 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
121 friend class Win32AttachListener;
a61af66fc99e Initial load
duke
parents:
diff changeset
122
a61af66fc99e Initial load
duke
parents:
diff changeset
123 enum {
a61af66fc99e Initial load
duke
parents:
diff changeset
124 pipe_name_max = 256 // maximum pipe name
a61af66fc99e Initial load
duke
parents:
diff changeset
125 };
a61af66fc99e Initial load
duke
parents:
diff changeset
126
a61af66fc99e Initial load
duke
parents:
diff changeset
127 char _pipe[pipe_name_max+1];
a61af66fc99e Initial load
duke
parents:
diff changeset
128
a61af66fc99e Initial load
duke
parents:
diff changeset
129 const char* pipe() const { return _pipe; }
a61af66fc99e Initial load
duke
parents:
diff changeset
130 void set_pipe(const char* pipe) {
a61af66fc99e Initial load
duke
parents:
diff changeset
131 assert(strlen(pipe) <= pipe_name_max, "execeds maximum length of pipe name");
a61af66fc99e Initial load
duke
parents:
diff changeset
132 strcpy(_pipe, pipe);
a61af66fc99e Initial load
duke
parents:
diff changeset
133 }
a61af66fc99e Initial load
duke
parents:
diff changeset
134
a61af66fc99e Initial load
duke
parents:
diff changeset
135 HANDLE open_pipe();
a61af66fc99e Initial load
duke
parents:
diff changeset
136 static BOOL write_pipe(HANDLE hPipe, char* buf, int len);
a61af66fc99e Initial load
duke
parents:
diff changeset
137
a61af66fc99e Initial load
duke
parents:
diff changeset
138 Win32AttachOperation* _next;
a61af66fc99e Initial load
duke
parents:
diff changeset
139
a61af66fc99e Initial load
duke
parents:
diff changeset
140 Win32AttachOperation* next() const { return _next; }
a61af66fc99e Initial load
duke
parents:
diff changeset
141 void set_next(Win32AttachOperation* next) { _next = next; }
a61af66fc99e Initial load
duke
parents:
diff changeset
142
a61af66fc99e Initial load
duke
parents:
diff changeset
143 // noarg constructor as operation is preallocated
a61af66fc99e Initial load
duke
parents:
diff changeset
144 Win32AttachOperation() : AttachOperation("<noname>") {
a61af66fc99e Initial load
duke
parents:
diff changeset
145 set_pipe("<nopipe>");
a61af66fc99e Initial load
duke
parents:
diff changeset
146 set_next(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
147 }
a61af66fc99e Initial load
duke
parents:
diff changeset
148
a61af66fc99e Initial load
duke
parents:
diff changeset
149 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
150 void Win32AttachOperation::complete(jint result, bufferedStream* result_stream);
a61af66fc99e Initial load
duke
parents:
diff changeset
151 };
a61af66fc99e Initial load
duke
parents:
diff changeset
152
a61af66fc99e Initial load
duke
parents:
diff changeset
153
a61af66fc99e Initial load
duke
parents:
diff changeset
154 // preallocate the required number of operations
a61af66fc99e Initial load
duke
parents:
diff changeset
155 int Win32AttachListener::init() {
a61af66fc99e Initial load
duke
parents:
diff changeset
156 _mutex = (void*)::CreateMutex(NULL, FALSE, NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
157 guarantee(_mutex != (HANDLE)NULL, "mutex creation failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
158
a61af66fc99e Initial load
duke
parents:
diff changeset
159 _wakeup = ::CreateSemaphore(NULL, 0, 1, NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
160 guarantee(_wakeup != (HANDLE)NULL, "semaphore creation failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
161
a61af66fc99e Initial load
duke
parents:
diff changeset
162 set_head(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
163 set_tail(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
164
a61af66fc99e Initial load
duke
parents:
diff changeset
165 // preallocate a few operations
a61af66fc99e Initial load
duke
parents:
diff changeset
166 set_available(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
167 for (int i=0; i<preallocate_count; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
168 Win32AttachOperation* op = new Win32AttachOperation();
a61af66fc99e Initial load
duke
parents:
diff changeset
169 op->set_next(available());
a61af66fc99e Initial load
duke
parents:
diff changeset
170 set_available(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
171 }
a61af66fc99e Initial load
duke
parents:
diff changeset
172
a61af66fc99e Initial load
duke
parents:
diff changeset
173 return 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
174 }
a61af66fc99e Initial load
duke
parents:
diff changeset
175
a61af66fc99e Initial load
duke
parents:
diff changeset
176 // Enqueue an operation. This is called from a native thread that is not attached to VM.
a61af66fc99e Initial load
duke
parents:
diff changeset
177 // Also we need to be careful not to execute anything that results in more than a 4k stack.
a61af66fc99e Initial load
duke
parents:
diff changeset
178 //
a61af66fc99e Initial load
duke
parents:
diff changeset
179 int Win32AttachListener::enqueue(char* cmd, char* arg0, char* arg1, char* arg2, char* pipename) {
a61af66fc99e Initial load
duke
parents:
diff changeset
180 // listener not running
a61af66fc99e Initial load
duke
parents:
diff changeset
181 if (!AttachListener::is_initialized()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
182 return ATTACH_ERROR_DISABLED;
a61af66fc99e Initial load
duke
parents:
diff changeset
183 }
a61af66fc99e Initial load
duke
parents:
diff changeset
184
a61af66fc99e Initial load
duke
parents:
diff changeset
185 // check that all paramteres to the operation
a61af66fc99e Initial load
duke
parents:
diff changeset
186 if (strlen(cmd) > AttachOperation::name_length_max) return ATTACH_ERROR_ILLEGALARG;
a61af66fc99e Initial load
duke
parents:
diff changeset
187 if (strlen(arg0) > AttachOperation::arg_length_max) return ATTACH_ERROR_ILLEGALARG;
a61af66fc99e Initial load
duke
parents:
diff changeset
188 if (strlen(arg0) > AttachOperation::arg_length_max) return ATTACH_ERROR_ILLEGALARG;
a61af66fc99e Initial load
duke
parents:
diff changeset
189 if (strlen(pipename) > Win32AttachOperation::pipe_name_max) return ATTACH_ERROR_ILLEGALARG;
a61af66fc99e Initial load
duke
parents:
diff changeset
190
a61af66fc99e Initial load
duke
parents:
diff changeset
191 // check for a well-formed pipename
a61af66fc99e Initial load
duke
parents:
diff changeset
192 if (strstr(pipename, "\\\\.\\pipe\\") != pipename) return ATTACH_ERROR_ILLEGALARG;
a61af66fc99e Initial load
duke
parents:
diff changeset
193
a61af66fc99e Initial load
duke
parents:
diff changeset
194 // grab the lock for the list
a61af66fc99e Initial load
duke
parents:
diff changeset
195 DWORD res = ::WaitForSingleObject(mutex(), INFINITE);
a61af66fc99e Initial load
duke
parents:
diff changeset
196 if (res != WAIT_OBJECT_0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
197 return ATTACH_ERROR_INTERNAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
198 }
a61af66fc99e Initial load
duke
parents:
diff changeset
199
a61af66fc99e Initial load
duke
parents:
diff changeset
200 // try to get an operation from the available list
a61af66fc99e Initial load
duke
parents:
diff changeset
201 Win32AttachOperation* op = available();
a61af66fc99e Initial load
duke
parents:
diff changeset
202 if (op != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
203 set_available(op->next());
a61af66fc99e Initial load
duke
parents:
diff changeset
204
a61af66fc99e Initial load
duke
parents:
diff changeset
205 // add to end (tail) of list
a61af66fc99e Initial load
duke
parents:
diff changeset
206 op->set_next(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
207 if (tail() == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
208 set_head(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
209 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
210 tail()->set_next(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
211 }
a61af66fc99e Initial load
duke
parents:
diff changeset
212 set_tail(op);
a61af66fc99e Initial load
duke
parents:
diff changeset
213
a61af66fc99e Initial load
duke
parents:
diff changeset
214 op->set_name(cmd);
a61af66fc99e Initial load
duke
parents:
diff changeset
215 op->set_arg(0, arg0);
a61af66fc99e Initial load
duke
parents:
diff changeset
216 op->set_arg(1, arg1);
a61af66fc99e Initial load
duke
parents:
diff changeset
217 op->set_arg(2, arg2);
a61af66fc99e Initial load
duke
parents:
diff changeset
218 op->set_pipe(pipename);
a61af66fc99e Initial load
duke
parents:
diff changeset
219
a61af66fc99e Initial load
duke
parents:
diff changeset
220 // wakeup the thread waiting for operations
a61af66fc99e Initial load
duke
parents:
diff changeset
221 ::ReleaseSemaphore(wakeup(), 1, NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
222 }
a61af66fc99e Initial load
duke
parents:
diff changeset
223 ::ReleaseMutex(mutex());
a61af66fc99e Initial load
duke
parents:
diff changeset
224
a61af66fc99e Initial load
duke
parents:
diff changeset
225 return (op != NULL) ? 0 : ATTACH_ERROR_RESOURCE;
a61af66fc99e Initial load
duke
parents:
diff changeset
226 }
a61af66fc99e Initial load
duke
parents:
diff changeset
227
a61af66fc99e Initial load
duke
parents:
diff changeset
228
a61af66fc99e Initial load
duke
parents:
diff changeset
229 // dequeue the operation from the head of the operation list. If
a61af66fc99e Initial load
duke
parents:
diff changeset
230 Win32AttachOperation* Win32AttachListener::dequeue() {
a61af66fc99e Initial load
duke
parents:
diff changeset
231 for (;;) {
a61af66fc99e Initial load
duke
parents:
diff changeset
232 DWORD res = ::WaitForSingleObject(wakeup(), INFINITE);
a61af66fc99e Initial load
duke
parents:
diff changeset
233 guarantee(res == WAIT_OBJECT_0, "wait failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
234
a61af66fc99e Initial load
duke
parents:
diff changeset
235 res = ::WaitForSingleObject(mutex(), INFINITE);
a61af66fc99e Initial load
duke
parents:
diff changeset
236 guarantee(res == WAIT_OBJECT_0, "wait failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
237
a61af66fc99e Initial load
duke
parents:
diff changeset
238 Win32AttachOperation* op = head();
a61af66fc99e Initial load
duke
parents:
diff changeset
239 if (op != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
240 set_head(op->next());
a61af66fc99e Initial load
duke
parents:
diff changeset
241 if (head() == NULL) { // list is empty
a61af66fc99e Initial load
duke
parents:
diff changeset
242 set_tail(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
243 }
a61af66fc99e Initial load
duke
parents:
diff changeset
244 }
a61af66fc99e Initial load
duke
parents:
diff changeset
245 ::ReleaseMutex(mutex());
a61af66fc99e Initial load
duke
parents:
diff changeset
246
a61af66fc99e Initial load
duke
parents:
diff changeset
247 if (op != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
248 return op;
a61af66fc99e Initial load
duke
parents:
diff changeset
249 }
a61af66fc99e Initial load
duke
parents:
diff changeset
250 }
a61af66fc99e Initial load
duke
parents:
diff changeset
251 }
a61af66fc99e Initial load
duke
parents:
diff changeset
252
a61af66fc99e Initial load
duke
parents:
diff changeset
253
a61af66fc99e Initial load
duke
parents:
diff changeset
254 // open the pipe to the client
a61af66fc99e Initial load
duke
parents:
diff changeset
255 HANDLE Win32AttachOperation::open_pipe() {
a61af66fc99e Initial load
duke
parents:
diff changeset
256 HANDLE hPipe;
a61af66fc99e Initial load
duke
parents:
diff changeset
257
a61af66fc99e Initial load
duke
parents:
diff changeset
258 hPipe = ::CreateFile( pipe(), // pipe name
a61af66fc99e Initial load
duke
parents:
diff changeset
259 GENERIC_WRITE, // write only
a61af66fc99e Initial load
duke
parents:
diff changeset
260 0, // no sharing
a61af66fc99e Initial load
duke
parents:
diff changeset
261 NULL, // default security attributes
a61af66fc99e Initial load
duke
parents:
diff changeset
262 OPEN_EXISTING, // opens existing pipe
a61af66fc99e Initial load
duke
parents:
diff changeset
263 0, // default attributes
a61af66fc99e Initial load
duke
parents:
diff changeset
264 NULL); // no template file
a61af66fc99e Initial load
duke
parents:
diff changeset
265
a61af66fc99e Initial load
duke
parents:
diff changeset
266 if (hPipe != INVALID_HANDLE_VALUE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
267 // shouldn't happen as there is a pipe created per operation
a61af66fc99e Initial load
duke
parents:
diff changeset
268 if (::GetLastError() == ERROR_PIPE_BUSY) {
a61af66fc99e Initial load
duke
parents:
diff changeset
269 return INVALID_HANDLE_VALUE;
a61af66fc99e Initial load
duke
parents:
diff changeset
270 }
a61af66fc99e Initial load
duke
parents:
diff changeset
271 }
a61af66fc99e Initial load
duke
parents:
diff changeset
272 return hPipe;
a61af66fc99e Initial load
duke
parents:
diff changeset
273 }
a61af66fc99e Initial load
duke
parents:
diff changeset
274
a61af66fc99e Initial load
duke
parents:
diff changeset
275 // write to the pipe
a61af66fc99e Initial load
duke
parents:
diff changeset
276 BOOL Win32AttachOperation::write_pipe(HANDLE hPipe, char* buf, int len) {
a61af66fc99e Initial load
duke
parents:
diff changeset
277 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
278 DWORD nwrote;
a61af66fc99e Initial load
duke
parents:
diff changeset
279
a61af66fc99e Initial load
duke
parents:
diff changeset
280 BOOL fSuccess = WriteFile( hPipe, // pipe handle
a61af66fc99e Initial load
duke
parents:
diff changeset
281 (LPCVOID)buf, // message
a61af66fc99e Initial load
duke
parents:
diff changeset
282 (DWORD)len, // message length
a61af66fc99e Initial load
duke
parents:
diff changeset
283 &nwrote, // bytes written
a61af66fc99e Initial load
duke
parents:
diff changeset
284 NULL); // not overlapped
a61af66fc99e Initial load
duke
parents:
diff changeset
285 if (!fSuccess) {
a61af66fc99e Initial load
duke
parents:
diff changeset
286 return fSuccess;
a61af66fc99e Initial load
duke
parents:
diff changeset
287 }
a61af66fc99e Initial load
duke
parents:
diff changeset
288 buf += nwrote;
a61af66fc99e Initial load
duke
parents:
diff changeset
289 len -= nwrote;
a61af66fc99e Initial load
duke
parents:
diff changeset
290 }
a61af66fc99e Initial load
duke
parents:
diff changeset
291 while (len > 0);
a61af66fc99e Initial load
duke
parents:
diff changeset
292 return TRUE;
a61af66fc99e Initial load
duke
parents:
diff changeset
293 }
a61af66fc99e Initial load
duke
parents:
diff changeset
294
a61af66fc99e Initial load
duke
parents:
diff changeset
295 // Complete the operation:
a61af66fc99e Initial load
duke
parents:
diff changeset
296 // - open the pipe to the client
a61af66fc99e Initial load
duke
parents:
diff changeset
297 // - write the operation result (a jint)
a61af66fc99e Initial load
duke
parents:
diff changeset
298 // - write the operation output (the result stream)
a61af66fc99e Initial load
duke
parents:
diff changeset
299 //
a61af66fc99e Initial load
duke
parents:
diff changeset
300 void Win32AttachOperation::complete(jint result, bufferedStream* result_stream) {
a61af66fc99e Initial load
duke
parents:
diff changeset
301 JavaThread* thread = JavaThread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
302 ThreadBlockInVM tbivm(thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
303
a61af66fc99e Initial load
duke
parents:
diff changeset
304 thread->set_suspend_equivalent();
a61af66fc99e Initial load
duke
parents:
diff changeset
305 // cleared by handle_special_suspend_equivalent_condition() or
a61af66fc99e Initial load
duke
parents:
diff changeset
306 // java_suspend_self() via check_and_wait_while_suspended()
a61af66fc99e Initial load
duke
parents:
diff changeset
307
a61af66fc99e Initial load
duke
parents:
diff changeset
308 HANDLE hPipe = open_pipe();
a61af66fc99e Initial load
duke
parents:
diff changeset
309 if (hPipe != INVALID_HANDLE_VALUE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
310 BOOL fSuccess;
a61af66fc99e Initial load
duke
parents:
diff changeset
311
a61af66fc99e Initial load
duke
parents:
diff changeset
312 char msg[32];
a61af66fc99e Initial load
duke
parents:
diff changeset
313 sprintf(msg, "%d\n", result);
a61af66fc99e Initial load
duke
parents:
diff changeset
314
a61af66fc99e Initial load
duke
parents:
diff changeset
315 fSuccess = write_pipe(hPipe, msg, (int)strlen(msg));
a61af66fc99e Initial load
duke
parents:
diff changeset
316 if (fSuccess) {
a61af66fc99e Initial load
duke
parents:
diff changeset
317 write_pipe(hPipe, (char*) result_stream->base(), (int)(result_stream->size()));
a61af66fc99e Initial load
duke
parents:
diff changeset
318 }
a61af66fc99e Initial load
duke
parents:
diff changeset
319
a61af66fc99e Initial load
duke
parents:
diff changeset
320 // Need to flush buffers
a61af66fc99e Initial load
duke
parents:
diff changeset
321 FlushFileBuffers(hPipe);
a61af66fc99e Initial load
duke
parents:
diff changeset
322 CloseHandle(hPipe);
a61af66fc99e Initial load
duke
parents:
diff changeset
323 }
a61af66fc99e Initial load
duke
parents:
diff changeset
324
a61af66fc99e Initial load
duke
parents:
diff changeset
325 DWORD res = ::WaitForSingleObject(Win32AttachListener::mutex(), INFINITE);
a61af66fc99e Initial load
duke
parents:
diff changeset
326 if (res == WAIT_OBJECT_0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
327
a61af66fc99e Initial load
duke
parents:
diff changeset
328 // put the operation back on the available list
a61af66fc99e Initial load
duke
parents:
diff changeset
329 set_next(Win32AttachListener::available());
a61af66fc99e Initial load
duke
parents:
diff changeset
330 Win32AttachListener::set_available(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
331
a61af66fc99e Initial load
duke
parents:
diff changeset
332 ::ReleaseMutex(Win32AttachListener::mutex());
a61af66fc99e Initial load
duke
parents:
diff changeset
333 }
a61af66fc99e Initial load
duke
parents:
diff changeset
334
a61af66fc99e Initial load
duke
parents:
diff changeset
335 // were we externally suspended while we were waiting?
a61af66fc99e Initial load
duke
parents:
diff changeset
336 thread->check_and_wait_while_suspended();
a61af66fc99e Initial load
duke
parents:
diff changeset
337 }
a61af66fc99e Initial load
duke
parents:
diff changeset
338
a61af66fc99e Initial load
duke
parents:
diff changeset
339
a61af66fc99e Initial load
duke
parents:
diff changeset
340 // AttachOperation functions
a61af66fc99e Initial load
duke
parents:
diff changeset
341
a61af66fc99e Initial load
duke
parents:
diff changeset
342 AttachOperation* AttachListener::dequeue() {
a61af66fc99e Initial load
duke
parents:
diff changeset
343 JavaThread* thread = JavaThread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
344 ThreadBlockInVM tbivm(thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
345
a61af66fc99e Initial load
duke
parents:
diff changeset
346 thread->set_suspend_equivalent();
a61af66fc99e Initial load
duke
parents:
diff changeset
347 // cleared by handle_special_suspend_equivalent_condition() or
a61af66fc99e Initial load
duke
parents:
diff changeset
348 // java_suspend_self() via check_and_wait_while_suspended()
a61af66fc99e Initial load
duke
parents:
diff changeset
349
a61af66fc99e Initial load
duke
parents:
diff changeset
350 AttachOperation* op = Win32AttachListener::dequeue();
a61af66fc99e Initial load
duke
parents:
diff changeset
351
a61af66fc99e Initial load
duke
parents:
diff changeset
352 // were we externally suspended while we were waiting?
a61af66fc99e Initial load
duke
parents:
diff changeset
353 thread->check_and_wait_while_suspended();
a61af66fc99e Initial load
duke
parents:
diff changeset
354
a61af66fc99e Initial load
duke
parents:
diff changeset
355 return op;
a61af66fc99e Initial load
duke
parents:
diff changeset
356 }
a61af66fc99e Initial load
duke
parents:
diff changeset
357
a61af66fc99e Initial load
duke
parents:
diff changeset
358 int AttachListener::pd_init() {
a61af66fc99e Initial load
duke
parents:
diff changeset
359 return Win32AttachListener::init();
a61af66fc99e Initial load
duke
parents:
diff changeset
360 }
a61af66fc99e Initial load
duke
parents:
diff changeset
361
a61af66fc99e Initial load
duke
parents:
diff changeset
362 // always startup on Windows NT/2000/XP
a61af66fc99e Initial load
duke
parents:
diff changeset
363 bool AttachListener::init_at_startup() {
a61af66fc99e Initial load
duke
parents:
diff changeset
364 return os::win32::is_nt();
a61af66fc99e Initial load
duke
parents:
diff changeset
365 }
a61af66fc99e Initial load
duke
parents:
diff changeset
366
a61af66fc99e Initial load
duke
parents:
diff changeset
367 // no trigger mechanism on Windows to start Attach Listener lazily
a61af66fc99e Initial load
duke
parents:
diff changeset
368 bool AttachListener::is_init_trigger() {
a61af66fc99e Initial load
duke
parents:
diff changeset
369 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
370 }
a61af66fc99e Initial load
duke
parents:
diff changeset
371
a61af66fc99e Initial load
duke
parents:
diff changeset
372 void AttachListener::abort() {
a61af66fc99e Initial load
duke
parents:
diff changeset
373 // nothing to do
a61af66fc99e Initial load
duke
parents:
diff changeset
374 }
a61af66fc99e Initial load
duke
parents:
diff changeset
375
a61af66fc99e Initial load
duke
parents:
diff changeset
376 void AttachListener::pd_data_dump() {
a61af66fc99e Initial load
duke
parents:
diff changeset
377 os::signal_notify(SIGBREAK);
a61af66fc99e Initial load
duke
parents:
diff changeset
378 }
a61af66fc99e Initial load
duke
parents:
diff changeset
379
a61af66fc99e Initial load
duke
parents:
diff changeset
380 AttachOperationFunctionInfo* AttachListener::pd_find_operation(const char* n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
381 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
382 }
a61af66fc99e Initial load
duke
parents:
diff changeset
383
a61af66fc99e Initial load
duke
parents:
diff changeset
384 jint AttachListener::pd_set_flag(AttachOperation* op, outputStream* out) {
a61af66fc99e Initial load
duke
parents:
diff changeset
385 out->print_cr("flag '%s' cannot be changed", op->arg(0));
a61af66fc99e Initial load
duke
parents:
diff changeset
386 return JNI_ERR;
a61af66fc99e Initial load
duke
parents:
diff changeset
387 }
a61af66fc99e Initial load
duke
parents:
diff changeset
388
a61af66fc99e Initial load
duke
parents:
diff changeset
389 void AttachListener::pd_detachall() {
a61af66fc99e Initial load
duke
parents:
diff changeset
390 // do nothing for now
a61af66fc99e Initial load
duke
parents:
diff changeset
391 }
a61af66fc99e Initial load
duke
parents:
diff changeset
392
a61af66fc99e Initial load
duke
parents:
diff changeset
393 // Native thread started by remote client executes this.
a61af66fc99e Initial load
duke
parents:
diff changeset
394 extern "C" {
a61af66fc99e Initial load
duke
parents:
diff changeset
395 JNIEXPORT jint JNICALL
a61af66fc99e Initial load
duke
parents:
diff changeset
396 JVM_EnqueueOperation(char* cmd, char* arg0, char* arg1, char* arg2, char* pipename) {
a61af66fc99e Initial load
duke
parents:
diff changeset
397 return (jint)Win32AttachListener::enqueue(cmd, arg0, arg1, arg2, pipename);
a61af66fc99e Initial load
duke
parents:
diff changeset
398 }
a61af66fc99e Initial load
duke
parents:
diff changeset
399
a61af66fc99e Initial load
duke
parents:
diff changeset
400 } // extern